asciidoctor-latexmath 1.0.0.beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/LICENSE +183 -0
- data/README.md +185 -0
- data/lib/asciidoctor-latexmath/renderer.rb +515 -0
- data/lib/asciidoctor-latexmath/treeprocessor.rb +369 -0
- data/lib/asciidoctor-latexmath/version.rb +11 -0
- data/lib/asciidoctor-latexmath.rb +12 -0
- metadata +116 -0
@@ -0,0 +1,369 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# SPDX-FileCopyrightText: 2025 Shuai Zhang
|
4
|
+
#
|
5
|
+
# SPDX-License-Identifier: LGPL-3.0-or-later WITH LGPL-3.0-linking-exception
|
6
|
+
|
7
|
+
require "asciidoctor/extensions"
|
8
|
+
require "fileutils"
|
9
|
+
require "pathname"
|
10
|
+
require "cgi"
|
11
|
+
require_relative "renderer"
|
12
|
+
|
13
|
+
module Asciidoctor
|
14
|
+
module Latexmath
|
15
|
+
class Treeprocessor < Asciidoctor::Extensions::Treeprocessor
|
16
|
+
LINE_FEED = %(
|
17
|
+
)
|
18
|
+
STEM_INLINE_MACRO_RX = /\\?(stem|latexmath):([a-z,]*)\[(.*?[^\\])\]/m
|
19
|
+
|
20
|
+
def process(document)
|
21
|
+
return unless needs_processing?(document)
|
22
|
+
|
23
|
+
renderer = Renderer.new(document)
|
24
|
+
image_output_dir, image_target_dir = image_output_and_target_dir(document)
|
25
|
+
context = {renderer: renderer, image_output_dir: image_output_dir, image_target_dir: image_target_dir}
|
26
|
+
|
27
|
+
(document.find_by(context: :stem, traverse_documents: true) || []).dup.each do |stem|
|
28
|
+
handle_stem_block(stem, context)
|
29
|
+
end
|
30
|
+
|
31
|
+
(document.find_by(traverse_documents: true) { |node| prose_candidate?(node) } || []).each do |prose|
|
32
|
+
handle_prose_block(prose, context)
|
33
|
+
end
|
34
|
+
|
35
|
+
(document.find_by(content: :section) || []).each do |section|
|
36
|
+
handle_section_title(section, context)
|
37
|
+
end
|
38
|
+
|
39
|
+
document.remove_attr "stem"
|
40
|
+
begin
|
41
|
+
(document.instance_variable_get :@header_attributes)&.delete("stem")
|
42
|
+
rescue
|
43
|
+
nil
|
44
|
+
end
|
45
|
+
|
46
|
+
nil
|
47
|
+
end
|
48
|
+
|
49
|
+
private
|
50
|
+
|
51
|
+
def needs_processing?(document)
|
52
|
+
stem_nodes = document.find_by(context: :stem, traverse_documents: true) || []
|
53
|
+
return true if stem_nodes.any? { |stem| latexmath_node?(stem) }
|
54
|
+
|
55
|
+
if (stem_attr = default_stem_style(document))
|
56
|
+
return true if stem_attr == "latexmath"
|
57
|
+
end
|
58
|
+
|
59
|
+
inline_candidates = document.find_by(traverse_documents: true) { |node| prose_candidate?(node) } || []
|
60
|
+
inline_candidates.any? { |node| contains_inline_stem?(node) }
|
61
|
+
end
|
62
|
+
|
63
|
+
def handle_stem_block(stem, context)
|
64
|
+
return unless latexmath_node?(stem)
|
65
|
+
|
66
|
+
content = extract_block_content(stem)
|
67
|
+
result = render_equation(
|
68
|
+
content,
|
69
|
+
display: true,
|
70
|
+
inline: false,
|
71
|
+
id: stem.id,
|
72
|
+
context: context,
|
73
|
+
asciidoc_source: stem.source,
|
74
|
+
source_location: stem.source_location
|
75
|
+
)
|
76
|
+
return unless result
|
77
|
+
|
78
|
+
parent = stem.parent
|
79
|
+
target, width, height = store_result(result, context)
|
80
|
+
return unless target
|
81
|
+
|
82
|
+
alt_text = stem.attr "alt", ((stem.context == :stem) ? stem.source : stem.content).to_s
|
83
|
+
attrs = {
|
84
|
+
"target" => target,
|
85
|
+
"alt" => alt_text,
|
86
|
+
"align" => "center"
|
87
|
+
}
|
88
|
+
if result.format == :png && width && height
|
89
|
+
attrs["width"] = width.to_i
|
90
|
+
attrs["height"] = height.to_i
|
91
|
+
end
|
92
|
+
|
93
|
+
replacement = create_image_block parent, attrs
|
94
|
+
replacement.id = stem.id if stem.id
|
95
|
+
if (title = stem.attributes["title"])
|
96
|
+
replacement.title = title
|
97
|
+
end
|
98
|
+
index = parent.blocks.index(stem)
|
99
|
+
parent.blocks[index] = replacement if index
|
100
|
+
rescue RenderingError => e
|
101
|
+
insert_block_error(stem, e)
|
102
|
+
end
|
103
|
+
|
104
|
+
def handle_prose_block(prose, context)
|
105
|
+
use_text_property = %i[list_item table_cell].include?(prose.context)
|
106
|
+
text = if use_text_property
|
107
|
+
prose.instance_variable_get(:@text)
|
108
|
+
else
|
109
|
+
(prose.lines || []) * LINE_FEED
|
110
|
+
end
|
111
|
+
|
112
|
+
updated_text, modified = inline_replace(text, prose, prose.document, context)
|
113
|
+
return unless modified
|
114
|
+
|
115
|
+
if use_text_property
|
116
|
+
prose.text = updated_text
|
117
|
+
else
|
118
|
+
prose.lines = updated_text.split(LINE_FEED)
|
119
|
+
end
|
120
|
+
end
|
121
|
+
|
122
|
+
def handle_section_title(section, context)
|
123
|
+
text = section.instance_variable_get(:@title)
|
124
|
+
updated_text, modified = inline_replace(text, section, section.document, context)
|
125
|
+
section.title = updated_text if modified
|
126
|
+
end
|
127
|
+
|
128
|
+
def latexmath_node?(node)
|
129
|
+
style = node.style
|
130
|
+
if style
|
131
|
+
style_name = style.to_s
|
132
|
+
return true if %w[latexmath tex].include?(style_name)
|
133
|
+
end
|
134
|
+
|
135
|
+
default_style = default_stem_style(node.document)
|
136
|
+
default_style == "latexmath"
|
137
|
+
end
|
138
|
+
|
139
|
+
def default_stem_style(document)
|
140
|
+
stem_attr = document.attr("stem")
|
141
|
+
return unless stem_attr
|
142
|
+
value = stem_attr.to_s.split(",").map(&:strip).find { |val| val == "latexmath" || val == "tex" }
|
143
|
+
(value == "tex") ? "latexmath" : value
|
144
|
+
end
|
145
|
+
|
146
|
+
def extract_block_content(stem)
|
147
|
+
case stem.context
|
148
|
+
when :stem
|
149
|
+
stem.content
|
150
|
+
else
|
151
|
+
stem.source
|
152
|
+
end
|
153
|
+
end
|
154
|
+
|
155
|
+
def render_equation(content, display:, inline:, context:, id: nil, asciidoc_source: nil, source_location: nil)
|
156
|
+
context[:renderer].render(
|
157
|
+
equation: normalize_equation(content),
|
158
|
+
display: display,
|
159
|
+
inline: inline,
|
160
|
+
id: id,
|
161
|
+
asciidoc_source: asciidoc_source,
|
162
|
+
source_location: source_location
|
163
|
+
)
|
164
|
+
end
|
165
|
+
|
166
|
+
def normalize_equation(content)
|
167
|
+
content.strip
|
168
|
+
end
|
169
|
+
|
170
|
+
def store_result(result, context)
|
171
|
+
return [nil, nil, nil] unless result.data
|
172
|
+
|
173
|
+
image_output_dir = context[:image_output_dir]
|
174
|
+
image_target_dir = context[:image_target_dir]
|
175
|
+
|
176
|
+
FileUtils.mkdir_p(image_output_dir) unless File.directory?(image_output_dir)
|
177
|
+
filename = "#{result.basename}.#{result.extension}"
|
178
|
+
output_path = ::File.join(image_output_dir, filename)
|
179
|
+
::File.binwrite(output_path, result.data)
|
180
|
+
|
181
|
+
target = if image_target_dir == "."
|
182
|
+
filename
|
183
|
+
else
|
184
|
+
::File.join(image_target_dir, filename)
|
185
|
+
end
|
186
|
+
|
187
|
+
[target, result.width, result.height]
|
188
|
+
end
|
189
|
+
|
190
|
+
def inline_replace(text, node, document, context)
|
191
|
+
return [text, false] unless text && !text.empty?
|
192
|
+
|
193
|
+
modified = false
|
194
|
+
default_style = default_stem_style(document) || "latexmath"
|
195
|
+
|
196
|
+
new_text = text.gsub(STEM_INLINE_MACRO_RX) do
|
197
|
+
match = Regexp.last_match
|
198
|
+
escaped = match[0].start_with?("\\")
|
199
|
+
if escaped
|
200
|
+
match[0][1..]
|
201
|
+
else
|
202
|
+
macro = match[1]
|
203
|
+
subs = match[2]
|
204
|
+
equation = match[3].rstrip
|
205
|
+
next "" if equation.empty?
|
206
|
+
|
207
|
+
style = (macro == "stem") ? default_style : "latexmath"
|
208
|
+
if style == "latexmath"
|
209
|
+
inline_subs = (subs.nil? || subs.empty?) ? [] : node.resolve_pass_subs(subs)
|
210
|
+
equation = node.apply_subs(equation, inline_subs) unless inline_subs.empty?
|
211
|
+
begin
|
212
|
+
result = render_equation(
|
213
|
+
equation,
|
214
|
+
display: false,
|
215
|
+
inline: true,
|
216
|
+
context: context,
|
217
|
+
asciidoc_source: match[0],
|
218
|
+
source_location: node.source_location
|
219
|
+
)
|
220
|
+
rescue RenderingError => e
|
221
|
+
location = node.source_location
|
222
|
+
log_rendering_error(e, location)
|
223
|
+
ensure_macros_substitution(node)
|
224
|
+
modified = true
|
225
|
+
next inline_error_markup(e, location)
|
226
|
+
end
|
227
|
+
next match[0] unless result
|
228
|
+
|
229
|
+
modified = true
|
230
|
+
|
231
|
+
if result.inline_markup
|
232
|
+
ensure_macros_substitution(node)
|
233
|
+
%(pass:[#{result.inline_markup}])
|
234
|
+
else
|
235
|
+
target, width, height = store_result(result, context)
|
236
|
+
next match[0] unless target
|
237
|
+
attrs = []
|
238
|
+
attrs << %(width=#{width.to_i}) if width
|
239
|
+
attrs << %(height=#{height.to_i}) if height
|
240
|
+
attr_text = attrs.join(",")
|
241
|
+
ensure_macros_substitution(node)
|
242
|
+
%(image:#{target}[#{attr_text}])
|
243
|
+
end
|
244
|
+
else
|
245
|
+
match[0]
|
246
|
+
end
|
247
|
+
end
|
248
|
+
end
|
249
|
+
|
250
|
+
[new_text, modified]
|
251
|
+
end
|
252
|
+
|
253
|
+
def prose_candidate?(node)
|
254
|
+
(node.content_model == :simple && node.subs.include?(:macros)) || %i[list_item table_cell].include?(node.context)
|
255
|
+
end
|
256
|
+
|
257
|
+
def contains_inline_stem?(node)
|
258
|
+
text = if node.context == :list_item || node.context == :table_cell
|
259
|
+
node.instance_variable_get(:@text)
|
260
|
+
else
|
261
|
+
(node.lines || []) * LINE_FEED
|
262
|
+
end
|
263
|
+
text && text =~ STEM_INLINE_MACRO_RX
|
264
|
+
end
|
265
|
+
|
266
|
+
def ensure_macros_substitution(node)
|
267
|
+
return unless node.respond_to?(:subs)
|
268
|
+
|
269
|
+
subs = node.subs
|
270
|
+
if subs.nil?
|
271
|
+
node.instance_variable_set(:@subs, [:macros])
|
272
|
+
elsif subs.include?(:macros)
|
273
|
+
# already enabled
|
274
|
+
else
|
275
|
+
updated = subs.dup
|
276
|
+
updated << :macros
|
277
|
+
node.instance_variable_set(:@subs, updated)
|
278
|
+
end
|
279
|
+
end
|
280
|
+
|
281
|
+
def image_output_and_target_dir(doc)
|
282
|
+
output_dir = doc.attr("imagesoutdir")
|
283
|
+
if output_dir
|
284
|
+
if doc.attr("imagesdir").nil_or_empty?
|
285
|
+
target_dir = output_dir
|
286
|
+
else
|
287
|
+
abs_imagesdir = ::Pathname.new doc.normalize_system_path(doc.attr("imagesdir"))
|
288
|
+
abs_outdir = ::Pathname.new doc.normalize_system_path(output_dir)
|
289
|
+
target_dir = abs_outdir.relative_path_from(abs_imagesdir).to_s
|
290
|
+
end
|
291
|
+
else
|
292
|
+
output_dir = doc.attr("imagesdir") || "."
|
293
|
+
target_dir = "."
|
294
|
+
end
|
295
|
+
|
296
|
+
output_dir = doc.normalize_system_path(output_dir, doc.attr("docdir"))
|
297
|
+
[output_dir, target_dir]
|
298
|
+
end
|
299
|
+
|
300
|
+
def insert_block_error(stem, error)
|
301
|
+
log_rendering_error(error, stem.source_location)
|
302
|
+
parent = stem.parent
|
303
|
+
return unless parent
|
304
|
+
|
305
|
+
text = block_error_text(error, stem.source_location)
|
306
|
+
replacement = Asciidoctor::Block.new(parent, :listing, source: text)
|
307
|
+
replacement.add_role("latexmath-error") if replacement.respond_to?(:add_role)
|
308
|
+
replacement.id = stem.id if stem.id
|
309
|
+
if (title = stem.attributes["title"])
|
310
|
+
replacement.title = title
|
311
|
+
end
|
312
|
+
|
313
|
+
index = parent.blocks.index(stem)
|
314
|
+
if index
|
315
|
+
parent.blocks[index] = replacement
|
316
|
+
else
|
317
|
+
parent.blocks << replacement
|
318
|
+
end
|
319
|
+
end
|
320
|
+
|
321
|
+
def block_error_text(error, source_location)
|
322
|
+
location_hint = format_error_location(source_location)
|
323
|
+
header = if location_hint.empty?
|
324
|
+
"Failed to render latexmath"
|
325
|
+
else
|
326
|
+
"Failed to render latexmath#{location_hint}"
|
327
|
+
end
|
328
|
+
header = "#{header}:"
|
329
|
+
message = error.message.to_s.rstrip
|
330
|
+
message.empty? ? header : "#{header}\n#{message}"
|
331
|
+
end
|
332
|
+
|
333
|
+
def inline_error_markup(error, source_location = nil)
|
334
|
+
location_hint = format_error_location(source_location)
|
335
|
+
prefix = "Failed to render latexmath"
|
336
|
+
prefix += location_hint unless location_hint.empty?
|
337
|
+
message = "#{prefix}: #{error.message}".strip
|
338
|
+
escaped = CGI.escapeHTML(message)
|
339
|
+
escaped = escaped.gsub(/\r?\n/, "<br>")
|
340
|
+
"+++<span class=\"latexmath-error\"><code>#{escaped}</code></span>+++"
|
341
|
+
end
|
342
|
+
|
343
|
+
def log_rendering_error(error, source_location)
|
344
|
+
location_hint = format_error_location(source_location)
|
345
|
+
first_line = error.message.to_s.lines.first&.strip || error.message.to_s
|
346
|
+
warn %(asciidoctor-latexmath: #{first_line}#{location_hint})
|
347
|
+
end
|
348
|
+
|
349
|
+
def format_error_location(source_location)
|
350
|
+
return "" unless source_location
|
351
|
+
|
352
|
+
file = if source_location.respond_to?(:file)
|
353
|
+
source_location.file
|
354
|
+
elsif source_location.respond_to?(:path)
|
355
|
+
source_location.path
|
356
|
+
elsif source_location.respond_to?(:filename)
|
357
|
+
source_location.filename
|
358
|
+
end
|
359
|
+
|
360
|
+
line = source_location.respond_to?(:lineno) ? source_location.lineno : nil
|
361
|
+
|
362
|
+
parts = []
|
363
|
+
parts << file if file && !file.to_s.empty?
|
364
|
+
parts << line if line
|
365
|
+
parts.empty? ? "" : " (#{parts.join(':')})"
|
366
|
+
end
|
367
|
+
end
|
368
|
+
end
|
369
|
+
end
|
@@ -0,0 +1,12 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# SPDX-FileCopyrightText: 2025 Shuai Zhang
|
4
|
+
#
|
5
|
+
# SPDX-License-Identifier: LGPL-3.0-or-later WITH LGPL-3.0-linking-exception
|
6
|
+
|
7
|
+
require_relative "asciidoctor-latexmath/version"
|
8
|
+
require_relative "asciidoctor-latexmath/treeprocessor"
|
9
|
+
|
10
|
+
Asciidoctor::Extensions.register do
|
11
|
+
treeprocessor Asciidoctor::Latexmath::Treeprocessor
|
12
|
+
end
|
metadata
ADDED
@@ -0,0 +1,116 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: asciidoctor-latexmath
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 1.0.0.beta.1
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Shuai Zhang
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
date: 2025-09-28 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: asciidoctor
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - ">="
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '2.0'
|
20
|
+
- - "<"
|
21
|
+
- !ruby/object:Gem::Version
|
22
|
+
version: '3.0'
|
23
|
+
type: :runtime
|
24
|
+
prerelease: false
|
25
|
+
version_requirements: !ruby/object:Gem::Requirement
|
26
|
+
requirements:
|
27
|
+
- - ">="
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: '2.0'
|
30
|
+
- - "<"
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '3.0'
|
33
|
+
- !ruby/object:Gem::Dependency
|
34
|
+
name: bundler
|
35
|
+
requirement: !ruby/object:Gem::Requirement
|
36
|
+
requirements:
|
37
|
+
- - "~>"
|
38
|
+
- !ruby/object:Gem::Version
|
39
|
+
version: '2.4'
|
40
|
+
type: :development
|
41
|
+
prerelease: false
|
42
|
+
version_requirements: !ruby/object:Gem::Requirement
|
43
|
+
requirements:
|
44
|
+
- - "~>"
|
45
|
+
- !ruby/object:Gem::Version
|
46
|
+
version: '2.4'
|
47
|
+
- !ruby/object:Gem::Dependency
|
48
|
+
name: rake
|
49
|
+
requirement: !ruby/object:Gem::Requirement
|
50
|
+
requirements:
|
51
|
+
- - "~>"
|
52
|
+
- !ruby/object:Gem::Version
|
53
|
+
version: '13.0'
|
54
|
+
type: :development
|
55
|
+
prerelease: false
|
56
|
+
version_requirements: !ruby/object:Gem::Requirement
|
57
|
+
requirements:
|
58
|
+
- - "~>"
|
59
|
+
- !ruby/object:Gem::Version
|
60
|
+
version: '13.0'
|
61
|
+
- !ruby/object:Gem::Dependency
|
62
|
+
name: rspec
|
63
|
+
requirement: !ruby/object:Gem::Requirement
|
64
|
+
requirements:
|
65
|
+
- - "~>"
|
66
|
+
- !ruby/object:Gem::Version
|
67
|
+
version: '3.13'
|
68
|
+
type: :development
|
69
|
+
prerelease: false
|
70
|
+
version_requirements: !ruby/object:Gem::Requirement
|
71
|
+
requirements:
|
72
|
+
- - "~>"
|
73
|
+
- !ruby/object:Gem::Version
|
74
|
+
version: '3.13'
|
75
|
+
description: Render latexmath blocks and inline macros to PDF/SVG/PNG assets using
|
76
|
+
your local LaTeX toolchain.
|
77
|
+
email:
|
78
|
+
- zhangshuai.ustc@gmail.com
|
79
|
+
executables: []
|
80
|
+
extensions: []
|
81
|
+
extra_rdoc_files: []
|
82
|
+
files:
|
83
|
+
- LICENSE
|
84
|
+
- README.md
|
85
|
+
- lib/asciidoctor-latexmath.rb
|
86
|
+
- lib/asciidoctor-latexmath/renderer.rb
|
87
|
+
- lib/asciidoctor-latexmath/treeprocessor.rb
|
88
|
+
- lib/asciidoctor-latexmath/version.rb
|
89
|
+
homepage: https://github.com/hcoona/asciidoctor-latexmath
|
90
|
+
licenses:
|
91
|
+
- LGPL-3.0-or-later WITH LGPL-3.0-linking-exception
|
92
|
+
metadata:
|
93
|
+
homepage_uri: https://github.com/hcoona/asciidoctor-latexmath
|
94
|
+
source_code_uri: https://github.com/hcoona/asciidoctor-latexmath
|
95
|
+
bug_tracker_uri: https://github.com/hcoona/asciidoctor-latexmath/issues
|
96
|
+
documentation_uri: https://github.com/hcoona/asciidoctor-latexmath#readme
|
97
|
+
post_install_message:
|
98
|
+
rdoc_options: []
|
99
|
+
require_paths:
|
100
|
+
- lib
|
101
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
102
|
+
requirements:
|
103
|
+
- - ">="
|
104
|
+
- !ruby/object:Gem::Version
|
105
|
+
version: '0'
|
106
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
107
|
+
requirements:
|
108
|
+
- - ">"
|
109
|
+
- !ruby/object:Gem::Version
|
110
|
+
version: 1.3.1
|
111
|
+
requirements: []
|
112
|
+
rubygems_version: 3.4.20
|
113
|
+
signing_key:
|
114
|
+
specification_version: 4
|
115
|
+
summary: Offline latexmath rendering for Asciidoctor.
|
116
|
+
test_files: []
|