review 3.1.0 → 5.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (225) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/ruby-win.yml +45 -0
  3. data/.github/workflows/ruby.yml +27 -0
  4. data/.rubocop.yml +161 -34
  5. data/.travis.yml +16 -15
  6. data/Dockerfile +21 -5
  7. data/NEWS.ja.md +394 -0
  8. data/NEWS.md +395 -1
  9. data/README.md +10 -7
  10. data/appveyor.yml +1 -3
  11. data/bin/review-catalog-converter +5 -5
  12. data/bin/review-check +10 -12
  13. data/bin/review-checkdep +2 -2
  14. data/bin/review-compile +17 -23
  15. data/bin/review-epubmaker +3 -35
  16. data/bin/review-idgxmlmaker +16 -0
  17. data/bin/review-index +2 -89
  18. data/bin/review-preproc +13 -13
  19. data/bin/review-validate +4 -4
  20. data/bin/review-vol +4 -78
  21. data/doc/config.yml.sample +47 -12
  22. data/doc/config.yml.sample-simple +3 -2
  23. data/doc/format.ja.md +120 -17
  24. data/doc/format.md +119 -27
  25. data/doc/makeindex.ja.md +2 -2
  26. data/doc/pdfmaker.ja.md +43 -1
  27. data/doc/pdfmaker.md +42 -1
  28. data/doc/quickstart.ja.md +45 -25
  29. data/doc/quickstart.md +37 -16
  30. data/lib/epubmaker/content.rb +3 -2
  31. data/lib/epubmaker/epubcommon.rb +34 -27
  32. data/lib/epubmaker/epubv2.rb +5 -6
  33. data/lib/epubmaker/epubv3.rb +21 -18
  34. data/lib/epubmaker/producer.rb +2 -1
  35. data/lib/review/book.rb +2 -2
  36. data/lib/review/book/base.rb +91 -96
  37. data/lib/review/book/bib.rb +21 -0
  38. data/lib/review/book/book_unit.rb +155 -0
  39. data/lib/review/book/chapter.rb +48 -29
  40. data/lib/review/book/index.rb +46 -240
  41. data/lib/review/book/index/item.rb +46 -0
  42. data/lib/review/book/page_metric.rb +7 -7
  43. data/lib/review/book/part.rb +45 -10
  44. data/lib/review/book/volume.rb +4 -5
  45. data/lib/review/builder.rb +172 -56
  46. data/lib/review/catalog.rb +14 -17
  47. data/lib/review/compiler.rb +219 -121
  48. data/lib/review/configure.rb +39 -7
  49. data/lib/review/converter.rb +1 -1
  50. data/lib/review/epub2html.rb +43 -5
  51. data/lib/review/epubmaker.rb +69 -27
  52. data/lib/review/extentions/string.rb +0 -4
  53. data/lib/review/htmlbuilder.rb +112 -101
  54. data/lib/review/htmlutils.rb +9 -13
  55. data/lib/review/i18n.rb +3 -3
  56. data/lib/review/idgxmlbuilder.rb +202 -78
  57. data/lib/review/idgxmlmaker.rb +186 -0
  58. data/lib/review/index_builder.rb +653 -0
  59. data/lib/review/init-web/finish.html +10 -0
  60. data/lib/review/init-web/index.html +190 -0
  61. data/lib/review/init-web/review-layout-design.js +691 -0
  62. data/lib/review/init.rb +129 -46
  63. data/lib/review/latexbuilder.rb +255 -92
  64. data/lib/review/lineinput.rb +1 -1
  65. data/lib/review/location.rb +32 -0
  66. data/lib/review/logger.rb +4 -8
  67. data/lib/review/makerhelper.rb +35 -5
  68. data/lib/review/markdownbuilder.rb +50 -38
  69. data/lib/review/md2inaobuilder.rb +3 -5
  70. data/lib/review/pdfmaker.rb +60 -57
  71. data/lib/review/plaintextbuilder.rb +154 -87
  72. data/lib/review/preprocessor.rb +20 -42
  73. data/lib/review/rstbuilder.rb +57 -38
  74. data/lib/review/sec_counter.rb +13 -0
  75. data/lib/review/textmaker.rb +23 -15
  76. data/lib/review/textutils.rb +76 -2
  77. data/lib/review/tocprinter.rb +230 -102
  78. data/lib/review/topbuilder.rb +139 -60
  79. data/lib/review/update.rb +24 -24
  80. data/lib/review/version.rb +1 -1
  81. data/lib/review/volumeprinter.rb +98 -0
  82. data/lib/review/webmaker.rb +20 -24
  83. data/lib/review/webtocprinter.rb +38 -35
  84. data/lib/review/yamlloader.rb +26 -16
  85. data/review.gemspec +6 -4
  86. data/samples/sample-book/README.md +7 -2
  87. data/samples/sample-book/src/.gitignore +154 -0
  88. data/samples/sample-book/src/config-ebook.yml +4 -0
  89. data/samples/sample-book/src/config-jlreq-ebook.yml +4 -0
  90. data/samples/sample-book/src/config-jlreq.yml +6 -0
  91. data/samples/sample-book/src/config.yml +2 -2
  92. data/samples/sample-book/src/lib/tasks/review.rake +29 -14
  93. data/samples/sample-book/src/lib/tasks/z01_copy_sty.rake +14 -8
  94. data/samples/syntax-book/Gemfile +1 -1
  95. data/samples/syntax-book/ch01.re +4 -2
  96. data/samples/syntax-book/ch02.re +8 -16
  97. data/samples/syntax-book/ch03.re +3 -6
  98. data/samples/syntax-book/config-jlreq-lualatex.yml +4 -0
  99. data/samples/syntax-book/config-jlreq.yml +5 -0
  100. data/samples/syntax-book/config-print.yml +3 -0
  101. data/samples/syntax-book/config.yml +1 -1
  102. data/samples/syntax-book/lib/tasks/review.rake +30 -15
  103. data/samples/syntax-book/lib/tasks/z01_copy_sty.rake +14 -8
  104. data/templates/latex/config.erb +39 -25
  105. data/templates/latex/layout.tex.erb +1 -0
  106. data/templates/latex/review-jlreq/README.md +3 -1
  107. data/templates/latex/review-jlreq/review-base.sty +161 -50
  108. data/templates/latex/review-jlreq/review-jlreq.cls +21 -22
  109. data/templates/latex/review-jlreq/review-style.sty +4 -1
  110. data/templates/latex/review-jsbook/README.md +46 -5
  111. data/templates/latex/review-jsbook/review-base.sty +123 -35
  112. data/templates/latex/review-jsbook/review-jsbook.cls +10 -4
  113. data/templates/latex/review-jsbook/review-style.sty +5 -2
  114. data/templates/opf/epubv3.opf.erb +1 -0
  115. data/templates/web/html/layout-html5.html.erb +3 -3
  116. data/test/assets/test_template.tex +19 -7
  117. data/test/assets/test_template_backmatter.tex +19 -7
  118. data/test/book_test_helper.rb +11 -5
  119. data/test/test_book.rb +124 -79
  120. data/test/test_book_chapter.rb +97 -54
  121. data/test/test_book_part.rb +3 -3
  122. data/test/test_builder.rb +38 -13
  123. data/test/test_catalog.rb +24 -42
  124. data/test/test_catalog_converter_cmd.rb +1 -1
  125. data/test/test_converter.rb +1 -0
  126. data/test/test_epub3maker.rb +2 -2
  127. data/test/test_epubmaker.rb +8 -0
  128. data/test/test_epubmaker_cmd.rb +14 -7
  129. data/test/test_helper.rb +18 -7
  130. data/test/test_htmlbuilder.rb +1491 -205
  131. data/test/test_htmlutils.rb +0 -12
  132. data/test/test_i18n.rb +37 -37
  133. data/test/test_idgxmlbuilder.rb +744 -42
  134. data/test/test_idgxmlmaker_cmd.rb +46 -0
  135. data/test/test_image_finder.rb +52 -70
  136. data/test/test_index.rb +94 -44
  137. data/test/test_indexbuilder.rb +52 -0
  138. data/test/test_latexbuilder.rb +1784 -161
  139. data/test/test_latexbuilder_v2.rb +671 -102
  140. data/test/test_logger.rb +17 -4
  141. data/test/test_makerhelper.rb +2 -14
  142. data/test/test_markdownbuilder.rb +137 -16
  143. data/test/test_md2inaobuilder.rb +32 -9
  144. data/test/test_pdfmaker.rb +30 -12
  145. data/test/test_pdfmaker_cmd.rb +100 -6
  146. data/test/test_plaintextbuilder.rb +791 -30
  147. data/test/test_preprocessor.rb +2 -16
  148. data/test/test_review_ext.rb +2 -1
  149. data/test/test_rstbuilder.rb +274 -27
  150. data/test/test_sec_counter.rb +156 -0
  151. data/test/test_textmaker_cmd.rb +54 -0
  152. data/test/test_textutils.rb +109 -2
  153. data/test/test_topbuilder.rb +724 -34
  154. data/test/test_update.rb +20 -11
  155. data/test/test_webtocprinter.rb +75 -43
  156. data/test/test_yamlloader.rb +13 -0
  157. data/vendor/gentombow/LICENSE +1 -1
  158. data/vendor/gentombow/Makefile +0 -1
  159. data/vendor/gentombow/bounddvi-en.pdf +0 -0
  160. data/vendor/gentombow/bounddvi-en.tex +1 -0
  161. data/vendor/gentombow/bounddvi.pdf +0 -0
  162. data/vendor/gentombow/bounddvi.sty +30 -7
  163. data/vendor/gentombow/bounddvi.tex +1 -0
  164. data/vendor/gentombow/create_archive.sh +1 -0
  165. data/vendor/gentombow/gentombow-ja.pdf +0 -0
  166. data/vendor/gentombow/gentombow-ja.tex +9 -0
  167. data/vendor/gentombow/gentombow.pdf +0 -0
  168. data/vendor/gentombow/gentombow.sty +32 -10
  169. data/vendor/gentombow/gentombow.tex +8 -0
  170. data/vendor/gentombow/tests/gentombow-01-pdfx.tex +8 -0
  171. data/vendor/gentombow/tests/gentombow-02-pdfx.tex +8 -0
  172. data/vendor/jsclasses/LICENSE +1 -1
  173. data/vendor/jsclasses/Makefile +3 -2
  174. data/vendor/jsclasses/create_archive.sh +5 -5
  175. data/vendor/jsclasses/jis/Makefile +3 -2
  176. data/vendor/jsclasses/jis/jsarticle.cls +74 -31
  177. data/vendor/jsclasses/jis/jsbook.cls +74 -31
  178. data/vendor/jsclasses/jis/jsclasses.dtx +176 -36
  179. data/vendor/jsclasses/jis/jsclasses.ins +15 -5
  180. data/vendor/jsclasses/jis/jslogo.dtx +4 -4
  181. data/vendor/jsclasses/jis/jslogo.ins +9 -0
  182. data/vendor/jsclasses/jis/jslogo.sty +4 -16
  183. data/vendor/jsclasses/jis/jspf.cls +73 -30
  184. data/vendor/jsclasses/jis/jsreport.cls +74 -31
  185. data/vendor/jsclasses/jis/jsverb.ins +9 -0
  186. data/vendor/jsclasses/jis/jsverb.sty +1 -13
  187. data/vendor/jsclasses/jis/kiyou.cls +74 -31
  188. data/vendor/jsclasses/jis/minijs.sty +65 -22
  189. data/vendor/jsclasses/jis/okumacro.dtx +4 -5
  190. data/vendor/jsclasses/jis/okumacro.ins +9 -0
  191. data/vendor/jsclasses/jis/okumacro.sty +4 -17
  192. data/vendor/jsclasses/jis/okuverb.ins +9 -0
  193. data/vendor/jsclasses/jis/okuverb.sty +1 -13
  194. data/vendor/jsclasses/jis/winjis.sty +23 -19
  195. data/vendor/jsclasses/jsarticle.cls +74 -31
  196. data/vendor/jsclasses/jsbook.cls +74 -31
  197. data/vendor/jsclasses/jsclasses.dtx +176 -36
  198. data/vendor/jsclasses/jsclasses.ins +15 -5
  199. data/vendor/jsclasses/jsclasses.pdf +0 -0
  200. data/vendor/jsclasses/jslogo.dtx +4 -4
  201. data/vendor/jsclasses/jslogo.ins +9 -0
  202. data/vendor/jsclasses/jslogo.pdf +0 -0
  203. data/vendor/jsclasses/jslogo.sty +4 -16
  204. data/vendor/jsclasses/jspf.cls +73 -30
  205. data/vendor/jsclasses/jsreport.cls +74 -31
  206. data/vendor/jsclasses/jsverb.ins +9 -0
  207. data/vendor/jsclasses/jsverb.pdf +0 -0
  208. data/vendor/jsclasses/jsverb.sty +1 -13
  209. data/vendor/jsclasses/kiyou.cls +74 -31
  210. data/vendor/jsclasses/minijs.sty +68 -22
  211. data/vendor/jsclasses/okumacro.dtx +4 -5
  212. data/vendor/jsclasses/okumacro.ins +9 -0
  213. data/vendor/jsclasses/okumacro.pdf +0 -0
  214. data/vendor/jsclasses/okumacro.sty +4 -17
  215. data/vendor/jsclasses/okuverb.ins +9 -0
  216. data/vendor/jsclasses/okuverb.pdf +0 -0
  217. data/vendor/jsclasses/okuverb.sty +1 -13
  218. data/vendor/jsclasses/tests/relfont.tex +10 -0
  219. data/vendor/jsclasses/winjis.sty +23 -19
  220. metadata +65 -12
  221. data/.rubocop_todo.yml +0 -7
  222. data/lib/review/book/compilable.rb +0 -173
  223. data/lib/review/tocparser.rb +0 -271
  224. data/samples/syntax-book/review-ext.rb +0 -14
  225. data/test/test_tocparser.rb +0 -25
