review 4.0.0 → 5.1.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (268) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/ruby-tex.yml +31 -0
  3. data/.github/workflows/ruby-win.yml +45 -0
  4. data/.github/workflows/ruby.yml +27 -0
  5. data/.rubocop.yml +141 -38
  6. data/Dockerfile +21 -5
  7. data/NEWS.ja.md +338 -1
  8. data/NEWS.md +339 -2
  9. data/README.md +11 -7
  10. data/Rakefile +7 -2
  11. data/appveyor.yml +0 -20
  12. data/bin/review +2 -4
  13. data/bin/review-catalog-converter +4 -4
  14. data/bin/review-check +8 -12
  15. data/bin/review-checkdep +2 -5
  16. data/bin/review-compile +11 -20
  17. data/bin/review-epub2html +1 -4
  18. data/bin/review-epubmaker +3 -4
  19. data/bin/review-idgxmlmaker +1 -3
  20. data/bin/review-index +5 -86
  21. data/bin/review-init +1 -4
  22. data/bin/review-pdfmaker +1 -3
  23. data/bin/review-preproc +3 -5
  24. data/bin/review-textmaker +1 -3
  25. data/bin/review-update +1 -4
  26. data/bin/review-validate +5 -5
  27. data/bin/review-vol +5 -82
  28. data/bin/review-webmaker +1 -3
  29. data/doc/config.yml.sample +55 -12
  30. data/doc/config.yml.sample-simple +4 -3
  31. data/doc/format.ja.md +119 -17
  32. data/doc/format.md +106 -17
  33. data/doc/makeindex.ja.md +2 -2
  34. data/doc/pdfmaker.ja.md +42 -0
  35. data/doc/pdfmaker.md +41 -0
  36. data/doc/quickstart.ja.md +55 -25
  37. data/doc/quickstart.md +47 -17
  38. data/lib/review.rb +1 -1
  39. data/lib/review/book.rb +2 -2
  40. data/lib/review/book/base.rb +69 -30
  41. data/lib/review/book/bib.rb +21 -0
  42. data/lib/review/book/book_unit.rb +158 -0
  43. data/lib/review/book/chapter.rb +33 -26
  44. data/lib/review/book/index.rb +39 -206
  45. data/lib/review/book/index/item.rb +8 -2
  46. data/lib/review/book/page_metric.rb +7 -7
  47. data/lib/review/book/part.rb +26 -11
  48. data/lib/review/book/volume.rb +5 -5
  49. data/lib/review/builder.rb +124 -27
  50. data/lib/review/call_hook.rb +20 -0
  51. data/lib/review/catalog.rb +3 -2
  52. data/lib/review/compiler.rb +197 -91
  53. data/lib/review/configure.rb +95 -9
  54. data/lib/review/converter.rb +1 -1
  55. data/lib/review/epub2html.rb +6 -1
  56. data/lib/review/epubmaker.rb +116 -117
  57. data/lib/review/epubmaker/content.rb +113 -0
  58. data/lib/review/epubmaker/epubcommon.rb +372 -0
  59. data/lib/review/epubmaker/epubv2.rb +178 -0
  60. data/lib/review/epubmaker/epubv3.rb +231 -0
  61. data/lib/review/epubmaker/producer.rb +168 -0
  62. data/lib/review/epubmaker/reviewheaderlistener.rb +12 -2
  63. data/lib/review/epubmaker/zip_exporter.rb +84 -0
  64. data/lib/review/exception.rb +6 -0
  65. data/lib/review/extentions/string.rb +0 -4
  66. data/lib/review/htmlbuilder.rb +132 -87
  67. data/lib/review/htmlutils.rb +8 -11
  68. data/lib/review/i18n.rb +2 -1
  69. data/lib/review/idgxmlbuilder.rb +156 -76
  70. data/lib/review/idgxmlmaker.rb +22 -18
  71. data/lib/review/img_math.rb +245 -0
  72. data/lib/review/index_builder.rb +654 -0
  73. data/lib/review/init.rb +17 -18
  74. data/lib/review/latexbox.rb +58 -0
  75. data/lib/review/latexbuilder.rb +173 -43
  76. data/lib/review/latexutils.rb +9 -1
  77. data/lib/review/lineinput.rb +112 -2
  78. data/lib/review/logger.rb +42 -3
  79. data/lib/review/makerhelper.rb +12 -187
  80. data/lib/review/markdownbuilder.rb +51 -2
  81. data/lib/review/pdfmaker.rb +78 -73
  82. data/lib/review/plaintextbuilder.rb +98 -14
  83. data/lib/review/preprocessor.rb +18 -12
  84. data/lib/review/rstbuilder.rb +28 -9
  85. data/lib/review/sec_counter.rb +14 -0
  86. data/lib/review/template.rb +6 -0
  87. data/lib/review/textmaker.rb +21 -19
  88. data/lib/review/textutils.rb +3 -12
  89. data/lib/review/tocprinter.rb +242 -97
  90. data/lib/review/topbuilder.rb +91 -24
  91. data/lib/review/update.rb +12 -13
  92. data/lib/review/version.rb +1 -1
  93. data/lib/review/volumeprinter.rb +97 -0
  94. data/lib/review/webmaker.rb +34 -33
  95. data/lib/review/webtocprinter.rb +39 -35
  96. data/lib/review/yamlloader.rb +2 -1
  97. data/review.gemspec +5 -3
  98. data/samples/sample-book/src/.gitignore +1 -0
  99. data/samples/sample-book/src/config-ebook.yml +4 -0
  100. data/samples/sample-book/src/config-epub2.yml +1 -1
  101. data/samples/sample-book/src/config-jlreq-ebook.yml +4 -0
  102. data/samples/sample-book/src/config.yml +3 -3
  103. data/samples/sample-book/src/lib/tasks/review.rake +26 -6
  104. data/samples/syntax-book/Gemfile +1 -1
  105. data/samples/syntax-book/ch01.re +3 -1
  106. data/samples/syntax-book/ch02.re +28 -21
  107. data/samples/syntax-book/ch03.re +1 -1
  108. data/samples/syntax-book/config-jlreq-lualatex.yml +4 -0
  109. data/samples/syntax-book/config-print.yml +3 -0
  110. data/samples/syntax-book/config.yml +1 -1
  111. data/samples/syntax-book/images/img3-2.png +0 -0
  112. data/samples/syntax-book/lib/tasks/review.rake +23 -8
  113. data/templates/html/_colophon.html.erb +23 -0
  114. data/templates/html/_colophon_history.html.erb +9 -0
  115. data/templates/html/_cover.html.erb +10 -0
  116. data/templates/html/_part_body.html.erb +6 -0
  117. data/templates/html/_titlepage.html.erb +20 -0
  118. data/templates/html/layout-html5.html.erb +6 -0
  119. data/templates/html/layout-xhtml1.html.erb +6 -0
  120. data/templates/latex/config.erb +41 -29
  121. data/templates/latex/review-jlreq/README.md +3 -1
  122. data/templates/latex/review-jlreq/review-base.sty +47 -22
  123. data/templates/latex/review-jlreq/review-jlreq.cls +25 -24
  124. data/templates/latex/review-jlreq/review-style.sty +6 -1
  125. data/templates/latex/review-jlreq/review-tcbox.sty +348 -0
  126. data/templates/latex/review-jlreq/reviewmacro.sty +5 -0
  127. data/templates/latex/review-jsbook/README.md +7 -5
  128. data/templates/latex/review-jsbook/review-base.sty +47 -21
  129. data/templates/latex/review-jsbook/review-jsbook.cls +12 -2
  130. data/templates/latex/review-jsbook/review-style.sty +6 -1
  131. data/templates/latex/review-jsbook/review-tcbox.sty +348 -0
  132. data/templates/latex/review-jsbook/reviewmacro.sty +5 -0
  133. data/templates/opf/epubv2.opf.erb +7 -7
  134. data/templates/opf/epubv3.opf.erb +7 -7
  135. data/templates/opf/opf_manifest_epubv2.opf.erb +10 -0
  136. data/templates/opf/opf_manifest_epubv3.opf.erb +10 -0
  137. data/templates/opf/opf_metainfo_epubv2.opf.erb +17 -0
  138. data/templates/opf/opf_metainfo_epubv3.opf.erb +49 -0
  139. data/templates/opf/opf_tocx_epubv2.opf.erb +9 -0
  140. data/templates/opf/opf_tocx_epubv3.opf.erb +17 -0
  141. data/templates/web/html/layout-html5.html.erb +9 -8
  142. data/templates/web/html/layout-xhtml1.html.erb +6 -0
  143. data/test/assets/header_listener.html +35 -0
  144. data/test/assets/img_math/img1.png +0 -0
  145. data/test/assets/img_math/img2.png +0 -0
  146. data/test/assets/img_math/img3.png +0 -0
  147. data/test/assets/syntax_book_index_detail.txt +58 -0
  148. data/test/assets/test_template.tex +12 -9
  149. data/test/assets/test_template_backmatter.tex +12 -9
  150. data/test/book_test_helper.rb +11 -5
  151. data/test/run_test.rb +1 -1
  152. data/test/test_book.rb +62 -63
  153. data/test/test_book_chapter.rb +99 -56
  154. data/test/test_book_part.rb +3 -3
  155. data/test/test_builder.rb +24 -15
  156. data/test/test_catalog.rb +1 -0
  157. data/test/test_catalog_converter_cmd.rb +1 -1
  158. data/test/test_converter.rb +1 -0
  159. data/test/test_epub3maker.rb +170 -126
  160. data/test/test_epubmaker.rb +249 -129
  161. data/test/test_epubmaker_cmd.rb +14 -7
  162. data/test/test_helper.rb +23 -11
  163. data/test/test_htmlbuilder.rb +956 -76
  164. data/test/test_htmlutils.rb +0 -12
  165. data/test/test_i18n.rb +33 -33
  166. data/test/test_idgxmlbuilder.rb +568 -10
  167. data/test/test_idgxmlmaker_cmd.rb +50 -0
  168. data/test/test_image_finder.rb +52 -70
  169. data/test/test_img_math.rb +111 -0
  170. data/test/test_index.rb +62 -52
  171. data/test/test_indexbuilder.rb +52 -0
  172. data/test/test_latexbuilder.rb +1056 -30
  173. data/test/test_latexbuilder_v2.rb +52 -12
  174. data/test/test_lineinput.rb +20 -93
  175. data/test/test_logger.rb +7 -7
  176. data/test/test_makerhelper.rb +0 -12
  177. data/test/test_markdownbuilder.rb +32 -0
  178. data/test/test_pdfmaker.rb +100 -11
  179. data/test/test_pdfmaker_cmd.rb +101 -7
  180. data/test/test_plaintextbuilder.rb +531 -25
  181. data/test/test_review_ext.rb +2 -1
  182. data/test/test_reviewheaderlistener.rb +49 -0
  183. data/test/test_rstbuilder.rb +25 -1
  184. data/test/test_sec_counter.rb +156 -0
  185. data/test/test_template.rb +12 -2
  186. data/test/test_textmaker_cmd.rb +58 -0
  187. data/test/test_tocprinter.rb +46 -0
  188. data/test/test_topbuilder.rb +365 -10
  189. data/test/test_update.rb +44 -44
  190. data/test/test_webtocprinter.rb +75 -43
  191. data/test/test_zip_exporter.rb +5 -6
  192. data/vendor/gentombow/LICENSE +1 -1
  193. data/vendor/gentombow/Makefile +0 -1
  194. data/vendor/gentombow/bounddvi-en.pdf +0 -0
  195. data/vendor/gentombow/bounddvi-en.tex +1 -0
  196. data/vendor/gentombow/bounddvi.pdf +0 -0
  197. data/vendor/gentombow/bounddvi.sty +30 -7
  198. data/vendor/gentombow/bounddvi.tex +1 -0
  199. data/vendor/gentombow/create_archive.sh +1 -0
  200. data/vendor/gentombow/gentombow-ja.pdf +0 -0
  201. data/vendor/gentombow/gentombow-ja.tex +9 -0
  202. data/vendor/gentombow/gentombow.pdf +0 -0
  203. data/vendor/gentombow/gentombow.sty +32 -10
  204. data/vendor/gentombow/gentombow.tex +8 -0
  205. data/vendor/gentombow/tests/gentombow-01-pdfx.tex +8 -0
  206. data/vendor/gentombow/tests/gentombow-02-pdfx.tex +8 -0
  207. data/vendor/jsclasses/LICENSE +1 -1
  208. data/vendor/jsclasses/Makefile +3 -2
  209. data/vendor/jsclasses/create_archive.sh +5 -5
  210. data/vendor/jsclasses/jis/Makefile +3 -2
  211. data/vendor/jsclasses/jis/jsarticle.cls +74 -31
  212. data/vendor/jsclasses/jis/jsbook.cls +74 -31
  213. data/vendor/jsclasses/jis/jsclasses.dtx +176 -36
  214. data/vendor/jsclasses/jis/jsclasses.ins +15 -5
  215. data/vendor/jsclasses/jis/jslogo.dtx +4 -4
  216. data/vendor/jsclasses/jis/jslogo.ins +9 -0
  217. data/vendor/jsclasses/jis/jslogo.sty +4 -16
  218. data/vendor/jsclasses/jis/jspf.cls +73 -30
  219. data/vendor/jsclasses/jis/jsreport.cls +74 -31
  220. data/vendor/jsclasses/jis/jsverb.ins +9 -0
  221. data/vendor/jsclasses/jis/jsverb.sty +1 -13
  222. data/vendor/jsclasses/jis/kiyou.cls +74 -31
  223. data/vendor/jsclasses/jis/minijs.sty +65 -22
  224. data/vendor/jsclasses/jis/okumacro.dtx +4 -5
  225. data/vendor/jsclasses/jis/okumacro.ins +9 -0
  226. data/vendor/jsclasses/jis/okumacro.sty +4 -17
  227. data/vendor/jsclasses/jis/okuverb.ins +9 -0
  228. data/vendor/jsclasses/jis/okuverb.sty +1 -13
  229. data/vendor/jsclasses/jis/winjis.sty +23 -19
  230. data/vendor/jsclasses/jsarticle.cls +74 -31
  231. data/vendor/jsclasses/jsbook.cls +74 -31
  232. data/vendor/jsclasses/jsclasses.dtx +176 -36
  233. data/vendor/jsclasses/jsclasses.ins +15 -5
  234. data/vendor/jsclasses/jsclasses.pdf +0 -0
  235. data/vendor/jsclasses/jslogo.dtx +4 -4
  236. data/vendor/jsclasses/jslogo.ins +9 -0
  237. data/vendor/jsclasses/jslogo.pdf +0 -0
  238. data/vendor/jsclasses/jslogo.sty +4 -16
  239. data/vendor/jsclasses/jspf.cls +73 -30
  240. data/vendor/jsclasses/jsreport.cls +74 -31
  241. data/vendor/jsclasses/jsverb.ins +9 -0
  242. data/vendor/jsclasses/jsverb.pdf +0 -0
  243. data/vendor/jsclasses/jsverb.sty +1 -13
  244. data/vendor/jsclasses/kiyou.cls +74 -31
  245. data/vendor/jsclasses/minijs.sty +68 -22
  246. data/vendor/jsclasses/okumacro.dtx +4 -5
  247. data/vendor/jsclasses/okumacro.ins +9 -0
  248. data/vendor/jsclasses/okumacro.pdf +0 -0
  249. data/vendor/jsclasses/okumacro.sty +4 -17
  250. data/vendor/jsclasses/okuverb.ins +9 -0
  251. data/vendor/jsclasses/okuverb.pdf +0 -0
  252. data/vendor/jsclasses/okuverb.sty +1 -13
  253. data/vendor/jsclasses/tests/relfont.tex +10 -0
  254. data/vendor/jsclasses/winjis.sty +23 -19
  255. metadata +109 -20
  256. data/.rubocop_todo.yml +0 -7
  257. data/lib/epubmaker.rb +0 -23
  258. data/lib/epubmaker/content.rb +0 -110
  259. data/lib/epubmaker/epubcommon.rb +0 -441
  260. data/lib/epubmaker/epubv2.rb +0 -143
  261. data/lib/epubmaker/epubv3.rb +0 -233
  262. data/lib/epubmaker/producer.rb +0 -375
  263. data/lib/epubmaker/zip_exporter.rb +0 -81
  264. data/lib/lineinput.rb +0 -155
  265. data/lib/review/book/compilable.rb +0 -178
  266. data/lib/review/tocparser.rb +0 -271
  267. data/samples/syntax-book/review-ext.rb +0 -14
  268. data/test/test_tocparser.rb +0 -25
