##// END OF EJS Templates
PDF: back out r5253 (#7794, #61)....
Toshi MARUYAMA -
r5161:516320e82b0f
parent child
Show More
@@ -1,464 +1,464
1 1 # Copyright (c) 2006 4ssoM LLC <www.4ssoM.com>
2 2 # 1.12 contributed by Ed Moss.
3 3 #
4 4 # The MIT License
5 5 #
6 6 # Permission is hereby granted, free of charge, to any person obtaining a copy
7 7 # of this software and associated documentation files (the "Software"), to deal
8 8 # in the Software without restriction, including without limitation the rights
9 9 # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 10 # copies of the Software, and to permit persons to whom the Software is
11 11 # furnished to do so, subject to the following conditions:
12 12 #
13 13 # The above copyright notice and this permission notice shall be included in
14 14 # all copies or substantial portions of the Software.
15 15 #
16 16 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 17 # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 18 # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 19 # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 20 # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 21 # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 22 # THE SOFTWARE.
23 23 #
24 24 # This is direct port of japanese.php
25 25 #
26 26 # Japanese PDF support.
27 27 #
28 28 # Usage is as follows:
29 29 #
30 30 # require 'fpdf'
31 31 # require 'chinese'
32 32 # pdf = FPDF.new
33 33 # pdf.extend(PDF_Japanese)
34 34 #
35 35 # This allows it to be combined with other extensions, such as the bookmark
36 36 # module.
37 37
38 38 module PDF_Japanese
39 CONTROL_CHARACTERS = (0x00...0x20).inject({}){|map, c| map[c.chr] = 278; map }.freeze
39
40 40 SJIS_widths={' ' => 278, '!' => 299, '"' => 353, '#' => 614, '$' => 614, '%' => 721, '&' => 735, '\'' => 216,
41 41 '(' => 323, ')' => 323, '*' => 449, '+' => 529, ',' => 219, '-' => 306, '.' => 219, '/' => 453, '0' => 614, '1' => 614,
42 42 '2' => 614, '3' => 614, '4' => 614, '5' => 614, '6' => 614, '7' => 614, '8' => 614, '9' => 614, ':' => 219, ';' => 219,
43 43 '<' => 529, '=' => 529, '>' => 529, '?' => 486, '@' => 744, 'A' => 646, 'B' => 604, 'C' => 617, 'D' => 681, 'E' => 567,
44 44 'F' => 537, 'G' => 647, 'H' => 738, 'I' => 320, 'J' => 433, 'K' => 637, 'L' => 566, 'M' => 904, 'N' => 710, 'O' => 716,
45 45 'P' => 605, 'Q' => 716, 'R' => 623, 'S' => 517, 'T' => 601, 'U' => 690, 'V' => 668, 'W' => 990, 'X' => 681, 'Y' => 634,
46 46 'Z' => 578, '[' => 316, '\\' => 614, ']' => 316, '^' => 529, '_' => 500, '`' => 387, 'a' => 509, 'b' => 566, 'c' => 478,
47 47 'd' => 565, 'e' => 503, 'f' => 337, 'g' => 549, 'h' => 580, 'i' => 275, 'j' => 266, 'k' => 544, 'l' => 276, 'm' => 854,
48 48 'n' => 579, 'o' => 550, 'p' => 578, 'q' => 566, 'r' => 410, 's' => 444, 't' => 340, 'u' => 575, 'v' => 512, 'w' => 760,
49 'x' => 503, 'y' => 529, 'z' => 453, '{' => 326, '|' => 380, '}' => 326, '~' => 387}.merge(CONTROL_CHARACTERS).freeze
49 'x' => 503, 'y' => 529, 'z' => 453, '{' => 326, '|' => 380, '}' => 326, '~' => 387}
50 50
51 51 def AddCIDFont(family,style,name,cw,cMap,registry)
52 52 fontkey=family.downcase+style.upcase
53 53 unless @fonts[fontkey].nil?
54 54 Error("CID font already added: family style")
55 55 end
56 56 i=@fonts.length+1
57 57 @fonts[fontkey]={'i'=>i,'type'=>'Type0','name'=>name,'up'=>-120,'ut'=>40,'cw'=>cw,
58 58 'CMap'=>cMap,'registry'=>registry}
59 59 end
60 60
61 61 def AddCIDFonts(family,name,cw,cMap,registry)
62 62 AddCIDFont(family,'',name,cw,cMap,registry)
63 63 AddCIDFont(family,'B',name+',Bold',cw,cMap,registry)
64 64 AddCIDFont(family,'I',name+',Italic',cw,cMap,registry)
65 65 AddCIDFont(family,'BI',name+',BoldItalic',cw,cMap,registry)
66 66 end
67 67
68 68 def AddSJISFont(family='SJIS')
69 69 #Add SJIS font with proportional Latin
70 70 name='KozMinPro-Regular-Acro'
71 71 cw=SJIS_widths
72 72 cMap='90msp-RKSJ-H'
73 73 registry={'ordering'=>'Japan1','supplement'=>2}
74 74 AddCIDFonts(family,name,cw,cMap,registry)
75 75 end
76 76
77 77 def AddSJIShwFont(family='SJIS-hw')
78 78 #Add SJIS font with half-width Latin
79 79 name='KozMinPro-Regular-Acro'
80 80 32.upto(126) do |i|
81 81 cw[i.chr]=500
82 82 end
83 83 cMap='90ms-RKSJ-H'
84 84 registry={'ordering'=>'Japan1','supplement'=>2}
85 85 AddCIDFonts(family,name,cw,cMap,registry)
86 86 end
87 87
88 88 def GetStringWidth(s)
89 89 if(@CurrentFont['type']=='Type0')
90 90 return GetSJISStringWidth(s)
91 91 else
92 92 return super(s)
93 93 end
94 94 end
95 95
96 96 def GetSJISStringWidth(s)
97 97 #SJIS version of GetStringWidth()
98 98 l=0
99 99 cw=@CurrentFont['cw']
100 100 nb=s.length
101 101 i=0
102 102 while(i<nb)
103 103 o=s[i]
104 104 if(o<128)
105 105 #ASCII
106 106 l+=cw[o.chr] if cw[o.chr]
107 107 i+=1
108 108 elsif(o>=161 and o<=223)
109 109 #Half-width katakana
110 110 l+=500
111 111 i+=1
112 112 else
113 113 #Full-width character
114 114 l+=1000
115 115 i+=2
116 116 end
117 117 end
118 118 return l*@FontSize/1000
119 119 end
120 120
121 121 def MultiCell(w,h,txt,border=0,align='L',fill=0)
122 122 if(@CurrentFont['type']=='Type0')
123 123 SJISMultiCell(w,h,txt,border,align,fill)
124 124 else
125 125 super(w,h,txt,border,align,fill)
126 126 end
127 127 end
128 128
129 129 def SJISMultiCell(w,h,txt,border=0,align='L',fill=0)
130 130 #Output text with automatic or explicit line breaks
131 131 cw=@CurrentFont['cw']
132 132 if(w==0)
133 133 w=@w-@rMargin-@x
134 134 end
135 135 wmax=(w-2*@cMargin)*1000/@FontSize
136 136 s=txt.gsub("\r",'')
137 137 nb=s.length
138 138 if(nb>0 and s[nb-1]=="\n")
139 139 nb-=1
140 140 end
141 141 b=0
142 142 if(border)
143 143 if(border==1)
144 144 border='LTRB'
145 145 b='LRT'
146 146 b2='LR'
147 147 else
148 148 b2=''
149 149 b2='L' unless border.to_s.index('L').nil?
150 150 b2=b2+'R' unless border.to_s.index('R').nil?
151 151 b=(border.to_s.index('T')) ? (b2+'T') : b2
152 152 end
153 153 end
154 154 sep=-1
155 155 i=0
156 156 j=0
157 157 l=0
158 158 nl=1
159 159 while(i<nb)
160 160 #Get next character
161 161 c=s[i]
162 162 o=c #o=ord(c)
163 163 if(o==10)
164 164 #Explicit line break
165 165 Cell(w,h,s[j,i-j],b,2,align,fill)
166 166 i+=1
167 167 sep=-1
168 168 j=i
169 169 l=0
170 170 nl+=1
171 171 if(border and nl==2)
172 172 b=b2
173 173 end
174 174 next
175 175 end
176 176 if(o<128)
177 177 #ASCII
178 178 l+=cw[c.chr] || 0
179 179 n=1
180 180 if(o==32)
181 181 sep=i
182 182 end
183 183 elsif(o>=161 and o<=223)
184 184 #Half-width katakana
185 185 l+=500
186 186 n=1
187 187 sep=i
188 188 else
189 189 #Full-width character
190 190 l+=1000
191 191 n=2
192 192 sep=i
193 193 end
194 194 if(l>wmax)
195 195 #Automatic line break
196 196 if(sep==-1 or i==j)
197 197 if(i==j)
198 198 i+=n
199 199 end
200 200 Cell(w,h,s[j,i-j],b,2,align,fill)
201 201 else
202 202 Cell(w,h,s[j,sep-j],b,2,align,fill)
203 203 i=(s[sep].chr==' ') ? sep+1 : sep
204 204 end
205 205 sep=-1
206 206 j=i
207 207 l=0
208 208 nl+=1
209 209 if(border and nl==2)
210 210 b=b2
211 211 end
212 212 else
213 213 i+=n
214 214 if(o>=128)
215 215 sep=i
216 216 end
217 217 end
218 218 end
219 219 #Last chunk
220 220 if(border and not border.to_s.index('B').nil?)
221 221 b+='B'
222 222 end
223 223 Cell(w,h,s[j,i-j],b,2,align,fill)
224 224 @x=@lMargin
225 225 end
226 226
227 227 def Write(h,txt,link='')
228 228 if(@CurrentFont['type']=='Type0')
229 229 SJISWrite(h,txt,link)
230 230 else
231 231 super(h,txt,link)
232 232 end
233 233 end
234 234
235 235 def SJISWrite(h,txt,link)
236 236 #SJIS version of Write()
237 237 cw=@CurrentFont['cw']
238 238 w=@w-@rMargin-@x
239 239 wmax=(w-2*@cMargin)*1000/@FontSize
240 240 s=txt.gsub("\r",'')
241 241 nb=s.length
242 242 sep=-1
243 243 i=0
244 244 j=0
245 245 l=0
246 246 nl=1
247 247 while(i<nb)
248 248 #Get next character
249 249 c=s[i]
250 250 o=c
251 251 if(o==10)
252 252 #Explicit line break
253 253 Cell(w,h,s[j,i-j],0,2,'',0,link)
254 254 i+=1
255 255 sep=-1
256 256 j=i
257 257 l=0
258 258 if(nl==1)
259 259 #Go to left margin
260 260 @x=@lMargin
261 261 w=@w-@rMargin-@x
262 262 wmax=(w-2*@cMargin)*1000/@FontSize
263 263 end
264 264 nl+=1
265 265 next
266 266 end
267 267 if(o<128)
268 268 #ASCII
269 269 l+=cw[c.chr] || 0
270 270 n=1
271 271 if(o==32)
272 272 sep=i
273 273 end
274 274 elsif(o>=161 and o<=223)
275 275 #Half-width katakana
276 276 l+=500
277 277 n=1
278 278 sep=i
279 279 else
280 280 #Full-width character
281 281 l+=1000
282 282 n=2
283 283 sep=i
284 284 end
285 285 if(l>wmax)
286 286 #Automatic line break
287 287 if(sep==-1 or i==j)
288 288 if(@x>@lMargin)
289 289 #Move to next line
290 290 @x=@lMargin
291 291 @y+=h
292 292 w=@w-@rMargin-@x
293 293 wmax=(w-2*@cMargin)*1000/@FontSize
294 294 i+=n
295 295 nl+=1
296 296 next
297 297 end
298 298 if(i==j)
299 299 i+=n
300 300 end
301 301 Cell(w,h,s[j,i-j],0,2,'',0,link)
302 302 else
303 303 Cell(w,h,s[j,sep-j],0,2,'',0,link)
304 304 i=(s[sep].chr==' ') ? sep+1 : sep
305 305 end
306 306 sep=-1
307 307 j=i
308 308 l=0
309 309 if(nl==1)
310 310 @x=@lMargin
311 311 w=@w-@rMargin-@x
312 312 wmax=(w-2*@cMargin)*1000/@FontSize
313 313 end
314 314 nl+=1
315 315 else
316 316 i+=n
317 317 if(o>=128)
318 318 sep=i
319 319 end
320 320 end
321 321 end
322 322 #Last chunk
323 323 if(i!=j)
324 324 Cell(l/1000*@FontSize,h,s[j,i-j],0,0,'',0,link)
325 325 end
326 326 end
327 327
328 328 private
329 329
330 330 def putfonts()
331 331 nf=@n
332 332 @diffs.each do |diff|
333 333 #Encodings
334 334 newobj()
335 335 out('<</Type /Encoding /BaseEncoding /WinAnsiEncoding /Differences ['+diff+']>>')
336 336 out('endobj')
337 337 end
338 338 # mqr=get_magic_quotes_runtime()
339 339 # set_magic_quotes_runtime(0)
340 340 @FontFiles.each_pair do |file, info|
341 341 #Font file embedding
342 342 newobj()
343 343 @FontFiles[file]['n']=@n
344 344 if(defined('FPDF_FONTPATH'))
345 345 file=FPDF_FONTPATH+file
346 346 end
347 347 size=filesize(file)
348 348 if(!size)
349 349 Error('Font file not found')
350 350 end
351 351 out('<</Length '+size)
352 352 if(file[-2]=='.z')
353 353 out('/Filter /FlateDecode')
354 354 end
355 355 out('/Length1 '+info['length1'])
356 356 unless info['length2'].nil?
357 357 out('/Length2 '+info['length2']+' /Length3 0')
358 358 end
359 359 out('>>')
360 360 f=fopen(file,'rb')
361 361 putstream(fread(f,size))
362 362 fclose(f)
363 363 out('endobj')
364 364 end
365 365 # set_magic_quotes_runtime(mqr)
366 366 @fonts.each_pair do |k, font|
367 367 #Font objects
368 368 newobj()
369 369 @fonts[k]['n']=@n
370 370 out('<</Type /Font')
371 371 if(font['type']=='Type0')
372 372 putType0(font)
373 373 else
374 374 name=font['name']
375 375 out('/BaseFont /'+name)
376 376 if(font['type']=='core')
377 377 #Standard font
378 378 out('/Subtype /Type1')
379 379 if(name!='Symbol' and name!='ZapfDingbats')
380 380 out('/Encoding /WinAnsiEncoding')
381 381 end
382 382 else
383 383 #Additional font
384 384 out('/Subtype /'+font['type'])
385 385 out('/FirstChar 32')
386 386 out('/LastChar 255')
387 387 out('/Widths '+(@n+1)+' 0 R')
388 388 out('/FontDescriptor '+(@n+2)+' 0 R')
389 389 if(font['enc'])
390 390 if !font['diff'].nil?
391 391 out('/Encoding '+(nf+font['diff'])+' 0 R')
392 392 else
393 393 out('/Encoding /WinAnsiEncoding')
394 394 end
395 395 end
396 396 end
397 397 out('>>')
398 398 out('endobj')
399 399 if(font['type']!='core')
400 400 #Widths
401 401 newobj()
402 402 cw=font['cw']
403 403 s='['
404 404 32.upto(255) do |i|
405 405 s+=cw[i.chr]+' '
406 406 end
407 407 out(s+']')
408 408 out('endobj')
409 409 #Descriptor
410 410 newobj()
411 411 s='<</Type /FontDescriptor /FontName /'+name
412 412 font['desc'].each_pair do |k, v|
413 413 s+=' /'+k+' '+v
414 414 end
415 415 file=font['file']
416 416 if(file)
417 417 s+=' /FontFile'+(font['type']=='Type1' ? '' : '2')+' '+@FontFiles[file]['n']+' 0 R'
418 418 end
419 419 out(s+'>>')
420 420 out('endobj')
421 421 end
422 422 end
423 423 end
424 424 end
425 425
426 426 def putType0(font)
427 427 #Type0
428 428 out('/Subtype /Type0')
429 429 out('/BaseFont /'+font['name']+'-'+font['CMap'])
430 430 out('/Encoding /'+font['CMap'])
431 431 out('/DescendantFonts ['+(@n+1).to_s+' 0 R]')
432 432 out('>>')
433 433 out('endobj')
434 434 #CIDFont
435 435 newobj()
436 436 out('<</Type /Font')
437 437 out('/Subtype /CIDFontType0')
438 438 out('/BaseFont /'+font['name'])
439 439 out('/CIDSystemInfo <</Registry (Adobe) /Ordering ('+font['registry']['ordering']+') /Supplement '+font['registry']['supplement'].to_s+'>>')
440 440 out('/FontDescriptor '+(@n+1).to_s+' 0 R')
441 441 w='/W [1 ['
442 442 font['cw'].keys.sort.each {|key|
443 443 w+=font['cw'][key].to_s + " "
444 444 # ActionController::Base::logger.debug key.to_s
445 445 # ActionController::Base::logger.debug font['cw'][key].to_s
446 446 }
447 447 out(w+'] 231 325 500 631 [500] 326 389 500]')
448 448 out('>>')
449 449 out('endobj')
450 450 #Font descriptor
451 451 newobj()
452 452 out('<</Type /FontDescriptor')
453 453 out('/FontName /'+font['name'])
454 454 out('/Flags 6')
455 455 out('/FontBBox [0 -200 1000 900]')
456 456 out('/ItalicAngle 0')
457 457 out('/Ascent 800')
458 458 out('/Descent -200')
459 459 out('/CapHeight 800')
460 460 out('/StemV 60')
461 461 out('>>')
462 462 out('endobj')
463 463 end
464 464 end
General Comments 0
You need to be logged in to leave comments. Login now