@@ -6,7 +6,7 @@ module ReVIEW
6
6
  n = 0
7
7
  while line = gets
8
8
  unless line.strip =~ /\A\#@/
9
- ungets line
9
+ ungets(line)
10
10
  return n
11
11
  end
12
12
  n += 1
@@ -0,0 +1,32 @@
1
+ # Copyright (c) 2009-2019 Minero Aoki, Kenshi Muto, Masayoshi Takahashi
2
+ # Copyright (c) 2002-2007 Minero Aoki
3
+ #
4
+ # This program is free software.
5
+ # You can distribute or modify this program under the terms of
6
+ # the GNU LGPL, Lesser General Public License version 2.1.
7
+ #
8
+
9
+ module ReVIEW
10
+ class Location
11
+ def initialize(filename, f)
12
+ @filename = filename
13
+ @f = f
14
+ end
15
+
16
+ attr_reader :filename
17
+
18
+ def lineno
19
+ @f.lineno
20
+ end
21
+
22
+ def string
23
+ begin
24
+ "#{@filename}:#{@f.lineno}"
25
+ rescue
26
+ "#{@filename}:nil"
27
+ end
28
+ end
29
+
30
+ alias_method :to_s, :string
31
+ end
32
+ end
@@ -2,18 +2,14 @@ require 'logger'
2
2
 