@@ -1,4 +1,4 @@
1
- # Copyright (c) 2010-2019 Kenshi Muto and Masayoshi Takahashi
1
+ # Copyright (c) 2010-2021 Kenshi Muto and Masayoshi Takahashi
2
2
  #
3
3
  # This program is free software.
4
4
  # You can distribute or modify this program under the terms of
@@ -22,11 +22,14 @@ require 'review/yamlloader'
22
22
  require 'review/version'
23
23
  require 'review/makerhelper'
24
24
  require 'review/template'
25
+ require 'review/latexbox'
26
+ require 'review/call_hook'
25
27
 
26
28
  module ReVIEW
27
29
  class PDFMaker
28
30
  include FileUtils
29
31
  include ReVIEW::LaTeXUtils
32
+ include ReVIEW::CallHook
30
33
 
31
34
  attr_accessor :config, :basedir
32
35
 
@@ -100,6 +103,7 @@ module ReVIEW
100
103
  def parse_opts(args)
101
104
  cmd_config = {}
102
105
  opts = OptionParser.new
106
+ @buildonly = nil
103
107
 
104
108
  opts.banner = 'Usage: review-pdfmaker configfile'
105
109
  opts.version = ReVIEW::VERSION
@@ -109,6 +113,7 @@ module ReVIEW
109
113
  end
