annotate .svn/pristine/99/99a04119c4a137489a9121a24c845e6c4a694597.svn-base @ 1628:9c5f8e24dadc live tip

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