3
3
  module ReVIEW
4
4
  class Logger < ::Logger
5
- def initialize(*logdev)
6
- if logdev.empty?
7
- super(STDERR)
8
- self.formatter = ->(severity, _datetime, _progname, msg) { "#{severity}: #{msg}\n" }
9
- else
10
- super
11
- end
5
+ def initialize(io = $stderr, progname: '--')
6
+ super(io, progname: progname)
7
+ self.formatter = ->(severity, _datetime, name, msg) { "#{severity} #{name}: #{msg}\n" }
12
8
  end
13
9
  end
14
10
 
15
11
  def self.logger
16
- @logger ||= ReVIEW::Logger.new
12
+ @logger ||= ReVIEW::Logger.new($stderr, progname: File.basename($PROGRAM_NAME, '.*'))
17
13
  end
18
14
 
19
15
  def self.logger=(logger)
@@ -1,4 +1,4 @@
1
- # Copyright (c) 2012-2018 Yuto HAYAMIZU, Kenshi Muto
1
+ # Copyright (c) 2012-2020 Yuto HAYAMIZU, Kenshi Muto
2
2
  #
3
3
  # This program is free software.
4
4
  # You can distribute or modify this program under the terms of
@@ -10,6 +10,12 @@ require 'fileutils'
10
10
  require 'yaml'