110
114
  opts.on('--[no-]debug', 'Keep temporary files.') { |debug| cmd_config['debug'] = debug }
111
115
  opts.on('--ignore-errors', 'Ignore review-compile errors.') { cmd_config['ignore-errors'] = true }
116
+ opts.on('-y', '--only file1,file2,...', 'Build only specified files.') { |v| @buildonly = v.split(/\s*,\s*/).map { |m| m.strip.sub(/\.re\Z/, '') } }
112
117
 
113
118
  opts.parse!(args)
114
119
  if args.size != 1
@@ -120,20 +125,13 @@ module ReVIEW
120
125
  end
121
126
 
122
127
  def execute(*args)
123
- @config = ReVIEW::Configure.values
124
- @config.maker = 'pdfmaker'
125
128
  cmd_config, yamlfile = parse_opts(args)
126
129
  error "#{yamlfile} not found." unless File.exist?(yamlfile)
127
130
 
128
- begin
129
- loader = ReVIEW::YAMLLoader.new
130
- @config.deep_merge!(loader.load_file(yamlfile))
131
- rescue => e
132
- error "yaml error #{e.message}"
133
- end
131
+ @config = ReVIEW::Configure.create(maker: 'pdfmaker',
132
+ yamlfile: yamlfile,
133
+ config: cmd_config)
134
134
 
