review 3.2.0 → 5.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (282) 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 +167 -42
  6. data/.travis.yml +1 -1
  7. data/Dockerfile +21 -5
  8. data/NEWS.ja.md +428 -1
  9. data/NEWS.md +429 -2
  10. data/README.md +11 -7
  11. data/Rakefile +7 -2
  12. data/appveyor.yml +0 -20
  13. data/bin/review +2 -4
  14. data/bin/review-catalog-converter +5 -5
  15. data/bin/review-check +13 -17
  16. data/bin/review-checkdep +3 -6
  17. data/bin/review-compile +13 -22
  18. data/bin/review-epub2html +1 -4
  19. data/bin/review-epubmaker +3 -4
  20. data/bin/review-idgxmlmaker +14 -0
  21. data/bin/review-index +5 -86
  22. data/bin/review-init +1 -4
  23. data/bin/review-pdfmaker +1 -3
  24. data/bin/review-preproc +11 -13
  25. data/bin/review-textmaker +1 -3
  26. data/bin/review-update +1 -4
  27. data/bin/review-validate +7 -7
  28. data/bin/review-vol +5 -82
  29. data/bin/review-webmaker +1 -3
  30. data/doc/config.yml.sample +67 -16
  31. data/doc/config.yml.sample-simple +4 -3
  32. data/doc/format.ja.md +133 -21
  33. data/doc/format.md +135 -34
  34. data/doc/makeindex.ja.md +2 -2
  35. data/doc/pdfmaker.ja.md +43 -1
  36. data/doc/pdfmaker.md +42 -1
  37. data/doc/quickstart.ja.md +55 -25
  38. data/doc/quickstart.md +47 -17
  39. data/lib/review.rb +1 -1
  40. data/lib/review/book.rb +2 -2
  41. data/lib/review/book/base.rb +89 -90
  42. data/lib/review/book/bib.rb +21 -0
  43. data/lib/review/book/book_unit.rb +158 -0
  44. data/lib/review/book/chapter.rb +51 -29
  45. data/lib/review/book/index.rb +42 -248
  46. data/lib/review/book/index/item.rb +46 -0
  47. data/lib/review/book/page_metric.rb +7 -7
  48. data/lib/review/book/part.rb +45 -10
  49. data/lib/review/book/volume.rb +5 -5
  50. data/lib/review/builder.rb +171 -54
  51. data/lib/review/call_hook.rb +20 -0
  52. data/lib/review/catalog.rb +15 -17
  53. data/lib/review/compiler.rb +240 -102
  54. data/lib/review/configure.rb +101 -12
  55. data/lib/review/converter.rb +1 -1
  56. data/lib/review/epub2html.rb +6 -1
  57. data/lib/review/epubmaker.rb +120 -118
  58. data/lib/review/epubmaker/content.rb +113 -0
  59. data/lib/review/epubmaker/epubcommon.rb +372 -0
  60. data/lib/review/epubmaker/epubv2.rb +178 -0
  61. data/lib/review/epubmaker/epubv3.rb +231 -0
  62. data/lib/review/epubmaker/producer.rb +168 -0
  63. data/lib/review/epubmaker/reviewheaderlistener.rb +12 -2
  64. data/lib/review/epubmaker/zip_exporter.rb +84 -0
  65. data/lib/review/exception.rb +6 -0
  66. data/lib/review/extentions/string.rb +0 -4
  67. data/lib/review/htmlbuilder.rb +147 -149
  68. data/lib/review/htmlutils.rb +10 -14
  69. data/lib/review/i18n.rb +4 -3
  70. data/lib/review/idgxmlbuilder.rb +229 -103
  71. data/lib/review/idgxmlmaker.rb +188 -0
  72. data/lib/review/img_math.rb +245 -0
  73. data/lib/review/index_builder.rb +654 -0
  74. data/lib/review/init-web/finish.html +10 -0
  75. data/lib/review/init-web/index.html +190 -0
  76. data/lib/review/init-web/review-layout-design.js +691 -0
  77. data/lib/review/init.rb +124 -41
  78. data/lib/review/latexbox.rb +58 -0
  79. data/lib/review/latexbuilder.rb +249 -97
  80. data/lib/review/latexutils.rb +9 -1
  81. data/lib/review/lineinput.rb +113 -3
  82. data/lib/review/logger.rb +43 -8
  83. data/lib/review/makerhelper.rb +13 -186
  84. data/lib/review/markdownbuilder.rb +75 -39
  85. data/lib/review/md2inaobuilder.rb +3 -5
  86. data/lib/review/pdfmaker.rb +88 -83
  87. data/lib/review/plaintextbuilder.rb +158 -83
  88. data/lib/review/preprocessor.rb +30 -24
  89. data/lib/review/rstbuilder.rb +58 -39
  90. data/lib/review/sec_counter.rb +14 -0
  91. data/lib/review/template.rb +6 -0
  92. data/lib/review/textmaker.rb +29 -17
  93. data/lib/review/textutils.rb +68 -2
  94. data/lib/review/tocprinter.rb +242 -97
  95. data/lib/review/topbuilder.rb +147 -61
  96. data/lib/review/update.rb +27 -28
  97. data/lib/review/version.rb +1 -1
  98. data/lib/review/volumeprinter.rb +97 -0
  99. data/lib/review/webmaker.rb +36 -35
  100. data/lib/review/webtocprinter.rb +39 -35
  101. data/lib/review/yamlloader.rb +4 -0
  102. data/review.gemspec +8 -5
  103. data/samples/sample-book/README.md +7 -2
  104. data/samples/sample-book/src/.gitignore +154 -0
  105. data/samples/sample-book/src/config-ebook.yml +4 -0
  106. data/samples/sample-book/src/config-epub2.yml +1 -1
  107. data/samples/sample-book/src/config-jlreq-ebook.yml +4 -0
  108. data/samples/sample-book/src/config-jlreq.yml +6 -0
  109. data/samples/sample-book/src/config.yml +3 -3
  110. data/samples/sample-book/src/lib/tasks/review.rake +45 -14
  111. data/samples/sample-book/src/lib/tasks/z01_copy_sty.rake +14 -8
  112. data/samples/syntax-book/Gemfile +1 -1
  113. data/samples/syntax-book/ch01.re +3 -1
  114. data/samples/syntax-book/ch02.re +28 -21
  115. data/samples/syntax-book/ch03.re +4 -7
  116. data/samples/syntax-book/config-jlreq-lualatex.yml +4 -0
  117. data/samples/syntax-book/config-jlreq.yml +5 -0
  118. data/samples/syntax-book/config-print.yml +3 -0
  119. data/samples/syntax-book/config.yml +1 -1
  120. data/samples/syntax-book/images/img3-2.png +0 -0
  121. data/samples/syntax-book/lib/tasks/review.rake +30 -15
  122. data/samples/syntax-book/lib/tasks/z01_copy_sty.rake +14 -8
  123. data/templates/html/_colophon.html.erb +23 -0
  124. data/templates/html/_colophon_history.html.erb +9 -0
  125. data/templates/html/_cover.html.erb +10 -0
  126. data/templates/html/_part_body.html.erb +6 -0
  127. data/templates/html/_titlepage.html.erb +20 -0
  128. data/templates/html/layout-html5.html.erb +6 -0
  129. data/templates/html/layout-xhtml1.html.erb +6 -0
  130. data/templates/latex/config.erb +45 -27
  131. data/templates/latex/layout.tex.erb +1 -0
  132. data/templates/latex/review-jlreq/README.md +3 -1
  133. data/templates/latex/review-jlreq/review-base.sty +124 -37
  134. data/templates/latex/review-jlreq/review-jlreq.cls +29 -22
  135. data/templates/latex/review-jlreq/review-style.sty +9 -1
  136. data/templates/latex/review-jlreq/review-tcbox.sty +348 -0
  137. data/templates/latex/review-jlreq/reviewmacro.sty +5 -0
  138. data/templates/latex/review-jsbook/README.md +46 -5
  139. data/templates/latex/review-jsbook/review-base.sty +111 -30
  140. data/templates/latex/review-jsbook/review-jsbook.cls +16 -2
  141. data/templates/latex/review-jsbook/review-style.sty +10 -2
  142. data/templates/latex/review-jsbook/review-tcbox.sty +348 -0
  143. data/templates/latex/review-jsbook/reviewmacro.sty +5 -0
  144. data/templates/opf/epubv2.opf.erb +7 -7
  145. data/templates/opf/epubv3.opf.erb +7 -7
  146. data/templates/opf/opf_manifest_epubv2.opf.erb +10 -0
  147. data/templates/opf/opf_manifest_epubv3.opf.erb +10 -0
  148. data/templates/opf/opf_metainfo_epubv2.opf.erb +17 -0
  149. data/templates/opf/opf_metainfo_epubv3.opf.erb +49 -0
  150. data/templates/opf/opf_tocx_epubv2.opf.erb +9 -0
  151. data/templates/opf/opf_tocx_epubv3.opf.erb +17 -0
  152. data/templates/web/html/layout-html5.html.erb +9 -8
  153. data/templates/web/html/layout-xhtml1.html.erb +6 -0
  154. data/test/assets/header_listener.html +35 -0
  155. data/test/assets/img_math/img1.png +0 -0
  156. data/test/assets/img_math/img2.png +0 -0
  157. data/test/assets/img_math/img3.png +0 -0
  158. data/test/assets/syntax_book_index_detail.txt +58 -0
  159. data/test/assets/test_template.tex +20 -9
  160. data/test/assets/test_template_backmatter.tex +20 -9
  161. data/test/book_test_helper.rb +11 -5
  162. data/test/run_test.rb +1 -1
  163. data/test/test_book.rb +123 -78
  164. data/test/test_book_chapter.rb +99 -56
  165. data/test/test_book_part.rb +3 -3
  166. data/test/test_builder.rb +24 -15
  167. data/test/test_catalog.rb +19 -42
  168. data/test/test_catalog_converter_cmd.rb +1 -1
  169. data/test/test_converter.rb +1 -0
  170. data/test/test_epub3maker.rb +170 -126
  171. data/test/test_epubmaker.rb +249 -129
  172. data/test/test_epubmaker_cmd.rb +16 -9
  173. data/test/test_helper.rb +23 -11
  174. data/test/test_htmlbuilder.rb +1083 -114
  175. data/test/test_htmlutils.rb +0 -12
  176. data/test/test_i18n.rb +37 -37
  177. data/test/test_idgxmlbuilder.rb +627 -27
  178. data/test/test_idgxmlmaker_cmd.rb +50 -0
  179. data/test/test_image_finder.rb +52 -70
  180. data/test/test_img_math.rb +111 -0
  181. data/test/test_index.rb +62 -52
  182. data/test/test_indexbuilder.rb +52 -0
  183. data/test/test_latexbuilder.rb +1189 -59
  184. data/test/test_latexbuilder_v2.rb +74 -34
  185. data/test/test_lineinput.rb +20 -93
  186. data/test/test_logger.rb +17 -4
  187. data/test/test_makerhelper.rb +2 -14
  188. data/test/test_markdownbuilder.rb +77 -4
  189. data/test/test_md2inaobuilder.rb +12 -2
  190. data/test/test_pdfmaker.rb +101 -12
  191. data/test/test_pdfmaker_cmd.rb +102 -8
  192. data/test/test_plaintextbuilder.rb +562 -31
  193. data/test/test_review_ext.rb +2 -1
  194. data/test/test_reviewheaderlistener.rb +49 -0
  195. data/test/test_rstbuilder.rb +58 -5
  196. data/test/test_sec_counter.rb +156 -0
  197. data/test/test_template.rb +12 -2
  198. data/test/test_textmaker_cmd.rb +58 -0
  199. data/test/test_textutils.rb +109 -2
  200. data/test/test_tocprinter.rb +46 -0
  201. data/test/test_topbuilder.rb +400 -17
  202. data/test/test_update.rb +53 -44
  203. data/test/test_webtocprinter.rb +75 -43
  204. data/test/test_yamlloader.rb +13 -0
  205. data/test/test_zip_exporter.rb +5 -6
  206. data/vendor/gentombow/LICENSE +1 -1
  207. data/vendor/gentombow/Makefile +0 -1
  208. data/vendor/gentombow/bounddvi-en.pdf +0 -0
  209. data/vendor/gentombow/bounddvi-en.tex +1 -0
  210. data/vendor/gentombow/bounddvi.pdf +0 -0
  211. data/vendor/gentombow/bounddvi.sty +30 -7
  212. data/vendor/gentombow/bounddvi.tex +1 -0
  213. data/vendor/gentombow/create_archive.sh +1 -0
  214. data/vendor/gentombow/gentombow-ja.pdf +0 -0
  215. data/vendor/gentombow/gentombow-ja.tex +9 -0
  216. data/vendor/gentombow/gentombow.pdf +0 -0
  217. data/vendor/gentombow/gentombow.sty +32 -10
  218. data/vendor/gentombow/gentombow.tex +8 -0
  219. data/vendor/gentombow/tests/gentombow-01-pdfx.tex +8 -0
  220. data/vendor/gentombow/tests/gentombow-02-pdfx.tex +8 -0
  221. data/vendor/jsclasses/LICENSE +1 -1
  222. data/vendor/jsclasses/Makefile +3 -2
  223. data/vendor/jsclasses/create_archive.sh +5 -5
  224. data/vendor/jsclasses/jis/Makefile +3 -2
  225. data/vendor/jsclasses/jis/jsarticle.cls +74 -31
  226. data/vendor/jsclasses/jis/jsbook.cls +74 -31
  227. data/vendor/jsclasses/jis/jsclasses.dtx +176 -36
  228. data/vendor/jsclasses/jis/jsclasses.ins +15 -5
  229. data/vendor/jsclasses/jis/jslogo.dtx +4 -4
  230. data/vendor/jsclasses/jis/jslogo.ins +9 -0
  231. data/vendor/jsclasses/jis/jslogo.sty +4 -16
  232. data/vendor/jsclasses/jis/jspf.cls +73 -30
  233. data/vendor/jsclasses/jis/jsreport.cls +74 -31
  234. data/vendor/jsclasses/jis/jsverb.ins +9 -0
  235. data/vendor/jsclasses/jis/jsverb.sty +1 -13
  236. data/vendor/jsclasses/jis/kiyou.cls +74 -31
  237. data/vendor/jsclasses/jis/minijs.sty +65 -22
  238. data/vendor/jsclasses/jis/okumacro.dtx +4 -5
  239. data/vendor/jsclasses/jis/okumacro.ins +9 -0
  240. data/vendor/jsclasses/jis/okumacro.sty +4 -17
  241. data/vendor/jsclasses/jis/okuverb.ins +9 -0
  242. data/vendor/jsclasses/jis/okuverb.sty +1 -13
  243. data/vendor/jsclasses/jis/winjis.sty +23 -19
  244. data/vendor/jsclasses/jsarticle.cls +74 -31
  245. data/vendor/jsclasses/jsbook.cls +74 -31
  246. data/vendor/jsclasses/jsclasses.dtx +176 -36
  247. data/vendor/jsclasses/jsclasses.ins +15 -5
  248. data/vendor/jsclasses/jsclasses.pdf +0 -0
  249. data/vendor/jsclasses/jslogo.dtx +4 -4
  250. data/vendor/jsclasses/jslogo.ins +9 -0
  251. data/vendor/jsclasses/jslogo.pdf +0 -0
  252. data/vendor/jsclasses/jslogo.sty +4 -16
  253. data/vendor/jsclasses/jspf.cls +73 -30
  254. data/vendor/jsclasses/jsreport.cls +74 -31
  255. data/vendor/jsclasses/jsverb.ins +9 -0
  256. data/vendor/jsclasses/jsverb.pdf +0 -0
  257. data/vendor/jsclasses/jsverb.sty +1 -13
  258. data/vendor/jsclasses/kiyou.cls +74 -31
  259. data/vendor/jsclasses/minijs.sty +68 -22
  260. data/vendor/jsclasses/okumacro.dtx +4 -5
  261. data/vendor/jsclasses/okumacro.ins +9 -0
  262. data/vendor/jsclasses/okumacro.pdf +0 -0
  263. data/vendor/jsclasses/okumacro.sty +4 -17
  264. data/vendor/jsclasses/okuverb.ins +9 -0
  265. data/vendor/jsclasses/okuverb.pdf +0 -0
  266. data/vendor/jsclasses/okuverb.sty +1 -13
  267. data/vendor/jsclasses/tests/relfont.tex +10 -0
  268. data/vendor/jsclasses/winjis.sty +23 -19
  269. metadata +136 -23
  270. data/.rubocop_todo.yml +0 -7
  271. data/lib/epubmaker.rb +0 -23
  272. data/lib/epubmaker/content.rb +0 -110
  273. data/lib/epubmaker/epubcommon.rb +0 -441
  274. data/lib/epubmaker/epubv2.rb +0 -143
  275. data/lib/epubmaker/epubv3.rb +0 -233
  276. data/lib/epubmaker/producer.rb +0 -375
  277. data/lib/epubmaker/zip_exporter.rb +0 -81
  278. data/lib/lineinput.rb +0 -155
  279. data/lib/review/book/compilable.rb +0 -173
  280. data/lib/review/tocparser.rb +0 -271
  281. data/samples/syntax-book/review-ext.rb +0 -14
  282. data/test/test_tocparser.rb +0 -25
