annotate vendor/plugins/rfpdf/lib/fpdf/japanese.rb @ 507:0c939c159af4 redmine-1.2

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