11
11
  require 'shellwords'
12
12
 
13
+ begin
14
+ require 'cgi/escape'
15
+ rescue
16
+ require 'cgi/util'
17
+ end
18
+
13
19
  module ReVIEW
14
20
  module MakerHelper
15
21
  # Return review/bin directory
@@ -18,6 +24,11 @@ module ReVIEW
18
24
  end
19
25
  module_function :bindir
20
26
 
27
+ def h(str)
28
+ CGI.escapeHTML(str)
29
+ end
30
+ module_function :h
31
+
21
32
  # Copy image files under from_dir to to_dir recursively
22
33
  # ==== Args
23
34
  # from_dir :: path to the directory which has image files to be copied
@@ -55,7 +66,7 @@ module ReVIEW
55
66
  exts = options[:exts] || %w[png gif jpg jpeg svg pdf eps ai tif psd]
56
67
  exts_str = exts.join('|')
57
68
  if !is_converted && fname =~ /\.(#{exts_str})$/i
58
- FileUtils.cp "#{from_dir}/#{fname}", to_dir
69
+ FileUtils.cp("#{from_dir}/#{fname}", to_dir)
59
70
  image_files << "#{from_dir}/#{fname}"
60
71
  end
61
72
  end
@@ -66,8 +77,8 @@ module ReVIEW
66
77
  end
67
78
  module_function :copy_images_to_dir
68
79
 
69
- def cleanup_mathimg
70
- math_dir = "./#{@config['imagedir']}/_review_math"
80
+ def cleanup_mathimg(path = '_review_math')
81
+ math_dir = "./#{@config['imagedir']}/#{path}"
71
82
  if @config['imgmath'] && Dir.exist?(math_dir)
72
83
  FileUtils.rm_rf(math_dir)
73
84
  end
@@ -112,6 +123,22 @@ module ReVIEW
112
123
  \\end{document}
113
124
  EOB
114
125
 
126
+ hashes = File.readlines(File.join(math_dir, '__IMGMATH_BODY__.map')).sort.uniq
127
+ File.write(File.join(math_dir, '__IMGMATH_BODY__.map'), hashes.join)
128
+
129
+ File.open(File.join(math_dir, '__IMGMATH_BODY__.tex'), 'w') do |f|
130
+ File.open(File.join(math_dir, '__IMGMATH_BODY__.map')) do |map|
131
+ map.each_line do |l|
132
+ l.chomp!
133
+ f.puts "% #{l}"
134
+ f.puts File.read(File.join(math_dir, "__IMGMATH_BODY__#{l}.tex"))
135
+ File.unlink(File.join(math_dir, "__IMGMATH_BODY__#{l}.tex"))
136
+ f.puts '\\clearpage'
137
+ f.puts
138
+ end
139
+ end
140
+ end
141
+
115
142
  math_dir = File.realpath(math_dir)
116
143
  Dir.mktmpdir do |tmpdir|
117
144
  FileUtils.cp([File.join(math_dir, '__IMGMATH_BODY__.tex'),
@@ -139,8 +166,10 @@ EOB
139
166
  FileUtils.rm_f([File.join(math_dir, '__IMGMATH_BODY__.tex'),
140
167
  File.join(math_dir, '__IMGMATH_BODY__.map')])
141
168
  end
169
+ module_function :make_math_images
142
170
 
143
171
  def make_math_images_pdfcrop(dir, tex_path, math_dir)
172
+ # rubocop:disable Metrics/BlockLength
144
173
  Dir.chdir(dir) do
145
174
  dvi_path = '__IMGMATH__.dvi'
146
175
  pdf_path = '__IMGMATH__.pdf'
@@ -155,7 +184,6 @@ EOB
155
184
  raise CompileError
156
185
  end
157
186
  end
158
-
159
187
  args = @config['imgmath_options']['pdfcrop_cmd'].shellsplit
160
188
  args.map! do |m|
161
189
  m.sub('%i', pdf_path).
@@ -201,6 +229,7 @@ EOB
201
229
  args = @config['imgmath_options']['pdfcrop_pixelize_cmd'].shellsplit
202
230
  args.map! do |m|
203
231
  m.sub('%i', pdf_path2).
232
+ sub('%t', @config['imgmath_options']['format']).
204
233
  sub('%o', File.join(math_dir, "_gen_#{key}.#{@config['imgmath_options']['format']}")).
205
234
  sub('%O', File.join(math_dir, "_gen_#{key}")).
206
235
  sub('%p', page.to_s)
@@ -213,6 +242,7 @@ EOB
213
242
  end
214
243
  end
215
244
  end
245
+ # rubocop:enable Metrics/BlockLength
216
246
  end
217
247
 
218
248
  def make_math_images_dvipng(dir, tex_path, math_dir)
@@ -1,3 +1,5 @@
1
+ # Copyright (c) 2013-2019 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,30 +65,39 @@ 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
 
85
+ def listnum_body(lines, lang)
86
+ lang ||= ''
87
+ puts "```#{lang}"
88
+ lines.each_with_index do |line, i|
89
+ puts((i + 1).to_s.rjust(2) + ": #{detab(line)}")
90
+ end
91
+ puts '```'
92
+ end
93
+
83
94
  def ul_begin
84
95
  blank if @ul_indent == 0
85
96
  @ul_indent += 1
86
97
  end
87
98
 
88
99
  def ul_item_begin(lines)
89
- puts ' ' * (@ul_indent - 1) + '* ' + lines.join
100
+ puts ' ' * (@ul_indent - 1) + '* ' + join_lines_to_paragraph(lines)
90
101
  end
91
102
 
92
103
  def ul_item_end
@@ -102,7 +113,7 @@ module ReVIEW
102
113
  end
103
114
 
104
115
  def ol_item(lines, num)
105
- puts "#{num}. #{lines.join}"
116
+ puts "#{num}. #{join_lines_to_paragraph(lines)}"
106
117
  end
107
118
 
108
119
  def ol_end
@@ -118,7 +129,7 @@ module ReVIEW
118
129
  end
119
130
 
120
131
  def dd(lines)
121
- puts "<dd>#{lines.join}</dd>"
132
+ puts "<dd>#{join_lines_to_paragraph(lines)}</dd>"
122
133
  end
123
134
 
124
135
  def dl_end
@@ -127,9 +138,9 @@ module ReVIEW
127
138
 
128
139
  def emlist(lines, caption = nil, lang = nil)
129
140
  blank
130
- if caption
141
+ if caption.present?
131
142
  puts caption
132
- print "\n"
143
+ blank
133
144
  end
134
145
  lang ||= ''
135
146
  puts "```#{lang}"
@@ -144,10 +155,28 @@ module ReVIEW
144
155
  puts %Q(<div class="#{type}">)
145
156
  puts %Q(<p class="caption">#{compile_inline(caption)}</p>) if caption.present?
146
157
  blocked_lines = split_paragraph(lines)
147
- puts blocked_lines.join("\n")
158
+ puts blocked_lines.join("\n\n")
148
159
  puts '</div>'
149
160
  end
150
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
+
151
180
  def hr
152
181
  puts '----'
153
182
  end
@@ -229,7 +258,11 @@ module ReVIEW
229
258
  'jpg'
230
259
  end
231
260
 
232
- def cmd(lines)
261
+ def cmd(lines, caption = nil)
262
+ if caption.present?
263
+ puts caption
264
+ blank
265
+ end
233
266
  puts '```shell-session'
234
267
  lines.each do |line|
235
268
  puts detab(line)
@@ -237,32 +270,12 @@ module ReVIEW
237
270
  puts '```'
238
271
  end
239
272
 
240
- def table(lines, id = nil, caption = nil)
241
- rows = []
242
- sepidx = nil
243
- lines.each_with_index do |line, idx|
244
- if /\A[\=\-]{12}/ =~ line
245
- # just ignore
246
- # error "too many table separator" if sepidx
247
- sepidx ||= idx
248
- next
249
- end
250
- rows.push(line.strip.split(/\t+/).map { |s| s.sub(/\A\./, '') })
251
- end
252
- rows = adjust_n_cols(rows)
253
-
254
- begin
255
- table_header id, caption unless caption.nil?
256
- rescue KeyError
257
- error "no such table: #{id}"
258
- end
259
- table_begin rows.first.size
260
- return if rows.empty?
273
+ def table_rows(sepidx, rows)
261
274
  if sepidx
262
275
  sepidx.times do
263
276
  tr(rows.shift.map { |s| th(s) })
264
277
  end
265
- table_border rows.first.size
278
+ table_border(rows.first.size)
266
279
  rows.each do |cols|
267
280
  tr(cols.map { |s| td(s) })
268
281
  end
@@ -272,7 +285,6 @@ module ReVIEW
272
285
  tr([th(h)] + cs.map { |s| td(s) })
273
286
  end
274
287
  end
275
- table_end
276
288
  end
277
289
 
278
290
  def table_header(id, caption)
@@ -348,7 +360,7 @@ module ReVIEW
348
360
  return unless @book.config['draft']
349
361
  lines ||= []
350
362
  unless comment.blank?
351
- lines.unshift comment
363
+ lines.unshift(comment)
352
364
  end
353
365
  str = lines.join('<br />')
354
366
  puts %Q(<div class="red">#{escape(str)}</div>)
@@ -373,7 +385,7 @@ module ReVIEW
373
385
 
374
386
  def flushright(lines)
375
387
  puts %Q(<div class="flushright">)
376
- puts lines.join
388
+ puts split_paragraph(lines).join("\n")
377
389
  puts %Q(</div>)
378
390
  end
379
391
  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-2020 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
@@ -54,12 +54,12 @@ module ReVIEW
54
54
  end
55
55
 
56
56
  def error(msg)
57
- @logger.error "#{File.basename($PROGRAM_NAME, '.*')}: #{msg}"
57
+ @logger.error msg
58
58
  exit 1
59
59
  end
60
60
 
61
61
  def warn(msg)
62
- @logger.warn "#{File.basename($PROGRAM_NAME, '.*')}: #{msg}"
62
+ @logger.warn msg
63
63
  end
64
64
 
65
65
  def pdf_filepath
@@ -100,6 +100,7 @@ module ReVIEW
100
100
  def parse_opts(args)
101
101
  cmd_config = {}
102
102
  opts = OptionParser.new
103
+ @buildonly = nil
103
104
 
104
105
  opts.banner = 'Usage: review-pdfmaker configfile'
105
106
  opts.version = ReVIEW::VERSION
@@ -109,6 +110,7 @@ module ReVIEW
109
110
  end
110
111
  opts.on('--[no-]debug', 'Keep temporary files.') { |debug| cmd_config['debug'] = debug }
111
112
  opts.on('--ignore-errors', 'Ignore review-compile errors.') { cmd_config['ignore-errors'] = true }
113
+ opts.on('-y', '--only file1,file2,...', 'Build only specified files.') { |v| @buildonly = v.split(/\s*,\s*/).map { |m| m.strip.sub(/\.re\Z/, '') } }
112
114
 
113
115
  opts.parse!(args)
114
116
  if args.size != 1
@@ -120,20 +122,13 @@ module ReVIEW
120
122
  end
121
123
 
122
124
  def execute(*args)
123
- @config = ReVIEW::Configure.values
124
- @config.maker = 'pdfmaker'
125
125
  cmd_config, yamlfile = parse_opts(args)
126
126
  error "#{yamlfile} not found." unless File.exist?(yamlfile)
127
127
 
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
128
+ @config = ReVIEW::Configure.create(maker: 'pdfmaker',
129
+ yamlfile: yamlfile,
130
+ config: cmd_config)
134
131
 
135
- # YAML configs will be overridden by command line options.
136
- @config.deep_merge!(cmd_config)
137
132
  I18n.setup(@config['language'])
138
133
  @basedir = File.absolute_path(File.dirname(yamlfile))
139
134
 
@@ -153,21 +148,26 @@ module ReVIEW
153
148
  end
154
149
 
155
150
  begin
156
- generate_pdf(yamlfile)
151
+ generate_pdf
157
152
  rescue ApplicationError => e
158
153
  raise if @config['debug']
159
154
  error(e.message)
160
155
  end
161
156
  end
162
157
 
163
- def make_input_files(book, yamlfile)
158
+ def make_input_files(book)
164
159
  input_files = Hash.new { |h, key| h[key] = '' }
165
160
  book.parts.each do |part|
166
161
  if part.name.present?
167
162
  @config['use_part'] = true
168
163
  if part.file?
169
- output_chaps(part.name, yamlfile)
170
- input_files['CHAPS'] << %Q(\\input{#{part.name}.tex}\n)
164
+ if @buildonly && !@buildonly.include?(part.name)
165
+ warn "skip #{part.name}.re"
166
+ input_files['CHAPS'] << %Q(\\part{}\n)
167
+ else
168
+ output_chaps(part.name)
169
+ input_files['CHAPS'] << %Q(\\input{#{part.name}.tex}\n)
170
+ end
171
171
  else
172
172
  input_files['CHAPS'] << %Q(\\part{#{part.name}}\n)
173
173
  end
@@ -175,11 +175,18 @@ module ReVIEW
175
175
 
176
176
  part.chapters.each do |chap|
177
177
  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?
178
+ entry = "\\input{#{filename}.tex}\n"
179
+ if @buildonly && !@buildonly.include?(filename)
180
+ warn "skip #{filename}.re"
181
+ entry = "\\chapter{}\n"
182
+ else
183
+ output_chaps(filename)
184
+ end
185
+
186
+ input_files['PREDEF'] << entry if chap.on_predef?
187
+ input_files['CHAPS'] << entry if chap.on_chaps?
188
+ input_files['APPENDIX'] << entry if chap.on_appendix?
189
+ input_files['POSTDEF'] << entry if chap.on_postdef?
183
190
  end
184
191
  end
185
192
 
@@ -189,7 +196,7 @@ module ReVIEW
189
196
  def build_pdf
190
197
  template = template_content
191
198
  Dir.chdir(@path) do
192
- File.open("./#{@mastertex}.tex", 'wb') { |f| f.write(template) }
199
+ File.open("./#{@mastertex}.tex", 'wb') { |f| f.write template }
193
200
 
194
201
  call_hook('hook_beforetexcompile')
195
202
 
@@ -235,8 +242,9 @@ module ReVIEW
235
242
  end
236
243
 
237
244
  call_hook('hook_beforemakeindex')
238
- if @config['pdfmaker']['makeindex'] && File.exist?("#{@mastertex}.idx")
245
+ if @config['pdfmaker']['makeindex'] && File.size?("#{@mastertex}.idx")
239
246
  system_or_raise(*[makeindex_command, makeindex_options, @mastertex].flatten.compact)
247
+ system_or_raise(*[texcommand, texoptions, "#{@mastertex}.tex"].flatten.compact)
240
248
  end
241
249
  call_hook('hook_aftermakeindex')
242
250
 
@@ -250,18 +258,17 @@ module ReVIEW
250
258
  end
251
259
  end
252
260
 
253
- def generate_pdf(yamlfile)
261
+ def generate_pdf
254
262
  remove_old_file
255
263
  erb_config
256
264
  @path = build_path
257
265
  begin
258
266
  @compile_errors = nil
259
267
 
260
- book = ReVIEW::Book.load(File.dirname(yamlfile))
261
- book.config = @config
268
+ book = ReVIEW::Book::Base.new(@basedir, config: @config)
262
269
  @converter = ReVIEW::Converter.new(book, ReVIEW::LATEXBuilder.new)
263
270
 
264
- @input_files = make_input_files(book, yamlfile)
271
+ @input_files = make_input_files(book)
265
272
 
266
273
  check_compile_status(@config['ignore-errors'])
267
274
 
@@ -281,11 +288,11 @@ module ReVIEW
281
288
 
282
289
  FileUtils.cp(File.join(@path, "#{@mastertex}.pdf"), pdf_filepath)
283
290
  ensure
284
- remove_entry_secure @path unless @config['debug']
291
+ remove_entry_secure(@path) unless @config['debug']
285
292
  end
286
293
  end
287
294
 
288
- def output_chaps(filename, _yamlfile)
295
+ def output_chaps(filename)
289
296
  @logger.info "compiling #{filename}.tex"
290
297
  begin
291
298
  @converter.convert(filename + '.re', File.join(@path, filename + '.tex'))
@@ -296,23 +303,12 @@ module ReVIEW
296
303
  end
297
304
  end
298
305
 
299
- # PDFMaker#copy_images should copy image files _AND_ execute extractbb (or ebb).
306
+ # PDFMaker#copy_images should copy image files
300
307
  #
301
308
  def copy_images(from, to)
302
309
  return unless File.exist?(from)
303
310
  Dir.mkdir(to)
304
311
  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
312
  end
317
313
 
318
314
  def make_custom_page(file)
@@ -370,11 +366,11 @@ module ReVIEW
370
366
  items.each_with_index do |item, rev|
371
367
  editstr = edit == 0 ? ReVIEW::I18n.t('first_edition') : ReVIEW::I18n.t('nth_edition', (edit + 1).to_s)
372
368
  revstr = ReVIEW::I18n.t('nth_impression', (rev + 1).to_s)
373
- if item =~ /\A\d+\-\d+\-\d+\Z/
369
+ if item =~ /\A\d+-\d+-\d+\Z/
374
370
  buf << ReVIEW::I18n.t('published_by1', [date_to_s(item), editstr + revstr])
375
- elsif item =~ /\A(\d+\-\d+\-\d+)[\s ](.+)/
371
+ elsif item =~ /\A(\d+-\d+-\d+)[\s ](.+)/
376
372
  # 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]]) }
373
+ item.match(/\A(\d+-\d+-\d+)[\s ](.+)/) { |m| buf << ReVIEW::I18n.t('published_by3', [date_to_s(m[1]), m[2]]) }
378
374
  else
379
375
  # free format
380
376
  buf << item
@@ -440,26 +436,33 @@ module ReVIEW
440
436
  end
441
437
 
442
438
  @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]
439
+ part_tuple = I18n.get('part').split(/%[A-Za-z]{1,3}/, 2)
440
+ chapter_tuple = I18n.get('chapter').split(/%[A-Za-z]{1,3}/, 2)
441
+ appendix_tuple = I18n.get('appendix').split(/%[A-Za-z]{1,3}/, 2)
442
+ @locale_latex['prepartname'] = part_tuple[0].to_s
443
+ @locale_latex['postpartname'] = part_tuple[1].to_s
444
+ @locale_latex['prechaptername'] = chapter_tuple[0].to_s
445
+ @locale_latex['postchaptername'] = chapter_tuple[1].to_s
446
+ @locale_latex['preappendixname'] = appendix_tuple[0].to_s
447
+ @locale_latex['postappendixname'] = appendix_tuple[1].to_s
452
448
  end
453
449
 
454
450
  def erb_content(file)
455
451
  @texcompiler = File.basename(@config['texcommand'], '.*')
456
452
  erb = ReVIEW::Template.load(file, '-')
457
- @logger.debug "erb processes #{File.basename(file)}" if @config['debug']
453
+ @logger.debug("erb processes #{File.basename(file)}") if @config['debug']
458
454
  erb.result(binding)
459
455
  end
460
456
 
461
457
  def latex_config
462
- erb_content(File.expand_path('./latex/config.erb', ReVIEW::Template::TEMPLATE_DIR))
458
+ result = erb_content(File.expand_path('./latex/config.erb', ReVIEW::Template::TEMPLATE_DIR))
459
+ local_config_file = File.join(@basedir, 'layouts', 'config-local.tex.erb')
460
+ if File.exist?(local_config_file)
461
+ result << "%% BEGIN: config-local.tex.erb\n"
462
+ result << erb_content(local_config_file)
463
+ result << "%% END: config-local.tex.erb\n"
464
+ end
465
+ result
463
466
  end
464
467
 
465
468
  def template_content
@@ -489,7 +492,7 @@ module ReVIEW
489
492
  f.print erb_content(File.join(dirname, fname))
490
493
  end
491
494
  else
492
- FileUtils.cp File.join(dirname, fname), copybase
495
+ FileUtils.cp(File.join(dirname, fname), copybase)
493
496
  end
494
497
  end
495
498
  end