@@ -1,3 +1,5 @@
1
+ # Copyright (c) 2013-2020 KADO Masanori, Masayoshi Takahashi, Kenshi Muto
2
+ #
1
3
  # This program is free software.
2
4
  # You can distribute or modify this program under the terms of
3
5
  # the GNU LGPL, Lesser General Public License version 2.1.
@@ -50,11 +52,11 @@ module ReVIEW
50
52
 
51
53
  def paragraph(lines)
52
54
  if @noindent
53
- puts %Q(<p class="noindent">#{lines.join}</p>)
55
+ puts %Q(<p class="noindent">#{join_lines_to_paragraph(lines)}</p>)
54
56
  puts "\n"
55
57
  @noindent = nil
56
58
  else
57
- puts lines.join
59
+ puts join_lines_to_paragraph(lines)
58
60
  puts "\n"
59
61
  end
60
62
  end
@@ -63,24 +65,26 @@ module ReVIEW
63
65
  @noindent = true
64
66
  end
65
67
 
66
- def list_header(id, caption, lang)
68
+ def list_header(id, caption, _lang)
67
69
  if get_chap.nil?
68
70
  print %Q(リスト#{@chapter.list(id).number} #{compile_inline(caption)}\n\n)
69
71
  else
70
72
  print %Q(リスト#{get_chap}.#{@chapter.list(id).number} #{compile_inline(caption)}\n\n)
71
73
  end
72
- lang ||= ''
73
- puts "```#{lang}"
74
74
  end
75
75
 
76
- def list_body(_id, lines, _lang)
76
+ def list_body(_id, lines, lang)
77
+ lang ||= ''
78
+ puts "```#{lang}"
77
79
  lines.each do |line|
78
80
  puts detab(line)
79
81
  end
80
82
  puts '```'
81
83
  end
82
84
 
83
- def listnum_body(lines, _lang)
85
+ def listnum_body(lines, lang)
86
+ lang ||= ''
87
+ puts "```#{lang}"
84
88
  lines.each_with_index do |line, i|
85
89
  puts((i + 1).to_s.rjust(2) + ": #{detab(line)}")
86
90
  end
@@ -93,7 +97,7 @@ module ReVIEW
93
97
  end
94
98
 
95
99
  def ul_item_begin(lines)
96
- puts ' ' * (@ul_indent - 1) + '* ' + lines.join
100
+ puts ' ' * (@ul_indent - 1) + '* ' + join_lines_to_paragraph(lines)
97
101
  end
98
102
 
99
103
  def ul_item_end
@@ -109,7 +113,7 @@ module ReVIEW
109
113
  end
110
114
 
111
115
  def ol_item(lines, num)
112
- puts "#{num}. #{lines.join}"
116
+ puts "#{num}. #{join_lines_to_paragraph(lines)}"
113
117
  end
114
118
 
115
119
  def ol_end
@@ -125,7 +129,7 @@ module ReVIEW
125
129
  end
126
130
 
127
131
  def dd(lines)
128
- puts "<dd>#{lines.join}</dd>"
132
+ puts "<dd>#{join_lines_to_paragraph(lines)}</dd>"
129
133
  end
130
134
 
131
135
  def dl_end
@@ -134,9 +138,9 @@ module ReVIEW
134
138
 
135
139
  def emlist(lines, caption = nil, lang = nil)
136
140
  blank
137
- if caption
141
+ if caption.present?
138
142
  puts caption
139
- print "\n"
143
+ blank
140
144
  end
141
145
  lang ||= ''
142
146
  puts "```#{lang}"
@@ -151,10 +155,28 @@ module ReVIEW
151
155
  puts %Q(<div class="#{type}">)
152
156
  puts %Q(<p class="caption">#{compile_inline(caption)}</p>) if caption.present?
153
157
  blocked_lines = split_paragraph(lines)
154
- puts blocked_lines.join("\n")
158
+ puts blocked_lines.join("\n\n")
155
159
  puts '</div>'
156
160
  end
157
161
 
162
+ CAPTION_TITLES.each do |name|
163
+ class_eval %Q(
164
+ def #{name}_begin(caption = nil)
165
+ check_nested_minicolumn
166
+ @doc_status[:minicolumn] = '#{name}'
167
+ puts %Q(<div class="#{name}">)
168
+ if caption.present?
169
+ puts %Q(<p class="caption">\#{compile_inline(caption)}</p>)
170
+ end
171
+ end
172
+
173
+ def #{name}_end
174
+ puts '</div>'
175
+ @doc_status[:minicolumn] = nil
176
+ end
177
+ ), __FILE__, __LINE__ - 14
178
+ end
179
+
158
180
  def hr
159
181
  puts '----'
160
182
  end
@@ -202,6 +224,14 @@ module ReVIEW
202
224
  "<u>#{str}</u>"
203
225
  end
204
226
 
227
+ def inline_ins(str)
228
+ "<ins>#{str}</ins>"
229
+ end
230
+
231
+ def inline_del(str)
232
+ "~~#{str}~~"
233
+ end
234
+
205
235
  def image_image(id, caption, _metric)
206
236
  blank
207
237
  puts "![#{compile_inline(caption)}](#{@chapter.image(id).path.sub(%r{\A\./}, '')})"
@@ -222,6 +252,28 @@ module ReVIEW
222
252
  "<!-- DTP:#{str} -->"
223
253
  end
224
254
 
255
+ def inline_hd_chap(chap, id)
256
+ n = chap.headline_index.number(id)
257
+ if n.present? && chap.number && over_secnolevel?(n)
258
+ str = I18n.t('hd_quote', [n, compile_inline(chap.headline(id).caption)])
259
+ else
260
+ str = I18n.t('hd_quote_without_number', compile_inline(chap.headline(id).caption))
261
+ end
262
+ if @book.config['chapterlink']
263
+ if @chapter == chap
264
+ anchor = 'h' + n.tr('.', '-')
265
+ %Q(<a href="##{anchor}">#{str}</a>)
266
+ else
267
+ warn 'MARKDOWNBuilder does not support links to other chapters'
268
+ str
269
+ end
270
+ else
271
+ str
272
+ end
273
+ rescue KeyError
274
+ error "unknown headline: #{id}"
275
+ end
276
+
225
277
  def indepimage(_lines, id, caption = '', _metric = nil)
226
278
  blank
227
279
  puts "![#{compile_inline(caption)}](#{@chapter.image(id).path.sub(%r{\A\./}, '')})"
@@ -236,7 +288,11 @@ module ReVIEW
236
288
  'jpg'
237
289
  end
238
290
 
239
- def cmd(lines)
291
+ def cmd(lines, caption = nil)
292
+ if caption.present?
293
+ puts caption
294
+ blank
295
+ end
240
296
  puts '```shell-session'
241
297
  lines.each do |line|
242
298
  puts detab(line)
@@ -244,32 +300,12 @@ module ReVIEW
244
300
  puts '```'
245
301
  end
246
302
 
247
- def table(lines, id = nil, caption = nil)
248
- rows = []
249
- sepidx = nil
250
- lines.each_with_index do |line, idx|
251
- if /\A[\=\-]{12}/ =~ line
252
- # just ignore
253
- # error "too many table separator" if sepidx
254
- sepidx ||= idx
255
- next
256
- end
257
- rows.push(line.strip.split(/\t+/).map { |s| s.sub(/\A\./, '') })
258
- end
259
- rows = adjust_n_cols(rows)
260
- error 'no rows in the table' if rows.empty?
261
-
262
- begin
263
- table_header id, caption unless caption.nil?
264
- rescue KeyError
265
- error "no such table: #{id}"
266
- end
267
- table_begin rows.first.size
303
+ def table_rows(sepidx, rows)
268
304
  if sepidx
269
305
  sepidx.times do
270
306
  tr(rows.shift.map { |s| th(s) })
271
307
  end
272
- table_border rows.first.size
308
+ table_border(rows.first.size)
273
309
  rows.each do |cols|
274
310
  tr(cols.map { |s| td(s) })
275
311
  end
@@ -279,7 +315,6 @@ module ReVIEW
279
315
  tr([th(h)] + cs.map { |s| td(s) })
280
316
  end
281
317
  end
282
- table_end
283
318
  end
284
319
 
285
320
  def table_header(id, caption)
@@ -353,9 +388,10 @@ module ReVIEW
353
388
 
354
389
  def comment(lines, comment = nil)
355
390
  return unless @book.config['draft']
391
+
356
392
  lines ||= []
357
393
  unless comment.blank?
358
- lines.unshift comment
394
+ lines.unshift(comment)
359
395
  end
360
396
  str = lines.join('<br />')
361
397
  puts %Q(<div class="red">#{escape(str)}</div>)
@@ -380,7 +416,7 @@ module ReVIEW
380
416
 
381
417
  def flushright(lines)
382
418
  puts %Q(<div class="flushright">)
383
- puts lines.join
419
+ puts split_paragraph(lines).join("\n")
384
420
  puts %Q(</div>)
385
421
  end
386
422
  end
@@ -7,13 +7,11 @@ require 'review/markdownbuilder'
7
7
  module ReVIEW
8
8
  class MD2INAOBuilder < MARKDOWNBuilder
9
9
  def paragraph(lines)
10
- puts ' ' + lines.join
10
+ puts ' ' + join_lines_to_paragraph(lines)
11
11
  puts "\n"
12
12
  end
13
13
 
14
- def list_header(id, caption, lang)
15
- lang ||= ''
16
- puts "```#{lang}"
14
+ def list_header(id, caption, _lang)
17
15
  print %Q(●リスト#{@chapter.list(id).number}::#{compile_inline(caption)}\n\n)
18
16
  end
19
17
 
@@ -35,7 +33,7 @@ module ReVIEW
35
33
  end
36
34
 
37
35
  def dd(lines)
38
- puts "<dd>#{lines.join}</dd>"
36
+ puts "<dd>#{join_lines_to_paragraph(lines)}</dd>"
39
37
  end
40
38
 
41
39
  def dl_end
@@ -1,4 +1,4 @@
1
- # Copyright (c) 2010-2018 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
 
@@ -54,12 +57,12 @@ module ReVIEW
54
57
  end
55
58
 
56
59
  def error(msg)
57
- @logger.error "#{File.basename($PROGRAM_NAME, '.*')}: #{msg}"
60
+ @logger.error msg
58
61
  exit 1
59
62
  end
60
63
 
61
64
  def warn(msg)
62
- @logger.warn "#{File.basename($PROGRAM_NAME, '.*')}: #{msg}"
65
+ @logger.warn msg
63
66
  end
64
67
 
65
68
  def pdf_filepath
@@ -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
 
@@ -153,21 +151,27 @@ module ReVIEW
153
151
  end
154
152
 
155
153
  begin
156
- generate_pdf(yamlfile)
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
162
161
 
163
- def make_input_files(book, yamlfile)
162
+ def make_input_files(book)
164
163
  input_files = Hash.new { |h, key| h[key] = '' }
165
164
  book.parts.each do |part|
166
165
  if part.name.present?
167
166
  @config['use_part'] = true
168
167
  if part.file?
169
- output_chaps(part.name, yamlfile)
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, yamlfile)
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
 
@@ -189,9 +200,9 @@ module ReVIEW
189
200
  def build_pdf
190
201
  template = template_content
191
202
  Dir.chdir(@path) do
192
- File.open("./#{@mastertex}.tex", 'wb') { |f| f.write(template) }
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,34 +245,34 @@ 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
- def generate_pdf(yamlfile)
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(File.dirname(yamlfile))
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
- @input_files = make_input_files(book, yamlfile)
275
+ @input_files = make_input_files(book)
265
276
 
266
277
  check_compile_status(@config['ignore-errors'])
267
278
 
@@ -280,12 +291,13 @@ 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
- remove_entry_secure @path unless @config['debug']
296
+ remove_entry_secure(@path) unless @config['debug']
285
297
  end
286
298
  end
287
299
 
288
- def output_chaps(filename, _yamlfile)
300
+ def output_chaps(filename)
289
301
  @logger.info "compiling #{filename}.tex"
290
302
  begin
291
303
  @converter.convert(filename + '.re', File.join(@path, filename + '.tex'))
@@ -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,26 +497,17 @@ 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|
489
504
  f.print erb_content(File.join(dirname, fname))
490
505
  end
491
506
  else
492
- FileUtils.cp File.join(dirname, fname), copybase
507
+ FileUtils.cp(File.join(dirname, fname), copybase)
493
508
  end
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