135
- # YAML configs will be overridden by command line options.
136
- @config.deep_merge!(cmd_config)
137
135
  I18n.setup(@config['language'])
138
136
  @basedir = File.absolute_path(File.dirname(yamlfile))
139
137
 
@@ -156,6 +154,7 @@ module ReVIEW
156
154
  generate_pdf
157
155
  rescue ApplicationError => e
158
156
  raise if @config['debug']
157
+
159
158
  error(e.message)
160
159
  end
161
160
  end
@@ -166,8 +165,13 @@ module ReVIEW
166
165
  if part.name.present?
167
166
  @config['use_part'] = true
168
167
  if part.file?
169
- output_chaps(part.name)
170
- input_files['CHAPS'] << %Q(\\input{#{part.name}.tex}\n)
168
+ if @buildonly && !@buildonly.include?(part.name)
169
+ warn "skip #{part.name}.re"
170
+ input_files['CHAPS'] << %Q(\\part{}\n)
171
+ else
172
+ output_chaps(part.name)
173
+ input_files['CHAPS'] << %Q(\\input{#{part.name}.tex}\n)
174
+ end
171
175
  else
172
176
  input_files['CHAPS'] << %Q(\\part{#{part.name}}\n)
173
177
  end
@@ -175,11 +179,18 @@ module ReVIEW
175
179
 
176
180
  part.chapters.each do |chap|
177
181
  filename = File.basename(chap.path, '.*')
178
- output_chaps(filename)
179
- input_files['PREDEF'] << "\\input{#{filename}.tex}\n" if chap.on_predef?
180
- input_files['CHAPS'] << "\\input{#{filename}.tex}\n" if chap.on_chaps?
181
- input_files['APPENDIX'] << "\\input{#{filename}.tex}\n" if chap.on_appendix?
182
- input_files['POSTDEF'] << "\\input{#{filename}.tex}\n" if chap.on_postdef?
182
+ entry = "\\input{#{filename}.tex}\n"
183
+ if @buildonly && !@buildonly.include?(filename)
184
+ warn "skip #{filename}.re"
185
+ entry = "\\chapter{}\n"
186
+ else
187
+ output_chaps(filename)
188
+ end
189
+
190
+ input_files['PREDEF'] << entry if chap.on_predef?
191
+ input_files['CHAPS'] << entry if chap.on_chaps?
192
+ input_files['APPENDIX'] << entry if chap.on_appendix?
193
+ input_files['POSTDEF'] << entry if chap.on_postdef?
183
194
  end
184
195
  end
185
196
 
@@ -191,7 +202,7 @@ module ReVIEW
191
202
  Dir.chdir(@path) do
192
203
  File.open("./#{@mastertex}.tex", 'wb') { |f| f.write template }
193
204
 
194
- call_hook('hook_beforetexcompile')
205
+ call_hook('hook_beforetexcompile', Dir.pwd, @basedir, base_dir: @basedir)
195
206
 
196
207
  ## do compile
197
208
  if ENV['REVIEW_SAFE_MODE'].to_i & 4 > 0
@@ -234,32 +245,32 @@ module ReVIEW
234
245
  system_or_raise(*[texcommand, texoptions, "#{@mastertex}.tex"].flatten.compact)
235
246
  end
236
247
 
237
- call_hook('hook_beforemakeindex')
238
- if @config['pdfmaker']['makeindex'] && File.exist?("#{@mastertex}.idx")
248
+ call_hook('hook_beforemakeindex', Dir.pwd, @basedir, base_dir: @basedir)
249
+ if @config['pdfmaker']['makeindex'] && File.size?("#{@mastertex}.idx")
239
250
  system_or_raise(*[makeindex_command, makeindex_options, @mastertex].flatten.compact)
251
+ call_hook('hook_aftermakeindex', Dir.pwd, @basedir, base_dir: @basedir)
252
+ system_or_raise(*[texcommand, texoptions, "#{@mastertex}.tex"].flatten.compact)
240
253
  end
241
- call_hook('hook_aftermakeindex')
242
254
 
243
255
  system_or_raise(*[texcommand, texoptions, "#{@mastertex}.tex"].flatten.compact)
244
- call_hook('hook_aftertexcompile')
256
+ call_hook('hook_aftertexcompile', Dir.pwd, @basedir, base_dir: @basedir)
245
257
 
246
258
  if File.exist?("#{@mastertex}.dvi") && dvicommand.present?
247
259
  system_or_raise(*[dvicommand, dvioptions, "#{@mastertex}.dvi"].flatten.compact)
248
- call_hook('hook_afterdvipdf')
260
+ call_hook('hook_afterdvipdf', Dir.pwd, @basedir, base_dir: @basedir)
249
261
  end
250
262
  end
251
263
  end
252
264
 
253
265
  def generate_pdf
254
266
  remove_old_file
255
- erb_config
256
267
  @path = build_path
257
268
  begin
258
269
  @compile_errors = nil
259
270
 
260
- book = ReVIEW::Book.load(@basedir)
261
- book.config = @config
271
+ book = ReVIEW::Book::Base.new(@basedir, config: @config)
262
272
  @converter = ReVIEW::Converter.new(book, ReVIEW::LATEXBuilder.new)
273
+ erb_config
263
274
 
264
275
  @input_files = make_input_files(book)
265
276
 
@@ -280,6 +291,7 @@ module ReVIEW
280
291
  build_pdf
281
292
 
282
293
  FileUtils.cp(File.join(@path, "#{@mastertex}.pdf"), pdf_filepath)
294
+ @logger.success("built #{File.basename(pdf_filepath)}")
283
295
  ensure
284
296
  remove_entry_secure(@path) unless @config['debug']
285
297
  end
@@ -296,23 +308,13 @@ module ReVIEW
296
308
  end
297
309
  end
298
310
 
299
- # PDFMaker#copy_images should copy image files _AND_ execute extractbb (or ebb).
311
+ # PDFMaker#copy_images should copy image files
300
312
  #
301
313
  def copy_images(from, to)
302
314
  return unless File.exist?(from)
315
+
303
316
  Dir.mkdir(to)
304
317
  ReVIEW::MakerHelper.copy_images_to_dir(from, to)
305
- Dir.chdir(to) do
306
- images = Dir.glob('**/*').find_all { |f| File.file?(f) and f =~ /\.(jpg|jpeg|png|pdf|ai|eps|tif)\z/i }
307
- break if images.empty?
308
- if @config['pdfmaker']['bbox']
309
- system_with_info('extractbb', '-B', @config['pdfmaker']['bbox'], *images)
310
- system_or_raise('ebb', '-B', @config['pdfmaker']['bbox'], *images) unless system('extractbb', '-B', @config['pdfmaker']['bbox'], '-m', *images)
311
- else
312
- system_with_info('extractbb', *images)
313
- system_or_raise('ebb', *images) unless system('extractbb', '-m', *images)
314
- end
315
- end
316
318
  end
317
319
 
318
320
  def make_custom_page(file)
@@ -320,6 +322,7 @@ module ReVIEW
320
322
  if File.exist?(file_sty)
321
323
  return File.read(file_sty)
322
324
  end
325
+
323
326
  nil
324
327
  end
325
328
 
@@ -370,11 +373,11 @@ module ReVIEW
370
373
  items.each_with_index do |item, rev|
371
374
  editstr = edit == 0 ? ReVIEW::I18n.t('first_edition') : ReVIEW::I18n.t('nth_edition', (edit + 1).to_s)
372
375
  revstr = ReVIEW::I18n.t('nth_impression', (rev + 1).to_s)
373
- if item =~ /\A\d+\-\d+\-\d+\Z/
376
+ if item =~ /\A\d+-\d+-\d+\Z/
374
377
  buf << ReVIEW::I18n.t('published_by1', [date_to_s(item), editstr + revstr])
375
- elsif item =~ /\A(\d+\-\d+\-\d+)[\s ](.+)/
378
+ elsif item =~ /\A(\d+-\d+-\d+)[\s ](.+)/
376
379
  # custom date with string
377
- item.match(/\A(\d+\-\d+\-\d+)[\s ](.+)/) { |m| buf << ReVIEW::I18n.t('published_by3', [date_to_s(m[1]), m[2]]) }
380
+ item.match(/\A(\d+-\d+-\d+)[\s ](.+)/) { |m| buf << ReVIEW::I18n.t('published_by3', [date_to_s(m[1]), m[2]]) }
378
381
  else
379
382
  # free format
380
383
  buf << item
@@ -440,38 +443,49 @@ module ReVIEW
440
443
  end
441
444
 
442
445
  @locale_latex = {}
443
- part_tuple = I18n.get('part').split(/\%[A-Za-z]{1,3}/, 2)
444
- chapter_tuple = I18n.get('chapter').split(/\%[A-Za-z]{1,3}/, 2)
445
- appendix_tuple = I18n.get('appendix').split(/\%[A-Za-z]{1,3}/, 2)
446
- @locale_latex['prepartname'] = part_tuple[0]
447
- @locale_latex['postpartname'] = part_tuple[1]
448
- @locale_latex['prechaptername'] = chapter_tuple[0]
449
- @locale_latex['postchaptername'] = chapter_tuple[1]
450
- @locale_latex['preappendixname'] = appendix_tuple[0]
451
- @locale_latex['postappendixname'] = appendix_tuple[1]
452
- end
453
-
454
- def erb_content(file)
455
- @texcompiler = File.basename(@config['texcommand'], '.*')
456
- erb = ReVIEW::Template.load(file, '-')
457
- @logger.debug("erb processes #{File.basename(file)}") if @config['debug']
458
- erb.result(binding)
446
+ part_tuple = I18n.get('part').split(/%[A-Za-z]{1,3}/, 2)
447
+ chapter_tuple = I18n.get('chapter').split(/%[A-Za-z]{1,3}/, 2)
448
+ appendix_tuple = I18n.get('appendix').split(/%[A-Za-z]{1,3}/, 2)
449
+ @locale_latex['prepartname'] = part_tuple[0].to_s
450
+ @locale_latex['postpartname'] = part_tuple[1].to_s
451
+ @locale_latex['prechaptername'] = chapter_tuple[0].to_s
452
+ @locale_latex['postchaptername'] = chapter_tuple[1].to_s
453
+ @locale_latex['preappendixname'] = appendix_tuple[0].to_s
454
+ @locale_latex['postappendixname'] = appendix_tuple[1].to_s
455
+
456
+ if @config['pdfmaker']['boxsetting']
457
+ begin
458
+ @boxsetting = ReVIEW::LaTeXBox.new.tcbox(@config)
459
+ rescue ReVIEW::ConfigError => e
460
+ error e
461
+ end
462
+ end
459
463
  end
460
464
 
461
465
  def latex_config
462
- erb_content(File.expand_path('./latex/config.erb', ReVIEW::Template::TEMPLATE_DIR))
466
+ result = ReVIEW::Template.generate(path: './latex/config.erb', mode: '-', binding: binding)
467
+ local_config_file = File.join(@basedir, 'layouts', 'config-local.tex.erb')
468
+ if File.exist?(local_config_file)
469
+ result << "%% BEGIN: config-local.tex.erb\n"
470
+ result << ReVIEW::Template.generate(path: 'layouts/config-local.tex.erb', mode: '-', binding: binding, template_dir: @basedir)
471
+ result << "%% END: config-local.tex.erb\n"
472
+ end
473
+ result
463
474
  end
464
475
 
465
476
  def template_content
466
- template = File.expand_path('./latex/layout.tex.erb', ReVIEW::Template::TEMPLATE_DIR)
477
+ template_dir = ReVIEW::Template::TEMPLATE_DIR
467
478
  if @config.check_version('2', exception: false)
468
- template = File.expand_path('./latex-compat2/layout.tex.erb', ReVIEW::Template::TEMPLATE_DIR)
479
+ template_path = './latex-compat2/layout.tex.erb'
480
+ else
481
+ template_path = './latex/layout.tex.erb'
469
482
  end
470
483
  layout_file = File.join(@basedir, 'layouts', 'layout.tex.erb')
471
484
  if File.exist?(layout_file)
472
- template = layout_file
485
+ template_dir = @basedir
486
+ template_path = 'layouts/layout.tex.erb'
473
487
  end
474
- erb_content(template)
488
+ ReVIEW::Template.generate(path: template_path, mode: '-', binding: binding, template_dir: template_dir)
475
489
  end
476
490
 
477
491
  def copy_sty(dirname, copybase, extname = 'sty')
@@ -483,6 +497,7 @@ module ReVIEW
483
497
  Dir.open(dirname) do |dir|
484
498
  dir.sort.each do |fname|
485
499
  next unless File.extname(fname).downcase == '.' + extname
500
+
486
501
  FileUtils.mkdir_p(copybase) unless Dir.exist?(copybase)
487
502
  if extname == 'erb'
488
503
  File.open(File.join(copybase, fname.sub(/\.erb\Z/, '')), 'w') do |f|
@@ -494,15 +509,5 @@ module ReVIEW
494
509
  end
495
510
  end
496
511
  end
497
-
498
- def call_hook(hookname)
499
- return if !@config['pdfmaker'].is_a?(Hash) || @config['pdfmaker'][hookname].nil?
500
- hook = File.absolute_path(@config['pdfmaker'][hookname], @basedir)
501
- if ENV['REVIEW_SAFE_MODE'].to_i & 1 > 0
502
- warn 'hook configuration is prohibited in safe mode. ignored.'
503
- else
504
- system_or_raise(hook, Dir.pwd, @basedir)
505
- end
506
- end
507
512
  end
508
513
  end
@@ -1,4 +1,4 @@
1
- # Copyright (c) 2018-2019 Kenshi Muto
1
+ # Copyright (c) 2018-2020 Kenshi Muto
2
2
  #
3
3
  # This program is free software.
4
4
  # You can distribute or modify this program under the terms of
@@ -74,7 +74,26 @@ module ReVIEW
74
74
  private :blank
75
75
 
76
76
  def result
77
- @output.string
77
+ solve_nest(@output.string)
78
+ end
79
+
80
+ def solve_nest(s)
81
+ check_nest
82
+ lines = []
83
+ clevel = []
84
+ s.split("\n", -1).each do |l| # -1 means don't omit last "\n"
85
+ if l =~ /\A\x01→(dl|ul|ol)←\x01/
86
+ clevel.push($1)
87
+ lines.push("\x01→END←\x01")
88
+ elsif l =~ %r{\A\x01→/(dl|ul|ol)←\x01}
89
+ clevel.pop
90
+ lines.push("\x01→END←\x01")
91
+ else
92
+ lines.push("\t" * clevel.size + l)
93
+ end
94
+ end
95
+
96
+ lines.join("\n").gsub(/\n*\x01→END←\x01\n*/, "\n")
78
97
  end
79
98
 
80
99
  def headline(level, _label, caption)
@@ -118,7 +137,7 @@ module ReVIEW
118
137
 
119
138
  def dd(lines)
120
139
  split_paragraph(lines).each do |paragraph|
121
- puts paragraph.gsub(/\n/, '')
140
+ puts paragraph.delete("\n")
122
141
  end
123
142
  end
124
143
 
@@ -140,13 +159,19 @@ module ReVIEW
140
159
  def list(lines, id, caption, lang = nil)
141
160
  blank
142
161
  begin
143
- list_header(id, caption, lang)
162
+ if caption_top?('list')
163
+ list_header(id, caption, lang)
164
+ blank
165
+ end
166
+ list_body(id, lines, lang)
167
+ unless caption_top?('list')
168
+ blank
169
+ list_header(id, caption, lang)
170
+ end
144
171
  rescue KeyError
145
172
  error "no such list: #{id}"
146
173
  end
147
174
  blank
148
- list_body(id, lines, lang)
149
- blank
150
175
  end
151
176
 
152
177
  def list_header(id, caption, _lang)
@@ -165,8 +190,13 @@ module ReVIEW
165
190
 
166
191
  def base_block(_type, lines, caption = nil)
167
192
  blank
168
- puts compile_inline(caption) if caption.present?
193
+ if caption_top?('list') && caption.present?
194
+ puts compile_inline(caption)
195
+ end
169
196
  puts lines.join("\n")
197
+ if !caption_top?('list') && caption.present?
198
+ puts compile_inline(caption)
199
+ end
170
200
  blank
171
201
  end
172
202
 
@@ -183,23 +213,34 @@ module ReVIEW
183
213
 
184
214
  def emlistnum(lines, caption = nil, _lang = nil)
185
215
  blank
186
- puts compile_inline(caption) if caption.present?
216
+ if caption_top?('list')
217
+ puts compile_inline(caption) if caption.present?
218
+ end
187
219
  lines.each_with_index do |line, i|
188
220
  puts((i + 1).to_s.rjust(2) + ": #{line}")
189
221
  end
222
+ unless caption_top?('list')
223
+ puts compile_inline(caption) if caption.present?
224
+ end
190
225
  blank
191
226
  end
192
227
 
193
228
  def listnum(lines, id, caption, lang = nil)
194
229
  blank
195
230
  begin
196
- list_header(id, caption, lang)
231
+ if caption_top?('list')
232
+ list_header(id, caption, lang)
233
+ blank
234
+ end
235
+ listnum_body(lines, lang)
236
+ unless caption_top?('list')
237
+ blank
238
+ list_header(id, caption, lang)
239
+ end
197
240
  rescue KeyError
198
241
  error "no such list: #{id}"
199
242
  end
200
243
  blank
201
- listnum_body(lines, lang)
202
- blank
203
244
  end
204
245
 
205
246
  def listnum_body(lines, _lang)
@@ -228,8 +269,9 @@ module ReVIEW
228
269
 
229
270
  def texequation(lines, id = nil, caption = '')
230
271
  blank
231
- texequation_header(id, caption)
272
+ texequation_header(id, caption) if caption_top?('equation')
232
273
  puts lines.join("\n")
274
+ texequation_header(id, caption) unless caption_top?('equation')
233
275
  blank
234
276
  end
235
277
 
@@ -251,6 +293,10 @@ module ReVIEW
251
293
  end
252
294
 
253
295
  def table_header(id, caption)
296
+ unless caption_top?('table')
297
+ blank
298
+ end
299
+
254
300
  if id.nil?
255
301
  puts compile_inline(caption)
256
302
  elsif get_chap
@@ -258,7 +304,10 @@ module ReVIEW
258
304
  else
259
305
  puts "#{I18n.t('table')}#{I18n.t('format_number_without_chapter', [@chapter.table(id).number])}#{I18n.t('caption_prefix_idgxml')}#{compile_inline(caption)}"
260
306
  end
261
- blank
307
+
308
+ if caption_top?('table')
309
+ blank
310
+ end
262
311
  end
263
312
 
264
313
  def table_begin(_ncols)
@@ -471,18 +520,22 @@ module ReVIEW
471
520
  end
472
521
 
473
522
  def note(lines, caption = nil)
523
+ check_nested_minicolumn
474
524
  base_parablock('note', lines, caption)
475
525
  end
476
526
 
477
527
  def memo(lines, caption = nil)
528
+ check_nested_minicolumn
478
529
  base_parablock('memo', lines, caption)
479
530
  end
480
531
 
481
532
  def tip(lines, caption = nil)
533
+ check_nested_minicolumn
482
534
  base_parablock('tip', lines, caption)
483
535
  end
484
536
 
485
537
  def info(lines, caption = nil)
538
+ check_nested_minicolumn
486
539
  base_parablock('info', lines, caption)
487
540
  end
488
541
 
@@ -491,10 +544,12 @@ module ReVIEW
491
544
  end
492
545
 
493
546
  def best(lines, caption = nil)
547
+ check_nested_minicolumn
494
548
  base_parablock('best', lines, caption)
495
549
  end
496
550
 
497
551
  def important(lines, caption = nil)
552
+ check_nested_minicolumn
498
553
  base_parablock('important', lines, caption)
499
554
  end
500
555
 
@@ -503,6 +558,7 @@ module ReVIEW
503
558
  end
504
559
 
505
560
  def caution(lines, caption = nil)
561
+ check_nested_minicolumn
506
562
  base_parablock('caution', lines, caption)
507
563
  end
508
564
 
@@ -515,6 +571,7 @@ module ReVIEW
515
571
  end
516
572
 
517
573
  def notice(lines, caption = nil)
574
+ check_nested_minicolumn
518
575
  base_parablock('notice', lines, caption)
519
576
  end
520
577
 
@@ -543,11 +600,30 @@ module ReVIEW
543
600
  end
544
601
 
545
602
  def warning(lines, caption = nil)
603
+ check_nested_minicolumn
546
604
  base_parablock('warning', lines, caption)
547
605
  end
548
606
 
549
607
  alias_method :box, :insn
550
608
 
609
+ CAPTION_TITLES.each do |name|
610
+ class_eval %Q(
611
+ def #{name}_begin(caption = nil)
612
+ check_nested_minicolumn
613
+ @doc_status[:minicolumn] = '#{name}'
614
+ blank
615
+ if caption.present?
616
+ puts compile_inline(caption)
617
+ end
618
+ end
619
+
620
+ def #{name}_end
621
+ blank
622
+ @doc_status[:minicolumn] = nil
623
+ end
624
+ ), __FILE__, __LINE__ - 14
625
+ end
626
+
551
627
  def indepimage(_lines, _id, caption = nil, _metric = nil)
552
628
  blank
553
629
  puts "図 #{compile_inline(caption)}" if caption.present?
@@ -571,10 +647,18 @@ module ReVIEW
571
647
  ''
572
648
  end
573
649
 
650
+ def inline_ins(str)
651
+ str
652
+ end
653
+
574
654
  def inline_del(_str)
575
655
  ''
576
656
  end
577
657
 
658
+ def inline_tcy(str)
659
+ str
660
+ end
661
+
578
662
  def inline_br(_str)
579
663
  "\n"
580
664
  end
@@ -583,7 +667,7 @@ module ReVIEW
583
667
  str
584
668
  end
585
669
 
586
- def inline_chap(id)
670
+ def inline_chap(id) # rubocop:disable Lint/UselessMethodDefinition
587
671
  # "「第#{super}章 #{inline_title(id)}」"
588
672
  # "第#{super}章"
589
673
  super