1 /************************************************************************* 2 * 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * Copyright 2000, 2010 Oracle and/or its affiliates. 6 * 7 * OpenOffice.org - a multi-platform office productivity suite 8 * 9 * This file is part of OpenOffice.org. 10 * 11 * OpenOffice.org is free software: you can redistribute it and/or modify 12 * it under the terms of the GNU Lesser General Public License version 3 13 * only, as published by the Free Software Foundation. 14 * 15 * OpenOffice.org is distributed in the hope that it will be useful, 16 * but WITHOUT ANY WARRANTY; without even the implied warranty of 17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 18 * GNU Lesser General Public License version 3 for more details 19 * (a copy is included in the LICENSE file that accompanied this code). 20 * 21 * You should have received a copy of the GNU Lesser General Public License 22 * version 3 along with OpenOffice.org. If not, see 23 * <http://www.openoffice.org/license.html> 24 * for a copy of the LGPLv3 License. 25 * 26 ************************************************************************/ 27 28 // MARKER(update_precomp.py): autogen include statement, do not remove 29 #include "precompiled_vcl.hxx" 30 31 #define _USE_MATH_DEFINES 32 #include <math.h> 33 #include <algorithm> 34 35 #include <tools/urlobj.hxx> 36 37 #include <pdfwriter_impl.hxx> 38 39 #include <basegfx/polygon/b2dpolygon.hxx> 40 #include <basegfx/polygon/b2dpolypolygon.hxx> 41 #include <basegfx/polygon/b2dpolygontools.hxx> 42 #include <basegfx/polygon/b2dpolypolygontools.hxx> 43 #include <basegfx/polygon/b2dpolypolygoncutter.hxx> 44 #include <basegfx/matrix/b2dhommatrix.hxx> 45 46 #include <osl/thread.h> 47 #include <osl/file.h> 48 49 #include <rtl/crc.h> 50 #include <rtl/digest.h> 51 #include <rtl/ustrbuf.hxx> 52 53 #include <tools/debug.hxx> 54 #include <tools/zcodec.hxx> 55 #include <tools/stream.hxx> 56 57 #include <i18npool/mslangid.hxx> 58 59 #include <vcl/virdev.hxx> 60 #include <vcl/bmpacc.hxx> 61 #include <vcl/bitmapex.hxx> 62 #include <vcl/image.hxx> 63 #include <vcl/metric.hxx> 64 #include <vcl/svapp.hxx> 65 #include <vcl/lineinfo.hxx> 66 #include "vcl/cvtgrf.hxx" 67 #include "vcl/strhelper.hxx" 68 69 #include <fontsubset.hxx> 70 #include <outdev.h> 71 #include <sallayout.hxx> 72 #include <textlayout.hxx> 73 #include <salgdi.hxx> 74 75 #include <icc/sRGB-IEC61966-2.1.hxx> 76 77 #include <comphelper/processfactory.hxx> 78 79 #include <com/sun/star/lang/XMultiServiceFactory.hpp> 80 #include <com/sun/star/util/URL.hpp> 81 82 #include "cppuhelper/implbase1.hxx" 83 84 using namespace vcl; 85 using namespace rtl; 86 87 #if (OSL_DEBUG_LEVEL < 2) 88 #define COMPRESS_PAGES 89 #else 90 #define DEBUG_DISABLE_PDFCOMPRESSION // also do not compress streams 91 #endif 92 93 #ifdef DO_TEST_PDF 94 class PDFTestOutputStream : public PDFOutputStream 95 { 96 public: 97 virtual ~PDFTestOutputStream(); 98 virtual void write( const com::sun::star::uno::Reference< com::sun::star::io::XOutputStream >& xStream ); 99 }; 100 101 PDFTestOutputStream::~PDFTestOutputStream() 102 { 103 } 104 105 void PDFTestOutputStream::write( const com::sun::star::uno::Reference< com::sun::star::io::XOutputStream >& xStream ) 106 { 107 OString aStr( "lalala\ntest\ntest\ntest" ); 108 com::sun::star::uno::Sequence< sal_Int8 > aData( aStr.getLength() ); 109 rtl_copyMemory( aData.getArray(), aStr.getStr(), aStr.getLength() ); 110 xStream->writeBytes( aData ); 111 } 112 113 // this test code cannot be used to test PDF/A-1 because it forces 114 // control item (widgets) to bypass the structure controlling 115 // the embedding of such elements in actual run 116 void doTestCode() 117 { 118 static const char* pHome = getenv( "HOME" ); 119 rtl::OUString aTestFile( RTL_CONSTASCII_USTRINGPARAM( "file://" ) ); 120 aTestFile += rtl::OUString( pHome, strlen( pHome ), RTL_TEXTENCODING_MS_1252 ); 121 aTestFile += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/pdf_export_test.pdf" ) ); 122 123 PDFWriter::PDFWriterContext aContext; 124 aContext.URL = aTestFile; 125 aContext.Version = PDFWriter::PDF_1_4; 126 aContext.Tagged = true; 127 aContext.InitialPage = 2; 128 aContext.DocumentInfo.Title = OUString( RTL_CONSTASCII_USTRINGPARAM( "PDF export test document" ) ); 129 aContext.DocumentInfo.Producer = OUString( RTL_CONSTASCII_USTRINGPARAM( "VCL" ) ); 130 131 PDFWriter aWriter( aContext ); 132 aWriter.NewPage( 595, 842 ); 133 aWriter.BeginStructureElement( PDFWriter::Document ); 134 // set duration of 3 sec for first page 135 aWriter.SetAutoAdvanceTime( 3 ); 136 aWriter.SetMapMode( MapMode( MAP_100TH_MM ) ); 137 138 aWriter.SetFillColor( Color( COL_LIGHTRED ) ); 139 aWriter.SetLineColor( Color( COL_LIGHTGREEN ) ); 140 aWriter.DrawRect( Rectangle( Point( 2000, 200 ), Size( 8000, 3000 ) ), 5000, 2000 ); 141 142 aWriter.SetFont( Font( String( RTL_CONSTASCII_USTRINGPARAM( "Times" ) ), Size( 0, 500 ) ) ); 143 aWriter.SetTextColor( Color( COL_BLACK ) ); 144 aWriter.SetLineColor( Color( COL_BLACK ) ); 145 aWriter.SetFillColor( Color( COL_LIGHTBLUE ) ); 146 147 Rectangle aRect( Point( 5000, 5000 ), Size( 6000, 3000 ) ); 148 aWriter.DrawRect( aRect ); 149 aWriter.DrawText( aRect, String( RTL_CONSTASCII_USTRINGPARAM( "Link annot 1" ) ) ); 150 sal_Int32 nFirstLink = aWriter.CreateLink( aRect ); 151 PDFNote aNote; 152 aNote.Title = String( RTL_CONSTASCII_USTRINGPARAM( "A small test note" ) ); 153 aNote.Contents = String( RTL_CONSTASCII_USTRINGPARAM( "There is no business like show business like no business i know. Everything about it is appealing." ) ); 154 aWriter.CreateNote( Rectangle( Point( aRect.Right(), aRect.Top() ), Size( 6000, 3000 ) ), aNote ); 155 156 Rectangle aTargetRect( Point( 3000, 23000 ), Size( 12000, 6000 ) ); 157 aWriter.SetFillColor( Color( COL_LIGHTGREEN ) ); 158 aWriter.DrawRect( aTargetRect ); 159 aWriter.DrawText( aTargetRect, String( RTL_CONSTASCII_USTRINGPARAM( "Dest second link" ) ) ); 160 sal_Int32 nSecondDest = aWriter.CreateDest( aTargetRect ); 161 162 aWriter.BeginStructureElement( PDFWriter::Section ); 163 aWriter.BeginStructureElement( PDFWriter::Heading ); 164 aWriter.DrawText( Point(4500, 9000), String( RTL_CONSTASCII_USTRINGPARAM( "A small structure test" ) ) ); 165 aWriter.EndStructureElement(); 166 aWriter.BeginStructureElement( PDFWriter::Paragraph ); 167 aWriter.SetStructureAttribute( PDFWriter::WritingMode, PDFWriter::LrTb ); 168 aWriter.SetStructureAttribute( PDFWriter::TextDecorationType, PDFWriter::Underline ); 169 aWriter.DrawText( Rectangle( Point( 4500, 10000 ), Size( 12000, 6000 ) ), 170 String( RTL_CONSTASCII_USTRINGPARAM( "It was the best of PDF, it was the worst of PDF ... or so. This is a pretty nonsensical text to denote a paragraph. I suggest you stop reading it. Because if you read on you might get bored. So continue on your on risk. Hey, you're still here ? Why do you continue to read this as it is of no use at all ? OK, it's your time, but still... . Woah, i even get bored writing this, so let's end this here and now." ) ), 171 TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK 172 ); 173 aWriter.SetActualText( String( RTL_CONSTASCII_USTRINGPARAM( "It was the best of PDF, it was the worst of PDF ... or so. This is a pretty nonsensical text to denote a paragraph. I suggest you stop reading it. Because if you read on you might get bored. So continue on your on risk. Hey, you're still here ? Why do you continue to read this as it is of no use at all ? OK, it's your time, but still... . Woah, i even get bored writing this, so let's end this here and now." ) ) ); 174 aWriter.SetAlternateText( String( RTL_CONSTASCII_USTRINGPARAM( "This paragraph contains some lengthy nonsense to test structural element emission of PDFWriter." ) ) ); 175 aWriter.EndStructureElement(); 176 sal_Int32 nLongPara = aWriter.BeginStructureElement( PDFWriter::Paragraph ); 177 aWriter.SetStructureAttribute( PDFWriter::WritingMode, PDFWriter::LrTb ); 178 aWriter.DrawText( Rectangle( Point( 4500, 19000 ), Size( 12000, 1000 ) ), 179 String( RTL_CONSTASCII_USTRINGPARAM( "This paragraph is nothing special either but ends on the next page structurewise" ) ), 180 TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK 181 ); 182 183 aWriter.NewPage( 595, 842 ); 184 // test AddStream interface 185 aWriter.AddStream( String( RTL_CONSTASCII_USTRINGPARAM( "text/plain" ) ), new PDFTestOutputStream(), true ); 186 // set transitional mode 187 aWriter.SetPageTransition( PDFWriter::WipeRightToLeft, 1500 ); 188 aWriter.SetMapMode( MapMode( MAP_100TH_MM ) ); 189 aWriter.SetTextColor( Color( COL_BLACK ) ); 190 aWriter.SetFont( Font( String( RTL_CONSTASCII_USTRINGPARAM( "Times" ) ), Size( 0, 500 ) ) ); 191 aWriter.DrawText( Rectangle( Point( 4500, 1500 ), Size( 12000, 3000 ) ), 192 String( RTL_CONSTASCII_USTRINGPARAM( "Here's where all things come to an end ... well at least the paragaph from the last page." ) ), 193 TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK 194 ); 195 aWriter.EndStructureElement(); 196 197 aWriter.SetFillColor( Color( COL_LIGHTBLUE ) ); 198 // disable structure 199 aWriter.BeginStructureElement( PDFWriter::NonStructElement ); 200 aWriter.DrawRect( aRect ); 201 aWriter.BeginStructureElement( PDFWriter::Paragraph ); 202 aWriter.DrawText( aRect, String( RTL_CONSTASCII_USTRINGPARAM( "Link annot 2" ) ) ); 203 sal_Int32 nSecondLink = aWriter.CreateLink( aRect ); 204 205 aWriter.SetFillColor( Color( COL_LIGHTGREEN ) ); 206 aWriter.BeginStructureElement( PDFWriter::ListItem ); 207 aWriter.DrawRect( aTargetRect ); 208 aWriter.DrawText( aTargetRect, String( RTL_CONSTASCII_USTRINGPARAM( "Dest first link" ) ) ); 209 sal_Int32 nFirstDest = aWriter.CreateDest( aTargetRect ); 210 // enable structure 211 aWriter.EndStructureElement(); 212 // add something to the long paragraph as an afterthought 213 sal_Int32 nSaveStruct = aWriter.GetCurrentStructureElement(); 214 aWriter.SetCurrentStructureElement( nLongPara ); 215 aWriter.DrawText( Rectangle( Point( 4500,4500 ), Size( 12000, 1000 ) ), 216 String( RTL_CONSTASCII_USTRINGPARAM( "Add something to the longish paragraph above." ) ), 217 TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK ); 218 aWriter.SetCurrentStructureElement( nSaveStruct ); 219 aWriter.EndStructureElement(); 220 aWriter.EndStructureElement(); 221 aWriter.BeginStructureElement( PDFWriter::Figure ); 222 aWriter.BeginStructureElement( PDFWriter::Caption ); 223 aWriter.DrawText( Point( 4500, 9000 ), String( RTL_CONSTASCII_USTRINGPARAM( "Some drawing stuff inside the structure" ) ) ); 224 aWriter.EndStructureElement(); 225 226 // test clipping 227 basegfx::B2DPolyPolygon aClip; 228 basegfx::B2DPolygon aClipPoly; 229 aClipPoly.append( basegfx::B2DPoint( 8250, 9600 ) ); 230 aClipPoly.append( basegfx::B2DPoint( 16500, 11100 ) ); 231 aClipPoly.append( basegfx::B2DPoint( 8250, 12600 ) ); 232 aClipPoly.append( basegfx::B2DPoint( 4500, 11100 ) ); 233 aClipPoly.setClosed( true ); 234 //aClipPoly.flip(); 235 aClip.append( aClipPoly ); 236 237 aWriter.Push( PUSH_CLIPREGION | PUSH_FILLCOLOR ); 238 aWriter.SetClipRegion( aClip ); 239 aWriter.DrawEllipse( Rectangle( Point( 4500, 9600 ), Size( 12000, 3000 ) ) ); 240 aWriter.MoveClipRegion( 1000, 500 ); 241 aWriter.SetFillColor( Color( COL_RED ) ); 242 aWriter.DrawEllipse( Rectangle( Point( 4500, 9600 ), Size( 12000, 3000 ) ) ); 243 aWriter.Pop(); 244 // test transparency 245 // draw background 246 Rectangle aTranspRect( Point( 7500, 13500 ), Size( 9000, 6000 ) ); 247 aWriter.SetFillColor( Color( COL_LIGHTRED ) ); 248 aWriter.DrawRect( aTranspRect ); 249 aWriter.BeginTransparencyGroup(); 250 251 aWriter.SetFillColor( Color( COL_LIGHTGREEN ) ); 252 aWriter.DrawEllipse( aTranspRect ); 253 aWriter.SetTextColor( Color( COL_LIGHTBLUE ) ); 254 aWriter.DrawText( aTranspRect, 255 String( RTL_CONSTASCII_USTRINGPARAM( "Some transparent text" ) ), 256 TEXT_DRAW_CENTER | TEXT_DRAW_VCENTER | TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK ); 257 258 aWriter.EndTransparencyGroup( aTranspRect, 50 ); 259 260 // prepare an alpha mask 261 Bitmap aTransMask( Size( 256, 256 ), 8, &Bitmap::GetGreyPalette( 256 ) ); 262 BitmapWriteAccess* pAcc = aTransMask.AcquireWriteAccess(); 263 for( int nX = 0; nX < 256; nX++ ) 264 for( int nY = 0; nY < 256; nY++ ) 265 pAcc->SetPixel( nX, nY, BitmapColor( (sal_uInt8)((nX+nY)/2) ) ); 266 aTransMask.ReleaseAccess( pAcc ); 267 aTransMask.SetPrefMapMode( MAP_MM ); 268 aTransMask.SetPrefSize( Size( 10, 10 ) ); 269 270 aWriter.DrawBitmap( Point( 600, 13500 ), Size( 3000, 3000 ), aTransMask ); 271 272 aTranspRect = Rectangle( Point( 4200, 13500 ), Size( 3000, 3000 ) ); 273 aWriter.SetFillColor( Color( COL_LIGHTRED ) ); 274 aWriter.DrawRect( aTranspRect ); 275 aWriter.SetFillColor( Color( COL_LIGHTGREEN ) ); 276 aWriter.DrawEllipse( aTranspRect ); 277 aWriter.SetTextColor( Color( COL_LIGHTBLUE ) ); 278 aWriter.DrawText( aTranspRect, 279 String( RTL_CONSTASCII_USTRINGPARAM( "Some transparent text" ) ), 280 TEXT_DRAW_CENTER | TEXT_DRAW_VCENTER | TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK ); 281 aTranspRect = Rectangle( Point( 1500, 16500 ), Size( 4800, 3000 ) ); 282 aWriter.SetFillColor( Color( COL_LIGHTRED ) ); 283 aWriter.DrawRect( aTranspRect ); 284 aWriter.BeginTransparencyGroup(); 285 aWriter.SetFillColor( Color( COL_LIGHTGREEN ) ); 286 aWriter.DrawEllipse( aTranspRect ); 287 aWriter.SetTextColor( Color( COL_LIGHTBLUE ) ); 288 aWriter.DrawText( aTranspRect, 289 String( RTL_CONSTASCII_USTRINGPARAM( "Some transparent text" ) ), 290 TEXT_DRAW_CENTER | TEXT_DRAW_VCENTER | TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK ); 291 aWriter.EndTransparencyGroup( aTranspRect, aTransMask ); 292 293 Bitmap aImageBmp( Size( 256, 256 ), 24 ); 294 pAcc = aImageBmp.AcquireWriteAccess(); 295 pAcc->SetFillColor( Color( 0xff, 0, 0xff ) ); 296 pAcc->FillRect( Rectangle( Point( 0, 0 ), Size( 256, 256 ) ) ); 297 aImageBmp.ReleaseAccess( pAcc ); 298 BitmapEx aBmpEx( aImageBmp, AlphaMask( aTransMask ) ); 299 aWriter.DrawBitmapEx( Point( 1500, 19500 ), Size( 4800, 3000 ), aBmpEx ); 300 301 302 aWriter.EndStructureElement(); 303 aWriter.EndStructureElement(); 304 305 LineInfo aLI( LINE_DASH, 3 ); 306 aLI.SetDashCount( 2 ); 307 aLI.SetDashLen( 50 ); 308 aLI.SetDotCount( 2 ); 309 aLI.SetDotLen( 25 ); 310 aLI.SetDistance( 15 ); 311 Point aLIPoints[] = { Point( 4000, 10000 ), 312 Point( 8000, 12000 ), 313 Point( 3000, 19000 ) }; 314 Polygon aLIPoly( 3, aLIPoints ); 315 aWriter.SetLineColor( Color( COL_BLUE ) ); 316 aWriter.SetFillColor(); 317 aWriter.DrawPolyLine( aLIPoly, aLI ); 318 319 aLI.SetDashCount( 4 ); 320 aLIPoly.Move( 1000, 1000 ); 321 aWriter.DrawPolyLine( aLIPoly, aLI ); 322 323 aWriter.NewPage( 595, 842 ); 324 aWriter.SetMapMode( MapMode( MAP_100TH_MM ) ); 325 Wallpaper aWall( aTransMask ); 326 aWall.SetStyle( WALLPAPER_TILE ); 327 aWriter.DrawWallpaper( Rectangle( Point( 4400, 4200 ), Size( 10200, 6300 ) ), aWall ); 328 329 aWriter.Push( PUSH_ALL ); 330 aWriter.BeginPattern(Rectangle(Point(0,0),Size(2000,1000))); 331 aWriter.SetFillColor( Color( COL_RED ) ); 332 aWriter.SetLineColor( Color( COL_LIGHTBLUE ) ); 333 Point aFillPoints[] = { Point( 1000, 0 ), 334 Point( 0, 1000 ), 335 Point( 2000, 1000 ) }; 336 aWriter.DrawPolygon( Polygon( 3, aFillPoints ) ); 337 aWriter.DrawBitmap( Point( 200, 200 ), Size( 1600, 600 ), aTransMask ); 338 aWriter.DrawText( Rectangle( Point( 200, 200 ), Size( 1600, 600 ) ), String( RTL_CONSTASCII_USTRINGPARAM( "Pattern" ) ) ); 339 sal_Int32 nPattern = aWriter.EndPattern( SvtGraphicFill::Transform() ); 340 aWriter.Pop(); 341 Rectangle aPolyRect( Point( 3800, 11200 ), Size( 10200, 6300 ) ); 342 aWriter.DrawPolyPolygon( PolyPolygon( Polygon( aPolyRect ) ), nPattern, true ); 343 aWriter.SetFillColor(); 344 aWriter.SetLineColor( Color( COL_LIGHTBLUE ) ); 345 aWriter.DrawRect( aPolyRect ); 346 347 aWriter.NewPage( 595, 842 ); 348 aWriter.SetMapMode( MapMode( MAP_100TH_MM ) ); 349 aWriter.SetFont( Font( String( RTL_CONSTASCII_USTRINGPARAM( "Times" ) ), Size( 0, 500 ) ) ); 350 aWriter.SetTextColor( Color( COL_BLACK ) ); 351 aRect = Rectangle( Point( 4500, 6000 ), Size( 6000, 1500 ) ); 352 aWriter.DrawRect( aRect ); 353 aWriter.DrawText( aRect, String( RTL_CONSTASCII_USTRINGPARAM( "www.heise.de" ) ) ); 354 sal_Int32 nURILink = aWriter.CreateLink( aRect ); 355 aWriter.SetLinkURL( nURILink, OUString( RTL_CONSTASCII_USTRINGPARAM( "http://www.heise.de" ) ) ); 356 357 aWriter.SetLinkDest( nFirstLink, nFirstDest ); 358 aWriter.SetLinkDest( nSecondLink, nSecondDest ); 359 360 // include a button 361 PDFWriter::PushButtonWidget aBtn; 362 aBtn.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "testButton" ) ); 363 aBtn.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "A test button" ) ); 364 aBtn.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "hit me" ) ); 365 aBtn.Location = Rectangle( Point( 4500, 9000 ), Size( 4500, 3000 ) ); 366 aBtn.Border = aBtn.Background = true; 367 aWriter.CreateControl( aBtn ); 368 369 // include a uri button 370 PDFWriter::PushButtonWidget aUriBtn; 371 aUriBtn.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "wwwButton" ) ); 372 aUriBtn.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "A URI button" ) ); 373 aUriBtn.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "to www" ) ); 374 aUriBtn.Location = Rectangle( Point( 9500, 9000 ), Size( 4500, 3000 ) ); 375 aUriBtn.Border = aUriBtn.Background = true; 376 aUriBtn.URL = OUString( RTL_CONSTASCII_USTRINGPARAM( "http://www.heise.de" ) ); 377 aWriter.CreateControl( aUriBtn ); 378 379 // include a dest button 380 PDFWriter::PushButtonWidget aDstBtn; 381 aDstBtn.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "destButton" ) ); 382 aDstBtn.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "A Dest button" ) ); 383 aDstBtn.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "to paragraph" ) ); 384 aDstBtn.Location = Rectangle( Point( 14500, 9000 ), Size( 4500, 3000 ) ); 385 aDstBtn.Border = aDstBtn.Background = true; 386 aDstBtn.Dest = nFirstDest; 387 aWriter.CreateControl( aDstBtn ); 388 389 PDFWriter::CheckBoxWidget aCBox; 390 aCBox.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "textCheckBox" ) ); 391 aCBox.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "A test check box" ) ); 392 aCBox.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "check me" ) ); 393 aCBox.Location = Rectangle( Point( 4500, 13500 ), Size( 3000, 750 ) ); 394 aCBox.Checked = true; 395 aCBox.Border = aCBox.Background = false; 396 aWriter.CreateControl( aCBox ); 397 398 PDFWriter::CheckBoxWidget aCBox2; 399 aCBox2.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "textCheckBox2" ) ); 400 aCBox2.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "Another test check box" ) ); 401 aCBox2.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "check me right" ) ); 402 aCBox2.Location = Rectangle( Point( 4500, 14250 ), Size( 3000, 750 ) ); 403 aCBox2.Checked = true; 404 aCBox2.Border = aCBox2.Background = false; 405 aCBox2.ButtonIsLeft = false; 406 aWriter.CreateControl( aCBox2 ); 407 408 PDFWriter::RadioButtonWidget aRB1; 409 aRB1.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "rb1_1" ) ); 410 aRB1.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "radio 1 button 1" ) ); 411 aRB1.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "Despair" ) ); 412 aRB1.Location = Rectangle( Point( 4500, 15000 ), Size( 6000, 1000 ) ); 413 aRB1.Selected = true; 414 aRB1.RadioGroup = 1; 415 aRB1.Border = aRB1.Background = true; 416 aRB1.ButtonIsLeft = false; 417 aRB1.BorderColor = Color( COL_LIGHTGREEN ); 418 aRB1.BackgroundColor = Color( COL_LIGHTBLUE ); 419 aRB1.TextColor = Color( COL_LIGHTRED ); 420 aRB1.TextFont = Font( String( RTL_CONSTASCII_USTRINGPARAM( "Courier" ) ), Size( 0, 800 ) ); 421 aWriter.CreateControl( aRB1 ); 422 423 PDFWriter::RadioButtonWidget aRB2; 424 aRB2.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "rb2_1" ) ); 425 aRB2.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "radio 2 button 1" ) ); 426 aRB2.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "Joy" ) ); 427 aRB2.Location = Rectangle( Point( 10500, 15000 ), Size( 3000, 1000 ) ); 428 aRB2.Selected = true; 429 aRB2.RadioGroup = 2; 430 aWriter.CreateControl( aRB2 ); 431 432 PDFWriter::RadioButtonWidget aRB3; 433 aRB3.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "rb1_2" ) ); 434 aRB3.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "radio 1 button 2" ) ); 435 aRB3.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "Desperation" ) ); 436 aRB3.Location = Rectangle( Point( 4500, 16000 ), Size( 3000, 1000 ) ); 437 aRB3.Selected = true; 438 aRB3.RadioGroup = 1; 439 aWriter.CreateControl( aRB3 ); 440 441 PDFWriter::EditWidget aEditBox; 442 aEditBox.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "testEdit" ) ); 443 aEditBox.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "A test edit field" ) ); 444 aEditBox.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "A little test text" ) ); 445 aEditBox.TextStyle = TEXT_DRAW_LEFT | TEXT_DRAW_VCENTER; 446 aEditBox.Location = Rectangle( Point( 10000, 18000 ), Size( 5000, 1500 ) ); 447 aEditBox.MaxLen = 100; 448 aEditBox.Border = aEditBox.Background = true; 449 aEditBox.BorderColor = Color( COL_BLACK ); 450 aWriter.CreateControl( aEditBox ); 451 452 // normal list box 453 PDFWriter::ListBoxWidget aLstBox; 454 aLstBox.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "testListBox" ) ); 455 aLstBox.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "One" ) ); 456 aLstBox.Description = OUString( RTL_CONSTASCII_USTRINGPARAM( "select me" ) ); 457 aLstBox.Location = Rectangle( Point( 4500, 18000 ), Size( 3000, 1500 ) ); 458 aLstBox.Sort = true; 459 aLstBox.MultiSelect = true; 460 aLstBox.Border = aLstBox.Background = true; 461 aLstBox.BorderColor = Color( COL_BLACK ); 462 aLstBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "One" ) ) ); 463 aLstBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "Two" ) ) ); 464 aLstBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "Three" ) ) ); 465 aLstBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "Four" ) ) ); 466 aLstBox.SelectedEntries.push_back( 1 ); 467 aLstBox.SelectedEntries.push_back( 2 ); 468 aWriter.CreateControl( aLstBox ); 469 470 // dropdown list box 471 aLstBox.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "testDropDownListBox" ) ); 472 aLstBox.DropDown = true; 473 aLstBox.Location = Rectangle( Point( 4500, 19500 ), Size( 3000, 500 ) ); 474 aWriter.CreateControl( aLstBox ); 475 476 // combo box 477 PDFWriter::ComboBoxWidget aComboBox; 478 aComboBox.Name = OUString( RTL_CONSTASCII_USTRINGPARAM( "testComboBox" ) ); 479 aComboBox.Text = OUString( RTL_CONSTASCII_USTRINGPARAM( "test a combobox" ) ); 480 aComboBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "Larry" ) ) ); 481 aComboBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "Curly" ) ) ); 482 aComboBox.Entries.push_back( OUString( RTL_CONSTASCII_USTRINGPARAM( "Moe" ) ) ); 483 aComboBox.Location = Rectangle( Point( 4500, 20000 ), Size( 3000, 500 ) ); 484 aWriter.CreateControl( aComboBox ); 485 486 // test outlines 487 sal_Int32 nPage1OL = aWriter.CreateOutlineItem(); 488 aWriter.SetOutlineItemText( nPage1OL, OUString( RTL_CONSTASCII_USTRINGPARAM( "Page 1" ) ) ); 489 aWriter.SetOutlineItemDest( nPage1OL, nSecondDest ); 490 aWriter.CreateOutlineItem( nPage1OL, OUString( RTL_CONSTASCII_USTRINGPARAM( "Dest 2" ) ), nSecondDest ); 491 aWriter.CreateOutlineItem( nPage1OL, OUString( RTL_CONSTASCII_USTRINGPARAM( "Dest 2 revisited" ) ), nSecondDest ); 492 aWriter.CreateOutlineItem( nPage1OL, OUString( RTL_CONSTASCII_USTRINGPARAM( "Dest 2 again" ) ), nSecondDest ); 493 sal_Int32 nPage2OL = aWriter.CreateOutlineItem(); 494 aWriter.SetOutlineItemText( nPage2OL, OUString( RTL_CONSTASCII_USTRINGPARAM( "Page 2" ) ) ); 495 aWriter.CreateOutlineItem( nPage2OL, OUString( RTL_CONSTASCII_USTRINGPARAM( "Dest 1" ) ), nFirstDest ); 496 497 aWriter.EndStructureElement(); // close document 498 aWriter.Emit(); 499 } 500 #endif 501 502 static const sal_Int32 nLog10Divisor = 1; 503 static const double fDivisor = 10.0; 504 505 static inline double pixelToPoint( sal_Int32 px ) { return double(px)/fDivisor; } 506 static inline double pixelToPoint( double px ) { return px/fDivisor; } 507 static inline sal_Int32 pointToPixel( double pt ) { return sal_Int32(pt*fDivisor); } 508 509 const sal_uInt8 PDFWriterImpl::s_nPadString[32] = 510 { 511 0x28, 0xBF, 0x4E, 0x5E, 0x4E, 0x75, 0x8A, 0x41, 0x64, 0x00, 0x4E, 0x56, 0xFF, 0xFA, 0x01, 0x08, 512 0x2E, 0x2E, 0x00, 0xB6, 0xD0, 0x68, 0x3E, 0x80, 0x2F, 0x0C, 0xA9, 0xFE, 0x64, 0x53, 0x69, 0x7A 513 }; 514 515 static void appendHex( sal_Int8 nInt, OStringBuffer& rBuffer ) 516 { 517 static const sal_Char pHexDigits[] = { '0', '1', '2', '3', '4', '5', '6', '7', 518 '8', '9', 'A', 'B', 'C', 'D', 'E', 'F' }; 519 rBuffer.append( pHexDigits[ (nInt >> 4) & 15 ] ); 520 rBuffer.append( pHexDigits[ nInt & 15 ] ); 521 } 522 523 static void appendName( const OUString& rStr, OStringBuffer& rBuffer ) 524 { 525 // FIXME i59651 add a check for max length of 127 chars? Per PDF spec 1.4, appendix C.1 526 // I guess than when reading the #xx sequence it will count for a single character. 527 OString aStr( OUStringToOString( rStr, RTL_TEXTENCODING_UTF8 ) ); 528 const sal_Char* pStr = aStr.getStr(); 529 int nLen = aStr.getLength(); 530 for( int i = 0; i < nLen; i++ ) 531 { 532 /* #i16920# PDF recommendation: output UTF8, any byte 533 * outside the interval [33(=ASCII'!');126(=ASCII'~')] 534 * should be escaped hexadecimal 535 * for the sake of ghostscript which also reads PDF 536 * but has a narrower acceptance rate we only pass 537 * alphanumerics and '-' literally. 538 */ 539 if( (pStr[i] >= 'A' && pStr[i] <= 'Z' ) || 540 (pStr[i] >= 'a' && pStr[i] <= 'z' ) || 541 (pStr[i] >= '0' && pStr[i] <= '9' ) || 542 pStr[i] == '-' ) 543 { 544 rBuffer.append( pStr[i] ); 545 } 546 else 547 { 548 rBuffer.append( '#' ); 549 appendHex( (sal_Int8)pStr[i], rBuffer ); 550 } 551 } 552 } 553 554 static void appendName( const sal_Char* pStr, OStringBuffer& rBuffer ) 555 { 556 //FIXME i59651 see above 557 while( pStr && *pStr ) 558 { 559 if( (*pStr >= 'A' && *pStr <= 'Z' ) || 560 (*pStr >= 'a' && *pStr <= 'z' ) || 561 (*pStr >= '0' && *pStr <= '9' ) || 562 *pStr == '-' ) 563 { 564 rBuffer.append( *pStr ); 565 } 566 else 567 { 568 rBuffer.append( '#' ); 569 appendHex( (sal_Int8)*pStr, rBuffer ); 570 } 571 pStr++; 572 } 573 } 574 575 //used only to emit encoded passwords 576 static void appendLiteralString( const sal_Char* pStr, sal_Int32 nLength, OStringBuffer& rBuffer ) 577 { 578 while( nLength ) 579 { 580 switch( *pStr ) 581 { 582 case '\n' : 583 rBuffer.append( "\\n" ); 584 break; 585 case '\r' : 586 rBuffer.append( "\\r" ); 587 break; 588 case '\t' : 589 rBuffer.append( "\\t" ); 590 break; 591 case '\b' : 592 rBuffer.append( "\\b" ); 593 break; 594 case '\f' : 595 rBuffer.append( "\\f" ); 596 break; 597 case '(' : 598 case ')' : 599 case '\\' : 600 rBuffer.append( "\\" ); 601 rBuffer.append( (sal_Char) *pStr ); 602 break; 603 default: 604 rBuffer.append( (sal_Char) *pStr ); 605 break; 606 } 607 pStr++; 608 nLength--; 609 } 610 } 611 612 /**--->i56629 613 * Convert a string before using it. 614 * 615 * This string conversion function is needed because the destination name 616 * in a PDF file seen through an Internet browser should be 617 * specially crafted, in order to be used directly by the browser. 618 * In this way the fragment part of a hyperlink to a PDF file (e.g. something 619 * as 'test1/test2/a-file.pdf#thefragment) will be (hopefully) interpreted by the 620 * PDF reader (currently only Adobe Reader plug-in seems to be working that way) called 621 * from inside the Internet browser as: 'open the file test1/test2/a-file.pdf 622 * and go to named destination thefragment using default zoom'. 623 * The conversion is needed because in case of a fragment in the form: Slide%201 624 * (meaning Slide 1) as it is converted obeying the Inet rules, it will become Slide25201 625 * using this conversion, in both the generated named destinations, fragment and GoToR 626 * destination. 627 * 628 * The names for destinations are name objects and so they don't need to be encrypted 629 * even though they expose the content of PDF file (e.g. guessing the PDF content from the 630 * destination name). 631 * 632 * Fhurter limitation: it is advisable to use standard ASCII characters for 633 * OOo bookmarks. 634 */ 635 static void appendDestinationName( const rtl::OUString& rString, OStringBuffer& rBuffer ) 636 { 637 const sal_Unicode* pStr = rString.getStr(); 638 sal_Int32 nLen = rString.getLength(); 639 for( int i = 0; i < nLen; i++ ) 640 { 641 sal_Unicode aChar = pStr[i]; 642 if( (aChar >= '0' && aChar <= '9' ) || 643 (aChar >= 'a' && aChar <= 'z' ) || 644 (aChar >= 'A' && aChar <= 'Z' ) || 645 aChar == '-' ) 646 { 647 rBuffer.append((sal_Char)aChar); 648 } 649 else 650 { 651 sal_Int8 aValueHigh = sal_Int8(aChar >> 8); 652 if(aValueHigh > 0) 653 appendHex( aValueHigh, rBuffer ); 654 appendHex( (sal_Int8)(aChar & 255 ), rBuffer ); 655 } 656 } 657 } 658 //<--- i56629 659 660 static void appendUnicodeTextString( const rtl::OUString& rString, OStringBuffer& rBuffer ) 661 { 662 rBuffer.append( "FEFF" ); 663 const sal_Unicode* pStr = rString.getStr(); 664 sal_Int32 nLen = rString.getLength(); 665 for( int i = 0; i < nLen; i++ ) 666 { 667 sal_Unicode aChar = pStr[i]; 668 appendHex( (sal_Int8)(aChar >> 8), rBuffer ); 669 appendHex( (sal_Int8)(aChar & 255 ), rBuffer ); 670 } 671 } 672 673 void PDFWriterImpl::createWidgetFieldName( sal_Int32 i_nWidgetIndex, const PDFWriter::AnyWidget& i_rControl ) 674 { 675 /* #i80258# previously we use appendName here 676 however we need a slightly different coding scheme than the normal 677 name encoding for field names 678 */ 679 const OUString& rName = (m_aContext.Version > PDFWriter::PDF_1_2) ? i_rControl.Name : i_rControl.Text; 680 OString aStr( OUStringToOString( rName, RTL_TEXTENCODING_UTF8 ) ); 681 const sal_Char* pStr = aStr.getStr(); 682 int nLen = aStr.getLength(); 683 684 OStringBuffer aBuffer( rName.getLength()+64 ); 685 for( int i = 0; i < nLen; i++ ) 686 { 687 /* #i16920# PDF recommendation: output UTF8, any byte 688 * outside the interval [32(=ASCII' ');126(=ASCII'~')] 689 * should be escaped hexadecimal 690 */ 691 if( (pStr[i] >= 32 && pStr[i] <= 126 ) ) 692 aBuffer.append( pStr[i] ); 693 else 694 { 695 aBuffer.append( '#' ); 696 appendHex( (sal_Int8)pStr[i], aBuffer ); 697 } 698 } 699 700 OString aFullName( aBuffer.makeStringAndClear() ); 701 702 /* #i82785# create hierarchical fields down to the for each dot in i_rName */ 703 sal_Int32 nTokenIndex = 0, nLastTokenIndex = 0; 704 OString aPartialName; 705 OString aDomain; 706 do 707 { 708 nLastTokenIndex = nTokenIndex; 709 aPartialName = aFullName.getToken( 0, '.', nTokenIndex ); 710 if( nTokenIndex != -1 ) 711 { 712 // find or create a hierarchical field 713 // first find the fully qualified name up to this field 714 aDomain = aFullName.copy( 0, nTokenIndex-1 ); 715 std::hash_map< rtl::OString, sal_Int32, rtl::OStringHash >::const_iterator it = m_aFieldNameMap.find( aDomain ); 716 if( it == m_aFieldNameMap.end() ) 717 { 718 // create new hierarchy field 719 sal_Int32 nNewWidget = m_aWidgets.size(); 720 m_aWidgets.push_back( PDFWidget() ); 721 m_aWidgets[nNewWidget].m_nObject = createObject(); 722 m_aWidgets[nNewWidget].m_eType = PDFWriter::Hierarchy; 723 m_aWidgets[nNewWidget].m_aName = aPartialName; 724 m_aWidgets[i_nWidgetIndex].m_nParent = m_aWidgets[nNewWidget].m_nObject; 725 m_aFieldNameMap[aDomain] = nNewWidget; 726 m_aWidgets[i_nWidgetIndex].m_nParent = m_aWidgets[nNewWidget].m_nObject; 727 if( nLastTokenIndex > 0 ) 728 { 729 // this field is not a root field and 730 // needs to be inserted to its parent 731 OString aParentDomain( aDomain.copy( 0, nLastTokenIndex-1 ) ); 732 it = m_aFieldNameMap.find( aParentDomain ); 733 OSL_ENSURE( it != m_aFieldNameMap.end(), "field name not found" ); 734 if( it != m_aFieldNameMap.end() ) 735 { 736 OSL_ENSURE( it->second < sal_Int32(m_aWidgets.size()), "invalid field number entry" ); 737 if( it->second < sal_Int32(m_aWidgets.size()) ) 738 { 739 PDFWidget& rParentField( m_aWidgets[it->second] ); 740 rParentField.m_aKids.push_back( m_aWidgets[nNewWidget].m_nObject ); 741 rParentField.m_aKidsIndex.push_back( nNewWidget ); 742 m_aWidgets[nNewWidget].m_nParent = rParentField.m_nObject; 743 } 744 } 745 } 746 } 747 else if( m_aWidgets[it->second].m_eType != PDFWriter::Hierarchy ) 748 { 749 // this is invalid, someone tries to have a terminal field as parent 750 // example: a button with the name foo.bar exists and 751 // another button is named foo.bar.no 752 // workaround: put the second terminal field as much up in the hierarchy as 753 // necessary to have a non-terminal field as parent (or none at all) 754 // since it->second already is terminal, we just need to use its parent 755 aDomain = OString(); 756 aPartialName = aFullName.copy( aFullName.lastIndexOf( '.' )+1 ); 757 if( nLastTokenIndex > 0 ) 758 { 759 aDomain = aFullName.copy( 0, nLastTokenIndex-1 ); 760 OStringBuffer aBuf( aDomain.getLength() + 1 + aPartialName.getLength() ); 761 aBuf.append( aDomain ); 762 aBuf.append( '.' ); 763 aBuf.append( aPartialName ); 764 aFullName = aBuf.makeStringAndClear(); 765 } 766 else 767 aFullName = aPartialName; 768 break; 769 } 770 } 771 } while( nTokenIndex != -1 ); 772 773 // insert widget into its hierarchy field 774 if( aDomain.getLength() ) 775 { 776 std::hash_map< rtl::OString, sal_Int32, rtl::OStringHash >::const_iterator it = m_aFieldNameMap.find( aDomain ); 777 if( it != m_aFieldNameMap.end() ) 778 { 779 OSL_ENSURE( it->second >= 0 && it->second < sal_Int32( m_aWidgets.size() ), "invalid field index" ); 780 if( it->second >= 0 && it->second < sal_Int32(m_aWidgets.size()) ) 781 { 782 m_aWidgets[i_nWidgetIndex].m_nParent = m_aWidgets[it->second].m_nObject; 783 m_aWidgets[it->second].m_aKids.push_back( m_aWidgets[i_nWidgetIndex].m_nObject); 784 m_aWidgets[it->second].m_aKidsIndex.push_back( i_nWidgetIndex ); 785 } 786 } 787 } 788 789 if( aPartialName.getLength() == 0 ) 790 { 791 // how funny, an empty field name 792 if( i_rControl.getType() == PDFWriter::RadioButton ) 793 { 794 aPartialName = "RadioGroup"; 795 aPartialName += OString::valueOf( static_cast<const PDFWriter::RadioButtonWidget&>(i_rControl).RadioGroup ); 796 } 797 else 798 aPartialName = OString( "Widget" ); 799 } 800 801 if( ! m_aContext.AllowDuplicateFieldNames ) 802 { 803 std::hash_map<OString, sal_Int32, OStringHash>::iterator it = m_aFieldNameMap.find( aFullName ); 804 805 if( it != m_aFieldNameMap.end() ) // not unique 806 { 807 std::hash_map< OString, sal_Int32, OStringHash >::const_iterator check_it; 808 OString aTry; 809 sal_Int32 nTry = 2; 810 do 811 { 812 OStringBuffer aUnique( aFullName.getLength() + 16 ); 813 aUnique.append( aFullName ); 814 aUnique.append( '_' ); 815 aUnique.append( nTry++ ); 816 aTry = aUnique.makeStringAndClear(); 817 check_it = m_aFieldNameMap.find( aTry ); 818 } while( check_it != m_aFieldNameMap.end() ); 819 aFullName = aTry; 820 m_aFieldNameMap[ aFullName ] = i_nWidgetIndex; 821 aPartialName = aFullName.copy( aFullName.lastIndexOf( '.' )+1 ); 822 } 823 else 824 m_aFieldNameMap[ aFullName ] = i_nWidgetIndex; 825 } 826 827 // finally 828 m_aWidgets[i_nWidgetIndex].m_aName = aPartialName; 829 } 830 831 static void appendFixedInt( sal_Int32 nValue, OStringBuffer& rBuffer, sal_Int32 nPrecision = nLog10Divisor ) 832 { 833 if( nValue < 0 ) 834 { 835 rBuffer.append( '-' ); 836 nValue = -nValue; 837 } 838 sal_Int32 nFactor = 1, nDiv = nPrecision; 839 while( nDiv-- ) 840 nFactor *= 10; 841 842 sal_Int32 nInt = nValue / nFactor; 843 rBuffer.append( nInt ); 844 if( nFactor > 1 ) 845 { 846 sal_Int32 nDecimal = nValue % nFactor; 847 if( nDecimal ) 848 { 849 rBuffer.append( '.' ); 850 // omit trailing zeros 851 while( (nDecimal % 10) == 0 ) 852 nDecimal /= 10; 853 rBuffer.append( nDecimal ); 854 } 855 } 856 } 857 858 859 // appends a double. PDF does not accept exponential format, only fixed point 860 static void appendDouble( double fValue, OStringBuffer& rBuffer, sal_Int32 nPrecision = 5 ) 861 { 862 bool bNeg = false; 863 if( fValue < 0.0 ) 864 { 865 bNeg = true; 866 fValue=-fValue; 867 } 868 869 sal_Int64 nInt = (sal_Int64)fValue; 870 fValue -= (double)nInt; 871 // optimizing hardware may lead to a value of 1.0 after the subtraction 872 if( fValue == 1.0 || log10( 1.0-fValue ) <= -nPrecision ) 873 { 874 nInt++; 875 fValue = 0.0; 876 } 877 sal_Int64 nFrac = 0; 878 if( fValue ) 879 { 880 fValue *= pow( 10.0, (double)nPrecision ); 881 nFrac = (sal_Int64)fValue; 882 } 883 if( bNeg && ( nInt || nFrac ) ) 884 rBuffer.append( '-' ); 885 rBuffer.append( nInt ); 886 if( nFrac ) 887 { 888 int i; 889 rBuffer.append( '.' ); 890 sal_Int64 nBound = (sal_Int64)(pow( 10.0, nPrecision - 1.0 )+0.5); 891 for ( i = 0; ( i < nPrecision ) && nFrac; i++ ) 892 { 893 sal_Int64 nNumb = nFrac / nBound; 894 nFrac -= nNumb * nBound; 895 rBuffer.append( nNumb ); 896 nBound /= 10; 897 } 898 } 899 } 900 901 902 static void appendColor( const Color& rColor, OStringBuffer& rBuffer, bool bConvertToGrey = false ) 903 { 904 905 if( rColor != Color( COL_TRANSPARENT ) ) 906 { 907 if( bConvertToGrey ) 908 { 909 sal_uInt8 cByte = rColor.GetLuminance(); 910 appendDouble( (double)cByte / 255.0, rBuffer ); 911 } 912 else 913 { 914 appendDouble( (double)rColor.GetRed() / 255.0, rBuffer ); 915 rBuffer.append( ' ' ); 916 appendDouble( (double)rColor.GetGreen() / 255.0, rBuffer ); 917 rBuffer.append( ' ' ); 918 appendDouble( (double)rColor.GetBlue() / 255.0, rBuffer ); 919 } 920 } 921 } 922 923 void PDFWriterImpl::appendStrokingColor( const Color& rColor, OStringBuffer& rBuffer ) 924 { 925 if( rColor != Color( COL_TRANSPARENT ) ) 926 { 927 bool bGrey = m_aContext.ColorMode == PDFWriter::DrawGreyscale; 928 appendColor( rColor, rBuffer, bGrey ); 929 rBuffer.append( bGrey ? " G" : " RG" ); 930 } 931 } 932 933 void PDFWriterImpl::appendNonStrokingColor( const Color& rColor, OStringBuffer& rBuffer ) 934 { 935 if( rColor != Color( COL_TRANSPARENT ) ) 936 { 937 bool bGrey = m_aContext.ColorMode == PDFWriter::DrawGreyscale; 938 appendColor( rColor, rBuffer, bGrey ); 939 rBuffer.append( bGrey ? " g" : " rg" ); 940 } 941 } 942 943 // matrix helper class 944 // TODO: use basegfx matrix class instead or derive from it 945 namespace vcl // TODO: use anonymous namespace to keep this class local 946 { 947 /* for sparse matrices of the form (2D linear transformations) 948 * f[0] f[1] 0 949 * f[2] f[3] 0 950 * f[4] f[5] 1 951 */ 952 class Matrix3 953 { 954 double f[6]; 955 956 void set( double *pn ) { for( int i = 0 ; i < 6; i++ ) f[i] = pn[i]; } 957 public: 958 Matrix3(); 959 ~Matrix3() {} 960 961 void skew( double alpha, double beta ); 962 void scale( double sx, double sy ); 963 void rotate( double angle ); 964 void translate( double tx, double ty ); 965 bool invert(); 966 967 void append( PDFWriterImpl::PDFPage& rPage, OStringBuffer& rBuffer, Point* pBack = NULL ); 968 969 Point transform( const Point& rPoint ) const; 970 }; 971 } 972 973 Matrix3::Matrix3() 974 { 975 // initialize to unity 976 f[0] = 1.0; 977 f[1] = 0.0; 978 f[2] = 0.0; 979 f[3] = 1.0; 980 f[4] = 0.0; 981 f[5] = 0.0; 982 } 983 984 Point Matrix3::transform( const Point& rOrig ) const 985 { 986 double x = (double)rOrig.X(), y = (double)rOrig.Y(); 987 return Point( (int)(x*f[0] + y*f[2] + f[4]), (int)(x*f[1] + y*f[3] + f[5]) ); 988 } 989 990 void Matrix3::skew( double alpha, double beta ) 991 { 992 double fn[6]; 993 double tb = tan( beta ); 994 fn[0] = f[0] + f[2]*tb; 995 fn[1] = f[1]; 996 fn[2] = f[2] + f[3]*tb; 997 fn[3] = f[3]; 998 fn[4] = f[4] + f[5]*tb; 999 fn[5] = f[5]; 1000 if( alpha != 0.0 ) 1001 { 1002 double ta = tan( alpha ); 1003 fn[1] += f[0]*ta; 1004 fn[3] += f[2]*ta; 1005 fn[5] += f[4]*ta; 1006 } 1007 set( fn ); 1008 } 1009 1010 void Matrix3::scale( double sx, double sy ) 1011 { 1012 double fn[6]; 1013 fn[0] = sx*f[0]; 1014 fn[1] = sy*f[1]; 1015 fn[2] = sx*f[2]; 1016 fn[3] = sy*f[3]; 1017 fn[4] = sx*f[4]; 1018 fn[5] = sy*f[5]; 1019 set( fn ); 1020 } 1021 1022 void Matrix3::rotate( double angle ) 1023 { 1024 double fn[6]; 1025 double fSin = sin(angle); 1026 double fCos = cos(angle); 1027 fn[0] = f[0]*fCos - f[1]*fSin; 1028 fn[1] = f[0]*fSin + f[1]*fCos; 1029 fn[2] = f[2]*fCos - f[3]*fSin; 1030 fn[3] = f[2]*fSin + f[3]*fCos; 1031 fn[4] = f[4]*fCos - f[5]*fSin; 1032 fn[5] = f[4]*fSin + f[5]*fCos; 1033 set( fn ); 1034 } 1035 1036 void Matrix3::translate( double tx, double ty ) 1037 { 1038 f[4] += tx; 1039 f[5] += ty; 1040 } 1041 1042 bool Matrix3::invert() 1043 { 1044 // short circuit trivial cases 1045 if( f[1]==f[2] && f[1]==0.0 && f[0]==f[3] && f[0]==1.0 ) 1046 { 1047 f[4] = -f[4]; 1048 f[5] = -f[5]; 1049 return true; 1050 } 1051 1052 // check determinant 1053 const double fDet = f[0]*f[3]-f[1]*f[2]; 1054 if( fDet == 0.0 ) 1055 return false; 1056 1057 // invert the matrix 1058 double fn[6]; 1059 fn[0] = +f[3] / fDet; 1060 fn[1] = -f[1] / fDet; 1061 fn[2] = -f[2] / fDet; 1062 fn[3] = +f[0] / fDet; 1063 1064 // apply inversion to translation 1065 fn[4] = -(f[4]*fn[0] + f[5]*fn[2]); 1066 fn[5] = -(f[4]*fn[1] + f[5]*fn[3]); 1067 1068 set( fn ); 1069 return true; 1070 } 1071 1072 void Matrix3::append( PDFWriterImpl::PDFPage& rPage, OStringBuffer& rBuffer, Point* pBack ) 1073 { 1074 appendDouble( f[0], rBuffer ); 1075 rBuffer.append( ' ' ); 1076 appendDouble( f[1], rBuffer ); 1077 rBuffer.append( ' ' ); 1078 appendDouble( f[2], rBuffer ); 1079 rBuffer.append( ' ' ); 1080 appendDouble( f[3], rBuffer ); 1081 rBuffer.append( ' ' ); 1082 rPage.appendPoint( Point( (long)f[4], (long)f[5] ), rBuffer, false, pBack ); 1083 } 1084 1085 static void appendResourceMap( OStringBuffer& rBuf, const char* pPrefix, const PDFWriterImpl::ResourceMap& rList ) 1086 { 1087 if( rList.empty() ) 1088 return; 1089 rBuf.append( '/' ); 1090 rBuf.append( pPrefix ); 1091 rBuf.append( "<<" ); 1092 int ni = 0; 1093 for( PDFWriterImpl::ResourceMap::const_iterator it = rList.begin(); it != rList.end(); ++it ) 1094 { 1095 if( it->first.getLength() && it->second > 0 ) 1096 { 1097 rBuf.append( '/' ); 1098 rBuf.append( it->first ); 1099 rBuf.append( ' ' ); 1100 rBuf.append( it->second ); 1101 rBuf.append( " 0 R" ); 1102 if( ((++ni) & 7) == 0 ) 1103 rBuf.append( '\n' ); 1104 } 1105 } 1106 rBuf.append( ">>\n" ); 1107 } 1108 1109 void PDFWriterImpl::ResourceDict::append( OStringBuffer& rBuf, sal_Int32 nFontDictObject ) 1110 { 1111 rBuf.append( "<</Font " ); 1112 rBuf.append( nFontDictObject ); 1113 rBuf.append( " 0 R\n" ); 1114 appendResourceMap( rBuf, "XObject", m_aXObjects ); 1115 appendResourceMap( rBuf, "ExtGState", m_aExtGStates ); 1116 appendResourceMap( rBuf, "Shading", m_aShadings ); 1117 appendResourceMap( rBuf, "Pattern", m_aPatterns ); 1118 rBuf.append( "/ProcSet[/PDF/Text" ); 1119 if( !m_aXObjects.empty() ) 1120 rBuf.append( "/ImageC/ImageI/ImageB" ); 1121 rBuf.append( "]\n>>\n" ); 1122 }; 1123 1124 PDFWriterImpl::PDFPage::PDFPage( PDFWriterImpl* pWriter, sal_Int32 nPageWidth, sal_Int32 nPageHeight, PDFWriter::Orientation eOrientation ) 1125 : 1126 m_pWriter( pWriter ), 1127 m_nPageWidth( nPageWidth ), 1128 m_nPageHeight( nPageHeight ), 1129 m_eOrientation( eOrientation ), 1130 m_nPageObject( 0 ), // invalid object number 1131 m_nPageIndex( -1 ), // invalid index 1132 m_nStreamLengthObject( 0 ), 1133 m_nBeginStreamPos( 0 ), 1134 m_eTransition( PDFWriter::Regular ), 1135 m_nTransTime( 0 ), 1136 m_nDuration( 0 ), 1137 m_bHasWidgets( false ) 1138 { 1139 // object ref must be only ever updated in emit() 1140 m_nPageObject = m_pWriter->createObject(); 1141 } 1142 1143 PDFWriterImpl::PDFPage::~PDFPage() 1144 { 1145 } 1146 1147 void PDFWriterImpl::PDFPage::beginStream() 1148 { 1149 #if OSL_DEBUG_LEVEL > 1 1150 { 1151 OStringBuffer aLine( "PDFWriterImpl::PDFPage::beginStream, +" ); 1152 m_pWriter->emitComment( aLine.getStr() ); 1153 } 1154 #endif 1155 m_aStreamObjects.push_back(m_pWriter->createObject()); 1156 if( ! m_pWriter->updateObject( m_aStreamObjects.back() ) ) 1157 return; 1158 1159 m_nStreamLengthObject = m_pWriter->createObject(); 1160 // write content stream header 1161 OStringBuffer aLine; 1162 aLine.append( m_aStreamObjects.back() ); 1163 aLine.append( " 0 obj\n<</Length " ); 1164 aLine.append( m_nStreamLengthObject ); 1165 aLine.append( " 0 R" ); 1166 #if defined ( COMPRESS_PAGES ) && !defined ( DEBUG_DISABLE_PDFCOMPRESSION ) 1167 aLine.append( "/Filter/FlateDecode" ); 1168 #endif 1169 aLine.append( ">>\nstream\n" ); 1170 if( ! m_pWriter->writeBuffer( aLine.getStr(), aLine.getLength() ) ) 1171 return; 1172 if( osl_File_E_None != osl_getFilePos( m_pWriter->m_aFile, &m_nBeginStreamPos ) ) 1173 { 1174 osl_closeFile( m_pWriter->m_aFile ); 1175 m_pWriter->m_bOpen = false; 1176 } 1177 #if defined ( COMPRESS_PAGES ) && !defined ( DEBUG_DISABLE_PDFCOMPRESSION ) 1178 m_pWriter->beginCompression(); 1179 #endif 1180 m_pWriter->checkAndEnableStreamEncryption( m_aStreamObjects.back() ); 1181 } 1182 1183 void PDFWriterImpl::PDFPage::endStream() 1184 { 1185 #if defined ( COMPRESS_PAGES ) && !defined ( DEBUG_DISABLE_PDFCOMPRESSION ) 1186 m_pWriter->endCompression(); 1187 #endif 1188 sal_uInt64 nEndStreamPos; 1189 if( osl_File_E_None != osl_getFilePos( m_pWriter->m_aFile, &nEndStreamPos ) ) 1190 { 1191 osl_closeFile( m_pWriter->m_aFile ); 1192 m_pWriter->m_bOpen = false; 1193 return; 1194 } 1195 m_pWriter->disableStreamEncryption(); 1196 if( ! m_pWriter->writeBuffer( "\nendstream\nendobj\n\n", 19 ) ) 1197 return; 1198 // emit stream length object 1199 if( ! m_pWriter->updateObject( m_nStreamLengthObject ) ) 1200 return; 1201 OStringBuffer aLine; 1202 aLine.append( m_nStreamLengthObject ); 1203 aLine.append( " 0 obj\n" ); 1204 aLine.append( (sal_Int64)(nEndStreamPos-m_nBeginStreamPos) ); 1205 aLine.append( "\nendobj\n\n" ); 1206 m_pWriter->writeBuffer( aLine.getStr(), aLine.getLength() ); 1207 } 1208 1209 bool PDFWriterImpl::PDFPage::emit(sal_Int32 nParentObject ) 1210 { 1211 // emit page object 1212 if( ! m_pWriter->updateObject( m_nPageObject ) ) 1213 return false; 1214 OStringBuffer aLine; 1215 1216 aLine.append( m_nPageObject ); 1217 aLine.append( " 0 obj\n" 1218 "<</Type/Page/Parent " ); 1219 aLine.append( nParentObject ); 1220 aLine.append( " 0 R" ); 1221 aLine.append( "/Resources " ); 1222 aLine.append( m_pWriter->getResourceDictObj() ); 1223 aLine.append( " 0 R" ); 1224 if( m_nPageWidth && m_nPageHeight ) 1225 { 1226 aLine.append( "/MediaBox[0 0 " ); 1227 aLine.append( m_nPageWidth ); 1228 aLine.append( ' ' ); 1229 aLine.append( m_nPageHeight ); 1230 aLine.append( "]" ); 1231 } 1232 switch( m_eOrientation ) 1233 { 1234 case PDFWriter::Landscape: aLine.append( "/Rotate 90\n" );break; 1235 case PDFWriter::Seascape: aLine.append( "/Rotate -90\n" );break; 1236 case PDFWriter::Portrait: aLine.append( "/Rotate 0\n" );break; 1237 1238 case PDFWriter::Inherit: 1239 default: 1240 break; 1241 } 1242 int nAnnots = m_aAnnotations.size(); 1243 if( nAnnots > 0 ) 1244 { 1245 aLine.append( "/Annots[\n" ); 1246 for( int i = 0; i < nAnnots; i++ ) 1247 { 1248 aLine.append( m_aAnnotations[i] ); 1249 aLine.append( " 0 R" ); 1250 aLine.append( ((i+1)%15) ? " " : "\n" ); 1251 } 1252 aLine.append( "]\n" ); 1253 } 1254 #if 0 1255 // FIXME: implement tab order as Structure Tree 1256 if( m_bHasWidgets && m_pWriter->getVersion() >= PDFWriter::PDF_1_5 ) 1257 aLine.append( " /Tabs /S\n" ); 1258 #endif 1259 if( m_aMCIDParents.size() > 0 ) 1260 { 1261 OStringBuffer aStructParents( 1024 ); 1262 aStructParents.append( "[ " ); 1263 int nParents = m_aMCIDParents.size(); 1264 for( int i = 0; i < nParents; i++ ) 1265 { 1266 aStructParents.append( m_aMCIDParents[i] ); 1267 aStructParents.append( " 0 R" ); 1268 aStructParents.append( ((i%10) == 9) ? "\n" : " " ); 1269 } 1270 aStructParents.append( "]" ); 1271 m_pWriter->m_aStructParentTree.push_back( aStructParents.makeStringAndClear() ); 1272 1273 aLine.append( "/StructParents " ); 1274 aLine.append( sal_Int32(m_pWriter->m_aStructParentTree.size()-1) ); 1275 aLine.append( "\n" ); 1276 } 1277 if( m_nDuration > 0 ) 1278 { 1279 aLine.append( "/Dur " ); 1280 aLine.append( (sal_Int32)m_nDuration ); 1281 aLine.append( "\n" ); 1282 } 1283 if( m_eTransition != PDFWriter::Regular && m_nTransTime > 0 ) 1284 { 1285 // transition duration 1286 aLine.append( "/Trans<</D " ); 1287 appendDouble( (double)m_nTransTime/1000.0, aLine, 3 ); 1288 aLine.append( "\n" ); 1289 const char *pStyle = NULL, *pDm = NULL, *pM = NULL, *pDi = NULL; 1290 switch( m_eTransition ) 1291 { 1292 case PDFWriter::SplitHorizontalInward: 1293 pStyle = "Split"; pDm = "H"; pM = "I"; break; 1294 case PDFWriter::SplitHorizontalOutward: 1295 pStyle = "Split"; pDm = "H"; pM = "O"; break; 1296 case PDFWriter::SplitVerticalInward: 1297 pStyle = "Split"; pDm = "V"; pM = "I"; break; 1298 case PDFWriter::SplitVerticalOutward: 1299 pStyle = "Split"; pDm = "V"; pM = "O"; break; 1300 case PDFWriter::BlindsHorizontal: 1301 pStyle = "Blinds"; pDm = "H"; break; 1302 case PDFWriter::BlindsVertical: 1303 pStyle = "Blinds"; pDm = "V"; break; 1304 case PDFWriter::BoxInward: 1305 pStyle = "Box"; pM = "I"; break; 1306 case PDFWriter::BoxOutward: 1307 pStyle = "Box"; pM = "O"; break; 1308 case PDFWriter::WipeLeftToRight: 1309 pStyle = "Wipe"; pDi = "0"; break; 1310 case PDFWriter::WipeBottomToTop: 1311 pStyle = "Wipe"; pDi = "90"; break; 1312 case PDFWriter::WipeRightToLeft: 1313 pStyle = "Wipe"; pDi = "180"; break; 1314 case PDFWriter::WipeTopToBottom: 1315 pStyle = "Wipe"; pDi = "270"; break; 1316 case PDFWriter::Dissolve: 1317 pStyle = "Dissolve"; break; 1318 case PDFWriter::GlitterLeftToRight: 1319 pStyle = "Glitter"; pDi = "0"; break; 1320 case PDFWriter::GlitterTopToBottom: 1321 pStyle = "Glitter"; pDi = "270"; break; 1322 case PDFWriter::GlitterTopLeftToBottomRight: 1323 pStyle = "Glitter"; pDi = "315"; break; 1324 case PDFWriter::Regular: 1325 break; 1326 } 1327 // transition style 1328 if( pStyle ) 1329 { 1330 aLine.append( "/S/" ); 1331 aLine.append( pStyle ); 1332 aLine.append( "\n" ); 1333 } 1334 if( pDm ) 1335 { 1336 aLine.append( "/Dm/" ); 1337 aLine.append( pDm ); 1338 aLine.append( "\n" ); 1339 } 1340 if( pM ) 1341 { 1342 aLine.append( "/M/" ); 1343 aLine.append( pM ); 1344 aLine.append( "\n" ); 1345 } 1346 if( pDi ) 1347 { 1348 aLine.append( "/Di " ); 1349 aLine.append( pDi ); 1350 aLine.append( "\n" ); 1351 } 1352 aLine.append( ">>\n" ); 1353 } 1354 if( m_pWriter->getVersion() > PDFWriter::PDF_1_3 && ! m_pWriter->m_bIsPDF_A1 ) 1355 { 1356 aLine.append( "/Group<</S/Transparency/CS/DeviceRGB/I true>>" ); 1357 } 1358 aLine.append( "/Contents" ); 1359 unsigned int nStreamObjects = m_aStreamObjects.size(); 1360 if( nStreamObjects > 1 ) 1361 aLine.append( '[' ); 1362 for( unsigned int i = 0; i < m_aStreamObjects.size(); i++ ) 1363 { 1364 aLine.append( ' ' ); 1365 aLine.append( m_aStreamObjects[i] ); 1366 aLine.append( " 0 R" ); 1367 } 1368 if( nStreamObjects > 1 ) 1369 aLine.append( ']' ); 1370 aLine.append( ">>\nendobj\n\n" ); 1371 return m_pWriter->writeBuffer( aLine.getStr(), aLine.getLength() ); 1372 } 1373 1374 namespace vcl 1375 { 1376 template < class GEOMETRY > 1377 GEOMETRY lcl_convert( const MapMode& _rSource, const MapMode& _rDest, OutputDevice* _pPixelConversion, const GEOMETRY& _rObject ) 1378 { 1379 GEOMETRY aPoint; 1380 if ( MAP_PIXEL == _rSource.GetMapUnit() ) 1381 { 1382 aPoint = _pPixelConversion->PixelToLogic( _rObject, _rDest ); 1383 } 1384 else 1385 { 1386 aPoint = OutputDevice::LogicToLogic( _rObject, _rSource, _rDest ); 1387 } 1388 return aPoint; 1389 } 1390 } 1391 1392 void PDFWriterImpl::PDFPage::appendPoint( const Point& rPoint, OStringBuffer& rBuffer, bool bNeg, Point* pOutPoint ) const 1393 { 1394 if( pOutPoint ) 1395 { 1396 Point aPoint( lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1397 m_pWriter->m_aMapMode, 1398 m_pWriter->getReferenceDevice(), 1399 rPoint ) ); 1400 *pOutPoint = aPoint; 1401 } 1402 1403 Point aPoint( lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1404 m_pWriter->m_aMapMode, 1405 m_pWriter->getReferenceDevice(), 1406 rPoint ) ); 1407 1408 sal_Int32 nValue = aPoint.X(); 1409 if( bNeg ) 1410 nValue = -nValue; 1411 1412 appendFixedInt( nValue, rBuffer ); 1413 1414 rBuffer.append( ' ' ); 1415 1416 nValue = pointToPixel(getHeight()) - aPoint.Y(); 1417 if( bNeg ) 1418 nValue = -nValue; 1419 1420 appendFixedInt( nValue, rBuffer ); 1421 } 1422 1423 void PDFWriterImpl::PDFPage::appendPixelPoint( const basegfx::B2DPoint& rPoint, OStringBuffer& rBuffer ) const 1424 { 1425 double fValue = pixelToPoint(rPoint.getX()); 1426 1427 appendDouble( fValue, rBuffer, nLog10Divisor ); 1428 1429 rBuffer.append( ' ' ); 1430 1431 fValue = double(getHeight()) - pixelToPoint(rPoint.getY()); 1432 1433 appendDouble( fValue, rBuffer, nLog10Divisor ); 1434 } 1435 1436 void PDFWriterImpl::PDFPage::appendRect( const Rectangle& rRect, OStringBuffer& rBuffer ) const 1437 { 1438 appendPoint( rRect.BottomLeft() + Point( 0, 1 ), rBuffer ); 1439 rBuffer.append( ' ' ); 1440 appendMappedLength( (sal_Int32)rRect.GetWidth(), rBuffer, false ); 1441 rBuffer.append( ' ' ); 1442 appendMappedLength( (sal_Int32)rRect.GetHeight(), rBuffer, true ); 1443 rBuffer.append( " re" ); 1444 } 1445 1446 void PDFWriterImpl::PDFPage::convertRect( Rectangle& rRect ) const 1447 { 1448 Point aLL = lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1449 m_pWriter->m_aMapMode, 1450 m_pWriter->getReferenceDevice(), 1451 rRect.BottomLeft() + Point( 0, 1 ) 1452 ); 1453 Size aSize = lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1454 m_pWriter->m_aMapMode, 1455 m_pWriter->getReferenceDevice(), 1456 rRect.GetSize() ); 1457 rRect.Left() = aLL.X(); 1458 rRect.Right() = aLL.X() + aSize.Width(); 1459 rRect.Top() = pointToPixel(getHeight()) - aLL.Y(); 1460 rRect.Bottom() = rRect.Top() + aSize.Height(); 1461 } 1462 1463 void PDFWriterImpl::PDFPage::appendPolygon( const Polygon& rPoly, OStringBuffer& rBuffer, bool bClose ) const 1464 { 1465 sal_uInt16 nPoints = rPoly.GetSize(); 1466 /* 1467 * #108582# applications do weird things 1468 */ 1469 sal_uInt32 nBufLen = rBuffer.getLength(); 1470 if( nPoints > 0 ) 1471 { 1472 const sal_uInt8* pFlagArray = rPoly.GetConstFlagAry(); 1473 appendPoint( rPoly[0], rBuffer ); 1474 rBuffer.append( " m\n" ); 1475 for( sal_uInt16 i = 1; i < nPoints; i++ ) 1476 { 1477 if( pFlagArray && pFlagArray[i] == POLY_CONTROL && nPoints-i > 2 ) 1478 { 1479 // bezier 1480 DBG_ASSERT( pFlagArray[i+1] == POLY_CONTROL && pFlagArray[i+2] != POLY_CONTROL, "unexpected sequence of control points" ); 1481 appendPoint( rPoly[i], rBuffer ); 1482 rBuffer.append( " " ); 1483 appendPoint( rPoly[i+1], rBuffer ); 1484 rBuffer.append( " " ); 1485 appendPoint( rPoly[i+2], rBuffer ); 1486 rBuffer.append( " c" ); 1487 i += 2; // add additionally consumed points 1488 } 1489 else 1490 { 1491 // line 1492 appendPoint( rPoly[i], rBuffer ); 1493 rBuffer.append( " l" ); 1494 } 1495 if( (rBuffer.getLength() - nBufLen) > 65 ) 1496 { 1497 rBuffer.append( "\n" ); 1498 nBufLen = rBuffer.getLength(); 1499 } 1500 else 1501 rBuffer.append( " " ); 1502 } 1503 if( bClose ) 1504 rBuffer.append( "h\n" ); 1505 } 1506 } 1507 1508 void PDFWriterImpl::PDFPage::appendPolygon( const basegfx::B2DPolygon& rPoly, OStringBuffer& rBuffer, bool bClose ) const 1509 { 1510 basegfx::B2DPolygon aPoly( lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1511 m_pWriter->m_aMapMode, 1512 m_pWriter->getReferenceDevice(), 1513 rPoly ) ); 1514 1515 if( basegfx::tools::isRectangle( aPoly ) ) 1516 { 1517 basegfx::B2DRange aRange( aPoly.getB2DRange() ); 1518 basegfx::B2DPoint aBL( aRange.getMinX(), aRange.getMaxY() ); 1519 appendPixelPoint( aBL, rBuffer ); 1520 rBuffer.append( ' ' ); 1521 appendMappedLength( aRange.getWidth(), rBuffer, false, NULL, nLog10Divisor ); 1522 rBuffer.append( ' ' ); 1523 appendMappedLength( aRange.getHeight(), rBuffer, true, NULL, nLog10Divisor ); 1524 rBuffer.append( " re\n" ); 1525 return; 1526 } 1527 sal_uInt32 nPoints = aPoly.count(); 1528 if( nPoints > 0 ) 1529 { 1530 sal_uInt32 nBufLen = rBuffer.getLength(); 1531 basegfx::B2DPoint aLastPoint( aPoly.getB2DPoint( 0 ) ); 1532 appendPixelPoint( aLastPoint, rBuffer ); 1533 rBuffer.append( " m\n" ); 1534 for( sal_uInt32 i = 1; i <= nPoints; i++ ) 1535 { 1536 if( i != nPoints || aPoly.isClosed() ) 1537 { 1538 sal_uInt32 nCurPoint = i % nPoints; 1539 sal_uInt32 nLastPoint = i-1; 1540 basegfx::B2DPoint aPoint( aPoly.getB2DPoint( nCurPoint ) ); 1541 if( aPoly.isNextControlPointUsed( nLastPoint ) && 1542 aPoly.isPrevControlPointUsed( nCurPoint ) ) 1543 { 1544 appendPixelPoint( aPoly.getNextControlPoint( nLastPoint ), rBuffer ); 1545 rBuffer.append( ' ' ); 1546 appendPixelPoint( aPoly.getPrevControlPoint( nCurPoint ), rBuffer ); 1547 rBuffer.append( ' ' ); 1548 appendPixelPoint( aPoint, rBuffer ); 1549 rBuffer.append( " c" ); 1550 } 1551 else if( aPoly.isNextControlPointUsed( nLastPoint ) ) 1552 { 1553 appendPixelPoint( aPoly.getNextControlPoint( nLastPoint ), rBuffer ); 1554 rBuffer.append( ' ' ); 1555 appendPixelPoint( aPoint, rBuffer ); 1556 rBuffer.append( " y" ); 1557 } 1558 else if( aPoly.isPrevControlPointUsed( nCurPoint ) ) 1559 { 1560 appendPixelPoint( aPoly.getPrevControlPoint( nCurPoint ), rBuffer ); 1561 rBuffer.append( ' ' ); 1562 appendPixelPoint( aPoint, rBuffer ); 1563 rBuffer.append( " v" ); 1564 } 1565 else 1566 { 1567 appendPixelPoint( aPoint, rBuffer ); 1568 rBuffer.append( " l" ); 1569 } 1570 if( (rBuffer.getLength() - nBufLen) > 65 ) 1571 { 1572 rBuffer.append( "\n" ); 1573 nBufLen = rBuffer.getLength(); 1574 } 1575 else 1576 rBuffer.append( " " ); 1577 } 1578 } 1579 if( bClose ) 1580 rBuffer.append( "h\n" ); 1581 } 1582 } 1583 1584 void PDFWriterImpl::PDFPage::appendPolyPolygon( const PolyPolygon& rPolyPoly, OStringBuffer& rBuffer, bool bClose ) const 1585 { 1586 sal_uInt16 nPolygons = rPolyPoly.Count(); 1587 for( sal_uInt16 n = 0; n < nPolygons; n++ ) 1588 appendPolygon( rPolyPoly[n], rBuffer, bClose ); 1589 } 1590 1591 void PDFWriterImpl::PDFPage::appendPolyPolygon( const basegfx::B2DPolyPolygon& rPolyPoly, OStringBuffer& rBuffer, bool bClose ) const 1592 { 1593 sal_uInt32 nPolygons = rPolyPoly.count(); 1594 for( sal_uInt32 n = 0; n < nPolygons; n++ ) 1595 appendPolygon( rPolyPoly.getB2DPolygon( n ), rBuffer, bClose ); 1596 } 1597 1598 void PDFWriterImpl::PDFPage::appendMappedLength( sal_Int32 nLength, OStringBuffer& rBuffer, bool bVertical, sal_Int32* pOutLength ) const 1599 { 1600 sal_Int32 nValue = nLength; 1601 if ( nLength < 0 ) 1602 { 1603 rBuffer.append( '-' ); 1604 nValue = -nLength; 1605 } 1606 Size aSize( lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1607 m_pWriter->m_aMapMode, 1608 m_pWriter->getReferenceDevice(), 1609 Size( nValue, nValue ) ) ); 1610 nValue = bVertical ? aSize.Height() : aSize.Width(); 1611 if( pOutLength ) 1612 *pOutLength = ((nLength < 0 ) ? -nValue : nValue); 1613 1614 appendFixedInt( nValue, rBuffer, 1 ); 1615 } 1616 1617 void PDFWriterImpl::PDFPage::appendMappedLength( double fLength, OStringBuffer& rBuffer, bool bVertical, sal_Int32* pOutLength, sal_Int32 nPrecision ) const 1618 { 1619 Size aSize( lcl_convert( m_pWriter->m_aGraphicsStack.front().m_aMapMode, 1620 m_pWriter->m_aMapMode, 1621 m_pWriter->getReferenceDevice(), 1622 Size( 1000, 1000 ) ) ); 1623 if( pOutLength ) 1624 *pOutLength = (sal_Int32)(fLength*(double)(bVertical ? aSize.Height() : aSize.Width())/1000.0); 1625 fLength *= pixelToPoint((double)(bVertical ? aSize.Height() : aSize.Width()) / 1000.0); 1626 appendDouble( fLength, rBuffer, nPrecision ); 1627 } 1628 1629 bool PDFWriterImpl::PDFPage::appendLineInfo( const LineInfo& rInfo, OStringBuffer& rBuffer ) const 1630 { 1631 bool bRet = true; 1632 if( rInfo.GetStyle() == LINE_DASH ) 1633 { 1634 rBuffer.append( "[ " ); 1635 if( rInfo.GetDashLen() == rInfo.GetDotLen() ) // degraded case 1636 { 1637 appendMappedLength( (sal_Int32)rInfo.GetDashLen(), rBuffer ); 1638 rBuffer.append( ' ' ); 1639 appendMappedLength( (sal_Int32)rInfo.GetDistance(), rBuffer ); 1640 rBuffer.append( ' ' ); 1641 } 1642 else 1643 { 1644 // check for implementation limits of dash array 1645 // in PDF reader apps (e.g. acroread) 1646 if( 2*(rInfo.GetDashCount() + rInfo.GetDotCount()) > 10 ) 1647 bRet = false; 1648 for( int n = 0; n < rInfo.GetDashCount(); n++ ) 1649 { 1650 appendMappedLength( (sal_Int32)rInfo.GetDashLen(), rBuffer ); 1651 rBuffer.append( ' ' ); 1652 appendMappedLength( (sal_Int32)rInfo.GetDistance(), rBuffer ); 1653 rBuffer.append( ' ' ); 1654 } 1655 for( int m = 0; m < rInfo.GetDotCount(); m++ ) 1656 { 1657 appendMappedLength( (sal_Int32)rInfo.GetDotLen(), rBuffer ); 1658 rBuffer.append( ' ' ); 1659 appendMappedLength( (sal_Int32)rInfo.GetDistance(), rBuffer ); 1660 rBuffer.append( ' ' ); 1661 } 1662 } 1663 rBuffer.append( "] 0 d\n" ); 1664 } 1665 if( rInfo.GetWidth() > 1 ) 1666 { 1667 appendMappedLength( (sal_Int32)rInfo.GetWidth(), rBuffer ); 1668 rBuffer.append( " w\n" ); 1669 } 1670 else if( rInfo.GetWidth() == 0 ) 1671 { 1672 // "pixel" line 1673 appendDouble( 72.0/double(m_pWriter->getReferenceDevice()->ImplGetDPIX()), rBuffer ); 1674 rBuffer.append( " w\n" ); 1675 } 1676 return bRet; 1677 } 1678 1679 void PDFWriterImpl::PDFPage::appendWaveLine( sal_Int32 nWidth, sal_Int32 nY, sal_Int32 nDelta, OStringBuffer& rBuffer ) const 1680 { 1681 if( nWidth <= 0 ) 1682 return; 1683 if( nDelta < 1 ) 1684 nDelta = 1; 1685 1686 rBuffer.append( "0 " ); 1687 appendMappedLength( nY, rBuffer, true ); 1688 rBuffer.append( " m\n" ); 1689 for( sal_Int32 n = 0; n < nWidth; ) 1690 { 1691 n += nDelta; 1692 appendMappedLength( n, rBuffer, false ); 1693 rBuffer.append( ' ' ); 1694 appendMappedLength( nDelta+nY, rBuffer, true ); 1695 rBuffer.append( ' ' ); 1696 n += nDelta; 1697 appendMappedLength( n, rBuffer, false ); 1698 rBuffer.append( ' ' ); 1699 appendMappedLength( nY, rBuffer, true ); 1700 rBuffer.append( " v " ); 1701 if( n < nWidth ) 1702 { 1703 n += nDelta; 1704 appendMappedLength( n, rBuffer, false ); 1705 rBuffer.append( ' ' ); 1706 appendMappedLength( nY-nDelta, rBuffer, true ); 1707 rBuffer.append( ' ' ); 1708 n += nDelta; 1709 appendMappedLength( n, rBuffer, false ); 1710 rBuffer.append( ' ' ); 1711 appendMappedLength( nY, rBuffer, true ); 1712 rBuffer.append( " v\n" ); 1713 } 1714 } 1715 rBuffer.append( "S\n" ); 1716 } 1717 1718 /* 1719 * class PDFWriterImpl 1720 */ 1721 1722 PDFWriterImpl::PDFWriterImpl( const PDFWriter::PDFWriterContext& rContext, 1723 const com::sun::star::uno::Reference< com::sun::star::beans::XMaterialHolder >& xEnc, 1724 PDFWriter& i_rOuterFace) 1725 : 1726 m_pReferenceDevice( NULL ), 1727 m_aMapMode( MAP_POINT, Point(), Fraction( 1L, pointToPixel(1) ), Fraction( 1L, pointToPixel(1) ) ), 1728 m_nCurrentStructElement( 0 ), 1729 m_bEmitStructure( true ), 1730 m_bNewMCID( false ), 1731 m_nCurrentControl( -1 ), 1732 m_bEmbedStandardFonts( false ), 1733 m_nNextFID( 1 ), 1734 m_nInheritedPageWidth( 595 ), // default A4 1735 m_nInheritedPageHeight( 842 ), // default A4 1736 m_eInheritedOrientation( PDFWriter::Portrait ), 1737 m_nCurrentPage( -1 ), 1738 m_nResourceDict( -1 ), 1739 m_nFontDictObject( -1 ), 1740 m_pCodec( NULL ), 1741 m_aDocDigest( rtl_digest_createMD5() ), 1742 m_aCipher( (rtlCipher)NULL ), 1743 m_aDigest( NULL ), 1744 m_bEncryptThisStream( false ), 1745 m_pEncryptionBuffer( NULL ), 1746 m_nEncryptionBufferSize( 0 ), 1747 m_bIsPDF_A1( false ), 1748 m_rOuterFace( i_rOuterFace ) 1749 { 1750 #ifdef DO_TEST_PDF 1751 static bool bOnce = true; 1752 if( bOnce ) 1753 { 1754 bOnce = false; 1755 doTestCode(); 1756 } 1757 #endif 1758 m_aContext = rContext; 1759 m_aStructure.push_back( PDFStructureElement() ); 1760 m_aStructure[0].m_nOwnElement = 0; 1761 m_aStructure[0].m_nParentElement = 0; 1762 1763 Font aFont; 1764 aFont.SetName( String( RTL_CONSTASCII_USTRINGPARAM( "Times" ) ) ); 1765 aFont.SetSize( Size( 0, 12 ) ); 1766 1767 GraphicsState aState; 1768 aState.m_aMapMode = m_aMapMode; 1769 aState.m_aFont = aFont; 1770 m_aGraphicsStack.push_front( aState ); 1771 1772 oslFileError aError = osl_openFile( m_aContext.URL.pData, &m_aFile, osl_File_OpenFlag_Write | osl_File_OpenFlag_Create ); 1773 if( aError != osl_File_E_None ) 1774 { 1775 if( aError == osl_File_E_EXIST ) 1776 { 1777 aError = osl_openFile( m_aContext.URL.pData, &m_aFile, osl_File_OpenFlag_Write ); 1778 if( aError == osl_File_E_None ) 1779 aError = osl_setFileSize( m_aFile, 0 ); 1780 } 1781 } 1782 if( aError != osl_File_E_None ) 1783 return; 1784 1785 m_bOpen = true; 1786 1787 // setup DocInfo 1788 setupDocInfo(); 1789 1790 /* prepare the cypher engine, can be done in CTOR, free in DTOR */ 1791 m_aCipher = rtl_cipher_createARCFOUR( rtl_Cipher_ModeStream ); 1792 m_aDigest = rtl_digest_createMD5(); 1793 1794 /* the size of the Codec default maximum */ 1795 checkEncryptionBufferSize( 0x4000 ); 1796 1797 if( xEnc.is() ) 1798 prepareEncryption( xEnc ); 1799 1800 if( m_aContext.Encryption.Encrypt() ) 1801 { 1802 // sanity check 1803 if( m_aContext.Encryption.OValue.size() != ENCRYPTED_PWD_SIZE || 1804 m_aContext.Encryption.UValue.size() != ENCRYPTED_PWD_SIZE || 1805 m_aContext.Encryption.EncryptionKey.size() != MAXIMUM_RC4_KEY_LENGTH 1806 ) 1807 { 1808 // the field lengths are invalid ? This was not setup by initEncryption. 1809 // do not encrypt after all 1810 m_aContext.Encryption.OValue.clear(); 1811 m_aContext.Encryption.UValue.clear(); 1812 OSL_ENSURE( 0, "encryption data failed sanity check, encryption disabled" ); 1813 } 1814 else // setup key lengths 1815 m_nAccessPermissions = computeAccessPermissions( m_aContext.Encryption, m_nKeyLength, m_nRC4KeyLength ); 1816 } 1817 1818 // write header 1819 OStringBuffer aBuffer( 20 ); 1820 aBuffer.append( "%PDF-" ); 1821 switch( m_aContext.Version ) 1822 { 1823 case PDFWriter::PDF_1_2: aBuffer.append( "1.2" );break; 1824 case PDFWriter::PDF_1_3: aBuffer.append( "1.3" );break; 1825 case PDFWriter::PDF_A_1: 1826 default: 1827 case PDFWriter::PDF_1_4: aBuffer.append( "1.4" );break; 1828 case PDFWriter::PDF_1_5: aBuffer.append( "1.5" );break; 1829 } 1830 // append something binary as comment (suggested in PDF Reference) 1831 aBuffer.append( "\n%äüöß\n" ); 1832 if( !writeBuffer( aBuffer.getStr(), aBuffer.getLength() ) ) 1833 { 1834 osl_closeFile( m_aFile ); 1835 m_bOpen = false; 1836 return; 1837 } 1838 1839 // insert outline root 1840 m_aOutline.push_back( PDFOutlineEntry() ); 1841 1842 m_bIsPDF_A1 = (m_aContext.Version == PDFWriter::PDF_A_1); 1843 if( m_bIsPDF_A1 ) 1844 m_aContext.Version = PDFWriter::PDF_1_4; //meaning we need PDF 1.4, PDF/A flavour 1845 1846 m_bEmbedStandardFonts = m_aContext.EmbedStandardFonts; 1847 } 1848 1849 PDFWriterImpl::~PDFWriterImpl() 1850 { 1851 if( m_aDocDigest ) 1852 rtl_digest_destroyMD5( m_aDocDigest ); 1853 delete static_cast<VirtualDevice*>(m_pReferenceDevice); 1854 1855 if( m_aCipher ) 1856 rtl_cipher_destroyARCFOUR( m_aCipher ); 1857 if( m_aDigest ) 1858 rtl_digest_destroyMD5( m_aDigest ); 1859 1860 rtl_freeMemory( m_pEncryptionBuffer ); 1861 } 1862 1863 void PDFWriterImpl::setupDocInfo() 1864 { 1865 std::vector< sal_uInt8 > aId; 1866 computeDocumentIdentifier( aId, m_aContext.DocumentInfo, m_aCreationDateString, m_aCreationMetaDateString ); 1867 if( m_aContext.Encryption.DocumentIdentifier.empty() ) 1868 m_aContext.Encryption.DocumentIdentifier = aId; 1869 } 1870 1871 void PDFWriterImpl::computeDocumentIdentifier( std::vector< sal_uInt8 >& o_rIdentifier, 1872 const vcl::PDFWriter::PDFDocInfo& i_rDocInfo, 1873 rtl::OString& o_rCString1, 1874 rtl::OString& o_rCString2 1875 ) 1876 { 1877 o_rIdentifier.clear(); 1878 1879 //build the document id 1880 rtl::OString aInfoValuesOut; 1881 OStringBuffer aID( 1024 ); 1882 if( i_rDocInfo.Title.Len() ) 1883 appendUnicodeTextString( i_rDocInfo.Title, aID ); 1884 if( i_rDocInfo.Author.Len() ) 1885 appendUnicodeTextString( i_rDocInfo.Author, aID ); 1886 if( i_rDocInfo.Subject.Len() ) 1887 appendUnicodeTextString( i_rDocInfo.Subject, aID ); 1888 if( i_rDocInfo.Keywords.Len() ) 1889 appendUnicodeTextString( i_rDocInfo.Keywords, aID ); 1890 if( i_rDocInfo.Creator.Len() ) 1891 appendUnicodeTextString( i_rDocInfo.Creator, aID ); 1892 if( i_rDocInfo.Producer.Len() ) 1893 appendUnicodeTextString( i_rDocInfo.Producer, aID ); 1894 1895 TimeValue aTVal, aGMT; 1896 oslDateTime aDT; 1897 osl_getSystemTime( &aGMT ); 1898 osl_getLocalTimeFromSystemTime( &aGMT, &aTVal ); 1899 osl_getDateTimeFromTimeValue( &aTVal, &aDT ); 1900 rtl::OStringBuffer aCreationDateString(64), aCreationMetaDateString(64); 1901 aCreationDateString.append( "D:" ); 1902 aCreationDateString.append( (sal_Char)('0' + ((aDT.Year/1000)%10)) ); 1903 aCreationDateString.append( (sal_Char)('0' + ((aDT.Year/100)%10)) ); 1904 aCreationDateString.append( (sal_Char)('0' + ((aDT.Year/10)%10)) ); 1905 aCreationDateString.append( (sal_Char)('0' + ((aDT.Year)%10)) ); 1906 aCreationDateString.append( (sal_Char)('0' + ((aDT.Month/10)%10)) ); 1907 aCreationDateString.append( (sal_Char)('0' + ((aDT.Month)%10)) ); 1908 aCreationDateString.append( (sal_Char)('0' + ((aDT.Day/10)%10)) ); 1909 aCreationDateString.append( (sal_Char)('0' + ((aDT.Day)%10)) ); 1910 aCreationDateString.append( (sal_Char)('0' + ((aDT.Hours/10)%10)) ); 1911 aCreationDateString.append( (sal_Char)('0' + ((aDT.Hours)%10)) ); 1912 aCreationDateString.append( (sal_Char)('0' + ((aDT.Minutes/10)%10)) ); 1913 aCreationDateString.append( (sal_Char)('0' + ((aDT.Minutes)%10)) ); 1914 aCreationDateString.append( (sal_Char)('0' + ((aDT.Seconds/10)%10)) ); 1915 aCreationDateString.append( (sal_Char)('0' + ((aDT.Seconds)%10)) ); 1916 1917 //--> i59651, we fill the Metadata date string as well, if PDF/A is requested 1918 // according to ISO 19005-1:2005 6.7.3 the date is corrected for 1919 // local time zone offset UTC only, whereas Acrobat 8 seems 1920 // to use the localtime notation only 1921 // according to a raccomandation in XMP Specification (Jan 2004, page 75) 1922 // the Acrobat way seems the right approach 1923 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Year/1000)%10)) ); 1924 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Year/100)%10)) ); 1925 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Year/10)%10)) ); 1926 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Year)%10)) ); 1927 aCreationMetaDateString.append( "-" ); 1928 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Month/10)%10)) ); 1929 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Month)%10)) ); 1930 aCreationMetaDateString.append( "-" ); 1931 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Day/10)%10)) ); 1932 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Day)%10)) ); 1933 aCreationMetaDateString.append( "T" ); 1934 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Hours/10)%10)) ); 1935 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Hours)%10)) ); 1936 aCreationMetaDateString.append( ":" ); 1937 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Minutes/10)%10)) ); 1938 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Minutes)%10)) ); 1939 aCreationMetaDateString.append( ":" ); 1940 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Seconds/10)%10)) ); 1941 aCreationMetaDateString.append( (sal_Char)('0' + ((aDT.Seconds)%10)) ); 1942 1943 sal_uInt32 nDelta = 0; 1944 if( aGMT.Seconds > aTVal.Seconds ) 1945 { 1946 aCreationDateString.append( "-" ); 1947 nDelta = aGMT.Seconds-aTVal.Seconds; 1948 aCreationMetaDateString.append( "-" ); 1949 } 1950 else if( aGMT.Seconds < aTVal.Seconds ) 1951 { 1952 aCreationDateString.append( "+" ); 1953 nDelta = aTVal.Seconds-aGMT.Seconds; 1954 aCreationMetaDateString.append( "+" ); 1955 } 1956 else 1957 { 1958 aCreationDateString.append( "Z" ); 1959 aCreationMetaDateString.append( "Z" ); 1960 1961 } 1962 if( nDelta ) 1963 { 1964 aCreationDateString.append( (sal_Char)('0' + ((nDelta/36000)%10)) ); 1965 aCreationDateString.append( (sal_Char)('0' + ((nDelta/3600)%10)) ); 1966 aCreationDateString.append( "'" ); 1967 aCreationDateString.append( (sal_Char)('0' + ((nDelta/600)%6)) ); 1968 aCreationDateString.append( (sal_Char)('0' + ((nDelta/60)%10)) ); 1969 1970 aCreationMetaDateString.append( (sal_Char)('0' + ((nDelta/36000)%10)) ); 1971 aCreationMetaDateString.append( (sal_Char)('0' + ((nDelta/3600)%10)) ); 1972 aCreationMetaDateString.append( ":" ); 1973 aCreationMetaDateString.append( (sal_Char)('0' + ((nDelta/600)%6)) ); 1974 aCreationMetaDateString.append( (sal_Char)('0' + ((nDelta/60)%10)) ); 1975 } 1976 aCreationDateString.append( "'" ); 1977 aID.append( aCreationDateString.getStr(), aCreationDateString.getLength() ); 1978 1979 aInfoValuesOut = aID.makeStringAndClear(); 1980 o_rCString1 = aCreationDateString.makeStringAndClear(); 1981 o_rCString2 = aCreationMetaDateString.makeStringAndClear(); 1982 1983 rtlDigest aDigest = rtl_digest_createMD5(); 1984 OSL_ENSURE( aDigest != NULL, "PDFWriterImpl::computeDocumentIdentifier: cannot obtain a digest object !" ); 1985 if( aDigest ) 1986 { 1987 rtlDigestError nError = rtl_digest_updateMD5( aDigest, &aGMT, sizeof( aGMT ) ); 1988 if( nError == rtl_Digest_E_None ) 1989 nError = rtl_digest_updateMD5( aDigest, aInfoValuesOut.getStr(), aInfoValuesOut.getLength() ); 1990 if( nError == rtl_Digest_E_None ) 1991 { 1992 o_rIdentifier = std::vector< sal_uInt8 >( 16, 0 ); 1993 //the binary form of the doc id is needed for encryption stuff 1994 rtl_digest_getMD5( aDigest, &o_rIdentifier[0], 16 ); 1995 } 1996 } 1997 } 1998 1999 /* i12626 methods */ 2000 /* 2001 check if the Unicode string must be encrypted or not, perform the requested task, 2002 append the string as unicode hex, encrypted if needed 2003 */ 2004 inline void PDFWriterImpl::appendUnicodeTextStringEncrypt( const rtl::OUString& rInString, const sal_Int32 nInObjectNumber, OStringBuffer& rOutBuffer ) 2005 { 2006 rOutBuffer.append( "<" ); 2007 if( m_aContext.Encryption.Encrypt() ) 2008 { 2009 const sal_Unicode* pStr = rInString.getStr(); 2010 sal_Int32 nLen = rInString.getLength(); 2011 //prepare a unicode string, encrypt it 2012 if( checkEncryptionBufferSize( nLen*2 ) ) 2013 { 2014 enableStringEncryption( nInObjectNumber ); 2015 register sal_uInt8 *pCopy = m_pEncryptionBuffer; 2016 sal_Int32 nChars = 2; 2017 *pCopy++ = 0xFE; 2018 *pCopy++ = 0xFF; 2019 // we need to prepare a byte stream from the unicode string buffer 2020 for( register int i = 0; i < nLen; i++ ) 2021 { 2022 register sal_Unicode aUnChar = pStr[i]; 2023 *pCopy++ = (sal_uInt8)( aUnChar >> 8 ); 2024 *pCopy++ = (sal_uInt8)( aUnChar & 255 ); 2025 nChars += 2; 2026 } 2027 //encrypt in place 2028 rtl_cipher_encodeARCFOUR( m_aCipher, m_pEncryptionBuffer, nChars, m_pEncryptionBuffer, nChars ); 2029 //now append, hexadecimal (appendHex), the encrypted result 2030 for(register int i = 0; i < nChars; i++) 2031 appendHex( m_pEncryptionBuffer[i], rOutBuffer ); 2032 } 2033 } 2034 else 2035 appendUnicodeTextString( rInString, rOutBuffer ); 2036 rOutBuffer.append( ">" ); 2037 } 2038 2039 inline void PDFWriterImpl::appendLiteralStringEncrypt( rtl::OStringBuffer& rInString, const sal_Int32 nInObjectNumber, rtl::OStringBuffer& rOutBuffer ) 2040 { 2041 rOutBuffer.append( "(" ); 2042 sal_Int32 nChars = rInString.getLength(); 2043 //check for encryption, if ok, encrypt the string, then convert with appndLiteralString 2044 if( m_aContext.Encryption.Encrypt() && checkEncryptionBufferSize( nChars ) ) 2045 { 2046 //encrypt the string in a buffer, then append it 2047 enableStringEncryption( nInObjectNumber ); 2048 rtl_cipher_encodeARCFOUR( m_aCipher, rInString.getStr(), nChars, m_pEncryptionBuffer, nChars ); 2049 appendLiteralString( (const sal_Char*)m_pEncryptionBuffer, nChars, rOutBuffer ); 2050 } 2051 else 2052 appendLiteralString( rInString.getStr(), nChars , rOutBuffer ); 2053 rOutBuffer.append( ")" ); 2054 } 2055 2056 inline void PDFWriterImpl::appendLiteralStringEncrypt( const rtl::OString& rInString, const sal_Int32 nInObjectNumber, rtl::OStringBuffer& rOutBuffer ) 2057 { 2058 rtl::OStringBuffer aBufferString( rInString ); 2059 appendLiteralStringEncrypt( aBufferString, nInObjectNumber, rOutBuffer); 2060 } 2061 2062 void PDFWriterImpl::appendLiteralStringEncrypt( const rtl::OUString& rInString, const sal_Int32 nInObjectNumber, rtl::OStringBuffer& rOutBuffer, rtl_TextEncoding nEnc ) 2063 { 2064 rtl::OString aBufferString( rtl::OUStringToOString( rInString, nEnc ) ); 2065 sal_Int32 nLen = aBufferString.getLength(); 2066 rtl::OStringBuffer aBuf( nLen ); 2067 const sal_Char* pT = aBufferString.getStr(); 2068 2069 for( sal_Int32 i = 0; i < nLen; i++, pT++ ) 2070 { 2071 if( (*pT & 0x80) == 0 ) 2072 aBuf.append( *pT ); 2073 else 2074 { 2075 aBuf.append( '<' ); 2076 appendHex( *pT, aBuf ); 2077 aBuf.append( '>' ); 2078 } 2079 } 2080 aBufferString = aBuf.makeStringAndClear(); 2081 appendLiteralStringEncrypt( aBufferString, nInObjectNumber, rOutBuffer); 2082 } 2083 2084 /* end i12626 methods */ 2085 2086 void PDFWriterImpl::emitComment( const char* pComment ) 2087 { 2088 OStringBuffer aLine( 64 ); 2089 aLine.append( "% " ); 2090 aLine.append( (const sal_Char*)pComment ); 2091 aLine.append( "\n" ); 2092 writeBuffer( aLine.getStr(), aLine.getLength() ); 2093 } 2094 2095 bool PDFWriterImpl::compressStream( SvMemoryStream* pStream ) 2096 { 2097 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 2098 pStream->Seek( STREAM_SEEK_TO_END ); 2099 sal_uLong nEndPos = pStream->Tell(); 2100 pStream->Seek( STREAM_SEEK_TO_BEGIN ); 2101 ZCodec* pCodec = new ZCodec( 0x4000, 0x4000 ); 2102 SvMemoryStream aStream; 2103 pCodec->BeginCompression(); 2104 pCodec->Write( aStream, (const sal_uInt8*)pStream->GetData(), nEndPos ); 2105 pCodec->EndCompression(); 2106 delete pCodec; 2107 nEndPos = aStream.Tell(); 2108 pStream->Seek( STREAM_SEEK_TO_BEGIN ); 2109 aStream.Seek( STREAM_SEEK_TO_BEGIN ); 2110 pStream->SetStreamSize( nEndPos ); 2111 pStream->Write( aStream.GetData(), nEndPos ); 2112 return true; 2113 #else 2114 (void)pStream; 2115 return false; 2116 #endif 2117 } 2118 2119 void PDFWriterImpl::beginCompression() 2120 { 2121 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 2122 m_pCodec = new ZCodec( 0x4000, 0x4000 ); 2123 m_pMemStream = new SvMemoryStream(); 2124 m_pCodec->BeginCompression(); 2125 #endif 2126 } 2127 2128 void PDFWriterImpl::endCompression() 2129 { 2130 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 2131 if( m_pCodec ) 2132 { 2133 m_pCodec->EndCompression(); 2134 delete m_pCodec; 2135 m_pCodec = NULL; 2136 sal_uInt64 nLen = m_pMemStream->Tell(); 2137 m_pMemStream->Seek( 0 ); 2138 writeBuffer( m_pMemStream->GetData(), nLen ); 2139 delete m_pMemStream; 2140 m_pMemStream = NULL; 2141 } 2142 #endif 2143 } 2144 2145 bool PDFWriterImpl::writeBuffer( const void* pBuffer, sal_uInt64 nBytes ) 2146 { 2147 if( ! m_bOpen ) // we are already down the drain 2148 return false; 2149 2150 if( ! nBytes ) // huh ? 2151 return true; 2152 2153 if( m_aOutputStreams.begin() != m_aOutputStreams.end() ) 2154 { 2155 m_aOutputStreams.front().m_pStream->Seek( STREAM_SEEK_TO_END ); 2156 m_aOutputStreams.front().m_pStream->Write( pBuffer, sal::static_int_cast<sal_Size>(nBytes) ); 2157 return true; 2158 } 2159 2160 sal_uInt64 nWritten; 2161 if( m_pCodec ) 2162 { 2163 m_pCodec->Write( *m_pMemStream, static_cast<const sal_uInt8*>(pBuffer), (sal_uLong)nBytes ); 2164 nWritten = nBytes; 2165 } 2166 else 2167 { 2168 sal_Bool buffOK = sal_True; 2169 if( m_bEncryptThisStream ) 2170 { 2171 /* implement the encryption part of the PDF spec encryption algorithm 3.1 */ 2172 if( ( buffOK = checkEncryptionBufferSize( static_cast<sal_Int32>(nBytes) ) ) != sal_False ) 2173 rtl_cipher_encodeARCFOUR( m_aCipher, 2174 (sal_uInt8*)pBuffer, static_cast<sal_Size>(nBytes), 2175 m_pEncryptionBuffer, static_cast<sal_Size>(nBytes) ); 2176 } 2177 2178 const void* pWriteBuffer = ( m_bEncryptThisStream && buffOK ) ? m_pEncryptionBuffer : pBuffer; 2179 if( m_aDocDigest ) 2180 rtl_digest_updateMD5( m_aDocDigest, pWriteBuffer, static_cast<sal_uInt32>(nBytes) ); 2181 2182 if( osl_writeFile( m_aFile, 2183 pWriteBuffer, 2184 nBytes, &nWritten ) != osl_File_E_None ) 2185 nWritten = 0; 2186 2187 if( nWritten != nBytes ) 2188 { 2189 osl_closeFile( m_aFile ); 2190 m_bOpen = false; 2191 } 2192 } 2193 2194 return nWritten == nBytes; 2195 } 2196 2197 OutputDevice* PDFWriterImpl::getReferenceDevice() 2198 { 2199 if( ! m_pReferenceDevice ) 2200 { 2201 VirtualDevice* pVDev = new VirtualDevice( 0 ); 2202 2203 m_pReferenceDevice = pVDev; 2204 2205 if( m_aContext.DPIx == 0 || m_aContext.DPIy == 0 ) 2206 pVDev->SetReferenceDevice( VirtualDevice::REFDEV_MODE_PDF1 ); 2207 else 2208 pVDev->SetReferenceDevice( m_aContext.DPIx, m_aContext.DPIy ); 2209 2210 pVDev->SetOutputSizePixel( Size( 640, 480 ) ); 2211 pVDev->SetMapMode( MAP_MM ); 2212 2213 m_pReferenceDevice->mpPDFWriter = this; 2214 m_pReferenceDevice->ImplUpdateFontData( sal_True ); 2215 } 2216 return m_pReferenceDevice; 2217 } 2218 2219 class ImplPdfBuiltinFontData : public ImplFontData 2220 { 2221 private: 2222 const PDFWriterImpl::BuiltinFont& mrBuiltin; 2223 2224 public: 2225 enum {PDF_FONT_MAGIC = 0xBDFF0A1C }; 2226 ImplPdfBuiltinFontData( const PDFWriterImpl::BuiltinFont& ); 2227 const PDFWriterImpl::BuiltinFont* GetBuiltinFont() const { return &mrBuiltin; } 2228 2229 virtual ImplFontData* Clone() const { return new ImplPdfBuiltinFontData(*this); } 2230 virtual ImplFontEntry* CreateFontInstance( ImplFontSelectData& ) const; 2231 virtual sal_IntPtr GetFontId() const { return reinterpret_cast<sal_IntPtr>(&mrBuiltin); } 2232 }; 2233 2234 inline const ImplPdfBuiltinFontData* GetPdfFontData( const ImplFontData* pFontData ) 2235 { 2236 const ImplPdfBuiltinFontData* pFD = NULL; 2237 if( pFontData && pFontData->CheckMagic( ImplPdfBuiltinFontData::PDF_FONT_MAGIC ) ) 2238 pFD = static_cast<const ImplPdfBuiltinFontData*>( pFontData ); 2239 return pFD; 2240 } 2241 2242 static ImplDevFontAttributes GetDevFontAttributes( const PDFWriterImpl::BuiltinFont& rBuiltin ) 2243 { 2244 ImplDevFontAttributes aDFA; 2245 aDFA.maName = String::CreateFromAscii( rBuiltin.m_pName ); 2246 aDFA.maStyleName = String::CreateFromAscii( rBuiltin.m_pStyleName ); 2247 aDFA.meFamily = rBuiltin.m_eFamily; 2248 aDFA.mbSymbolFlag = (rBuiltin.m_eCharSet != RTL_TEXTENCODING_MS_1252 ); 2249 aDFA.mePitch = rBuiltin.m_ePitch; 2250 aDFA.meWeight = rBuiltin.m_eWeight; 2251 aDFA.meItalic = rBuiltin.m_eItalic; 2252 aDFA.meWidthType = rBuiltin.m_eWidthType; 2253 2254 aDFA.mbOrientation = true; 2255 aDFA.mbDevice = true; 2256 aDFA.mnQuality = 50000; 2257 aDFA.mbSubsettable = false; 2258 aDFA.mbEmbeddable = false; 2259 return aDFA; 2260 } 2261 2262 ImplPdfBuiltinFontData::ImplPdfBuiltinFontData( const PDFWriterImpl::BuiltinFont& rBuiltin ) 2263 : ImplFontData( GetDevFontAttributes(rBuiltin), PDF_FONT_MAGIC ), 2264 mrBuiltin( rBuiltin ) 2265 {} 2266 2267 ImplFontEntry* ImplPdfBuiltinFontData::CreateFontInstance( ImplFontSelectData& rFSD ) const 2268 { 2269 ImplFontEntry* pEntry = new ImplFontEntry( rFSD ); 2270 return pEntry; 2271 } 2272 2273 ImplDevFontList* PDFWriterImpl::filterDevFontList( ImplDevFontList* pFontList ) 2274 { 2275 DBG_ASSERT( m_aSubsets.size() == 0, "Fonts changing during PDF generation, document will be invalid" ); 2276 ImplDevFontList* pFiltered = pFontList->Clone( true, true ); 2277 2278 // append the PDF builtin fonts 2279 if( !m_bIsPDF_A1 && !m_bEmbedStandardFonts) 2280 for( unsigned int i = 0; i < sizeof(m_aBuiltinFonts)/sizeof(m_aBuiltinFonts[0]); i++ ) 2281 { 2282 ImplFontData* pNewData = new ImplPdfBuiltinFontData( m_aBuiltinFonts[i] ); 2283 pFiltered->Add( pNewData ); 2284 } 2285 return pFiltered; 2286 } 2287 2288 bool PDFWriterImpl::isBuiltinFont( const ImplFontData* pFont ) const 2289 { 2290 const ImplPdfBuiltinFontData* pFD = GetPdfFontData( pFont ); 2291 return (pFD != NULL); 2292 } 2293 2294 void PDFWriterImpl::getFontMetric( ImplFontSelectData* pSelect, ImplFontMetricData* pMetric ) const 2295 { 2296 const ImplPdfBuiltinFontData* pFD = GetPdfFontData( pSelect->mpFontData ); 2297 if( !pFD ) 2298 return; 2299 const BuiltinFont* pBuiltinFont = pFD->GetBuiltinFont(); 2300 2301 pMetric->mnOrientation = sal::static_int_cast<short>(pSelect->mnOrientation); 2302 pMetric->meFamily = pBuiltinFont->m_eFamily; 2303 pMetric->mePitch = pBuiltinFont->m_ePitch; 2304 pMetric->meWeight = pBuiltinFont->m_eWeight; 2305 pMetric->meItalic = pBuiltinFont->m_eItalic; 2306 pMetric->mbSymbolFlag = pFD->IsSymbolFont(); 2307 pMetric->mnWidth = pSelect->mnHeight; 2308 pMetric->mnAscent = ( pSelect->mnHeight * +pBuiltinFont->m_nAscent + 500 ) / 1000; 2309 pMetric->mnDescent = ( pSelect->mnHeight * -pBuiltinFont->m_nDescent + 500 ) / 1000; 2310 pMetric->mnIntLeading = 0; 2311 pMetric->mnExtLeading = 0; 2312 pMetric->mnSlant = 0; 2313 pMetric->mbScalableFont = true; 2314 pMetric->mbDevice = true; 2315 } 2316 2317 // ----------------------------------------------------------------------- 2318 2319 namespace vcl { 2320 2321 class PDFSalLayout : public GenericSalLayout 2322 { 2323 PDFWriterImpl& mrPDFWriterImpl; 2324 const PDFWriterImpl::BuiltinFont& mrBuiltinFont; 2325 bool mbIsSymbolFont; 2326 long mnPixelPerEM; 2327 String maOrigText; 2328 2329 public: 2330 PDFSalLayout( PDFWriterImpl&, 2331 const PDFWriterImpl::BuiltinFont&, 2332 long nPixelPerEM, int nOrientation ); 2333 2334 void SetText( const String& rText ) { maOrigText = rText; } 2335 virtual bool LayoutText( ImplLayoutArgs& ); 2336 virtual void InitFont() const; 2337 virtual void DrawText( SalGraphics& ) const; 2338 }; 2339 2340 } 2341 2342 // ----------------------------------------------------------------------- 2343 2344 PDFSalLayout::PDFSalLayout( PDFWriterImpl& rPDFWriterImpl, 2345 const PDFWriterImpl::BuiltinFont& rBuiltinFont, 2346 long nPixelPerEM, int nOrientation ) 2347 : mrPDFWriterImpl( rPDFWriterImpl ), 2348 mrBuiltinFont( rBuiltinFont ), 2349 mnPixelPerEM( nPixelPerEM ) 2350 { 2351 mbIsSymbolFont = (rBuiltinFont.m_eCharSet != RTL_TEXTENCODING_MS_1252); 2352 SetOrientation( nOrientation ); 2353 } 2354 2355 // ----------------------------------------------------------------------- 2356 2357 bool PDFSalLayout::LayoutText( ImplLayoutArgs& rArgs ) 2358 { 2359 const String aText( rArgs.mpStr+rArgs.mnMinCharPos, sal::static_int_cast<xub_StrLen>(rArgs.mnEndCharPos-rArgs.mnMinCharPos) ); 2360 SetText( aText ); 2361 SetUnitsPerPixel( 1000 ); 2362 2363 rtl_UnicodeToTextConverter aConv = rtl_createTextToUnicodeConverter( mrBuiltinFont.m_eCharSet ); 2364 2365 Point aNewPos( 0, 0 ); 2366 bool bRightToLeft; 2367 for( int nCharPos = -1; rArgs.GetNextPos( &nCharPos, &bRightToLeft ); ) 2368 { 2369 // TODO: handle unicode surrogates 2370 // on the other hand the PDF builtin fonts don't support them anyway 2371 sal_Unicode cChar = rArgs.mpStr[ nCharPos ]; 2372 if( bRightToLeft ) 2373 cChar = static_cast<sal_Unicode>(GetMirroredChar( cChar )); 2374 2375 if( 1 ) // TODO: shortcut for ASCII? 2376 { 2377 sal_Char aBuf[4]; 2378 sal_uInt32 nInfo; 2379 sal_Size nSrcCvtChars; 2380 2381 sal_Size nConv = rtl_convertUnicodeToText( aConv, 2382 NULL, 2383 &cChar, 1, 2384 aBuf, sizeof(aBuf)/sizeof(*aBuf), 2385 RTL_UNICODETOTEXT_FLAGS_UNDEFINED_ERROR, 2386 &nInfo, &nSrcCvtChars ); 2387 // check whether conversion was possible 2388 // else fallback font is needed as the standard fonts 2389 // are handled via WinAnsi encoding 2390 if( nConv > 0 ) 2391 cChar = ((sal_Unicode)aBuf[0]) & 0x00ff; 2392 } 2393 if( cChar & 0xff00 ) 2394 { 2395 cChar = 0; // NotDef glyph 2396 rArgs.NeedFallback( nCharPos, bRightToLeft ); 2397 } 2398 2399 long nGlyphWidth = (long)mrBuiltinFont.m_aWidths[cChar] * mnPixelPerEM; 2400 long nGlyphFlags = 0; // builtin fonts don't have diacritic glyphs 2401 if( bRightToLeft ) 2402 nGlyphFlags |= GlyphItem::IS_RTL_GLYPH; 2403 // TODO: get kerning from builtin fonts 2404 GlyphItem aGI( nCharPos, cChar, aNewPos, nGlyphFlags, nGlyphWidth ); 2405 AppendGlyph( aGI ); 2406 2407 aNewPos.X() += nGlyphWidth; 2408 } 2409 2410 rtl_destroyUnicodeToTextConverter( aConv ); 2411 2412 return true; 2413 } 2414 2415 // ----------------------------------------------------------------------- 2416 2417 void PDFSalLayout::InitFont() const 2418 { 2419 // TODO: recreate font with all its attributes 2420 } 2421 2422 // ----------------------------------------------------------------------- 2423 2424 void PDFSalLayout::DrawText( SalGraphics& ) const 2425 { 2426 mrPDFWriterImpl.drawLayout( *const_cast<PDFSalLayout*>(this), maOrigText, true ); 2427 } 2428 2429 // ----------------------------------------------------------------------- 2430 2431 SalLayout* PDFWriterImpl::GetTextLayout( ImplLayoutArgs& rArgs, ImplFontSelectData* pSelect ) 2432 { 2433 DBG_ASSERT( (pSelect->mpFontData != NULL), 2434 "PDFWriterImpl::GetTextLayout mpFontData is NULL" ); 2435 2436 const ImplPdfBuiltinFontData* pFD = GetPdfFontData( pSelect->mpFontData ); 2437 if( !pFD ) 2438 return NULL; 2439 const BuiltinFont* pBuiltinFont = pFD->GetBuiltinFont(); 2440 2441 long nPixelPerEM = pSelect->mnWidth ? pSelect->mnWidth : pSelect->mnHeight; 2442 int nOrientation = pSelect->mnOrientation; 2443 PDFSalLayout* pLayout = new PDFSalLayout( *this, *pBuiltinFont, nPixelPerEM, nOrientation ); 2444 pLayout->SetText( rArgs.mpStr ); 2445 return pLayout; 2446 } 2447 2448 sal_Int32 PDFWriterImpl::newPage( sal_Int32 nPageWidth, sal_Int32 nPageHeight, PDFWriter::Orientation eOrientation ) 2449 { 2450 endPage(); 2451 m_nCurrentPage = m_aPages.size(); 2452 m_aPages.push_back( PDFPage(this, nPageWidth, nPageHeight, eOrientation ) ); 2453 m_aPages.back().m_nPageIndex = m_nCurrentPage; 2454 m_aPages.back().beginStream(); 2455 2456 // setup global graphics state 2457 // linewidth is "1 pixel" by default 2458 OStringBuffer aBuf( 16 ); 2459 appendDouble( 72.0/double(getReferenceDevice()->ImplGetDPIX()), aBuf ); 2460 aBuf.append( " w\n" ); 2461 writeBuffer( aBuf.getStr(), aBuf.getLength() ); 2462 2463 return m_nCurrentPage; 2464 } 2465 2466 void PDFWriterImpl::endPage() 2467 { 2468 if( m_aPages.begin() != m_aPages.end() ) 2469 { 2470 // close eventual MC sequence 2471 endStructureElementMCSeq(); 2472 2473 // sanity check 2474 if( m_aOutputStreams.begin() != m_aOutputStreams.end() ) 2475 { 2476 DBG_ERROR( "redirection across pages !!!" ); 2477 m_aOutputStreams.clear(); // leak ! 2478 m_aMapMode.SetOrigin( Point() ); 2479 } 2480 2481 m_aGraphicsStack.clear(); 2482 m_aGraphicsStack.push_back( GraphicsState() ); 2483 2484 // this should pop the PDF graphics stack if necessary 2485 updateGraphicsState(); 2486 2487 m_aPages.back().endStream(); 2488 2489 // reset the default font 2490 Font aFont; 2491 aFont.SetName( String( RTL_CONSTASCII_USTRINGPARAM( "Times" ) ) ); 2492 aFont.SetSize( Size( 0, 12 ) ); 2493 2494 m_aCurrentPDFState = m_aGraphicsStack.front(); 2495 m_aGraphicsStack.front().m_aFont = aFont; 2496 2497 for( std::list<BitmapEmit>::iterator it = m_aBitmaps.begin(); 2498 it != m_aBitmaps.end(); ++it ) 2499 { 2500 if( ! it->m_aBitmap.IsEmpty() ) 2501 { 2502 writeBitmapObject( *it ); 2503 it->m_aBitmap = BitmapEx(); 2504 } 2505 } 2506 for( std::list<JPGEmit>::iterator jpeg = m_aJPGs.begin(); jpeg != m_aJPGs.end(); ++jpeg ) 2507 { 2508 if( jpeg->m_pStream ) 2509 { 2510 writeJPG( *jpeg ); 2511 delete jpeg->m_pStream; 2512 jpeg->m_pStream = NULL; 2513 jpeg->m_aMask = Bitmap(); 2514 } 2515 } 2516 for( std::list<TransparencyEmit>::iterator t = m_aTransparentObjects.begin(); 2517 t != m_aTransparentObjects.end(); ++t ) 2518 { 2519 if( t->m_pContentStream ) 2520 { 2521 writeTransparentObject( *t ); 2522 delete t->m_pContentStream; 2523 t->m_pContentStream = NULL; 2524 } 2525 } 2526 } 2527 } 2528 2529 sal_Int32 PDFWriterImpl::createObject() 2530 { 2531 m_aObjects.push_back( ~0U ); 2532 return m_aObjects.size(); 2533 } 2534 2535 bool PDFWriterImpl::updateObject( sal_Int32 n ) 2536 { 2537 if( ! m_bOpen ) 2538 return false; 2539 2540 sal_uInt64 nOffset = ~0U; 2541 oslFileError aError = osl_getFilePos( m_aFile, &nOffset ); 2542 DBG_ASSERT( aError == osl_File_E_None, "could not register object" ); 2543 if( aError != osl_File_E_None ) 2544 { 2545 osl_closeFile( m_aFile ); 2546 m_bOpen = false; 2547 } 2548 m_aObjects[ n-1 ] = nOffset; 2549 return aError == osl_File_E_None; 2550 } 2551 2552 #define CHECK_RETURN( x ) if( !(x) ) return 0 2553 2554 sal_Int32 PDFWriterImpl::emitStructParentTree( sal_Int32 nObject ) 2555 { 2556 if( nObject > 0 ) 2557 { 2558 OStringBuffer aLine( 1024 ); 2559 2560 aLine.append( nObject ); 2561 aLine.append( " 0 obj\n" 2562 "<</Nums[\n" ); 2563 sal_Int32 nTreeItems = m_aStructParentTree.size(); 2564 for( sal_Int32 n = 0; n < nTreeItems; n++ ) 2565 { 2566 aLine.append( n ); 2567 aLine.append( ' ' ); 2568 aLine.append( m_aStructParentTree[n] ); 2569 aLine.append( "\n" ); 2570 } 2571 aLine.append( "]>>\nendobj\n\n" ); 2572 CHECK_RETURN( updateObject( nObject ) ); 2573 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 2574 } 2575 return nObject; 2576 } 2577 2578 const sal_Char* PDFWriterImpl::getAttributeTag( PDFWriter::StructAttribute eAttr ) 2579 { 2580 static std::map< PDFWriter::StructAttribute, const char* > aAttributeStrings; 2581 // fill maps once 2582 if( aAttributeStrings.empty() ) 2583 { 2584 aAttributeStrings[ PDFWriter::Placement ] = "Placement"; 2585 aAttributeStrings[ PDFWriter::WritingMode ] = "WritingMode"; 2586 aAttributeStrings[ PDFWriter::SpaceBefore ] = "SpaceBefore"; 2587 aAttributeStrings[ PDFWriter::SpaceAfter ] = "SpaceAfter"; 2588 aAttributeStrings[ PDFWriter::StartIndent ] = "StartIndent"; 2589 aAttributeStrings[ PDFWriter::EndIndent ] = "EndIndent"; 2590 aAttributeStrings[ PDFWriter::TextIndent ] = "TextIndent"; 2591 aAttributeStrings[ PDFWriter::TextAlign ] = "TextAlign"; 2592 aAttributeStrings[ PDFWriter::Width ] = "Width"; 2593 aAttributeStrings[ PDFWriter::Height ] = "Height"; 2594 aAttributeStrings[ PDFWriter::BlockAlign ] = "BlockAlign"; 2595 aAttributeStrings[ PDFWriter::InlineAlign ] = "InlineAlign"; 2596 aAttributeStrings[ PDFWriter::LineHeight ] = "LineHeight"; 2597 aAttributeStrings[ PDFWriter::BaselineShift ] = "BaselineShift"; 2598 aAttributeStrings[ PDFWriter::TextDecorationType ] = "TextDecorationType"; 2599 aAttributeStrings[ PDFWriter::ListNumbering ] = "ListNumbering"; 2600 aAttributeStrings[ PDFWriter::RowSpan ] = "RowSpan"; 2601 aAttributeStrings[ PDFWriter::ColSpan ] = "ColSpan"; 2602 aAttributeStrings[ PDFWriter::LinkAnnotation ] = "LinkAnnotation"; 2603 } 2604 2605 std::map< PDFWriter::StructAttribute, const char* >::const_iterator it = 2606 aAttributeStrings.find( eAttr ); 2607 2608 #if OSL_DEBUG_LEVEL > 1 2609 if( it == aAttributeStrings.end() ) 2610 fprintf( stderr, "invalid PDFWriter::StructAttribute %d\n", eAttr ); 2611 #endif 2612 2613 return it != aAttributeStrings.end() ? it->second : ""; 2614 } 2615 2616 const sal_Char* PDFWriterImpl::getAttributeValueTag( PDFWriter::StructAttributeValue eVal ) 2617 { 2618 static std::map< PDFWriter::StructAttributeValue, const char* > aValueStrings; 2619 2620 if( aValueStrings.empty() ) 2621 { 2622 aValueStrings[ PDFWriter::NONE ] = "None"; 2623 aValueStrings[ PDFWriter::Block ] = "Block"; 2624 aValueStrings[ PDFWriter::Inline ] = "Inline"; 2625 aValueStrings[ PDFWriter::Before ] = "Before"; 2626 aValueStrings[ PDFWriter::After ] = "After"; 2627 aValueStrings[ PDFWriter::Start ] = "Start"; 2628 aValueStrings[ PDFWriter::End ] = "End"; 2629 aValueStrings[ PDFWriter::LrTb ] = "LrTb"; 2630 aValueStrings[ PDFWriter::RlTb ] = "RlTb"; 2631 aValueStrings[ PDFWriter::TbRl ] = "TbRl"; 2632 aValueStrings[ PDFWriter::Center ] = "Center"; 2633 aValueStrings[ PDFWriter::Justify ] = "Justify"; 2634 aValueStrings[ PDFWriter::Auto ] = "Auto"; 2635 aValueStrings[ PDFWriter::Middle ] = "Middle"; 2636 aValueStrings[ PDFWriter::Normal ] = "Normal"; 2637 aValueStrings[ PDFWriter::Underline ] = "Underline"; 2638 aValueStrings[ PDFWriter::Overline ] = "Overline"; 2639 aValueStrings[ PDFWriter::LineThrough ] = "LineThrough"; 2640 aValueStrings[ PDFWriter::Disc ] = "Disc"; 2641 aValueStrings[ PDFWriter::Circle ] = "Circle"; 2642 aValueStrings[ PDFWriter::Square ] = "Square"; 2643 aValueStrings[ PDFWriter::Decimal ] = "Decimal"; 2644 aValueStrings[ PDFWriter::UpperRoman ] = "UpperRoman"; 2645 aValueStrings[ PDFWriter::LowerRoman ] = "LowerRoman"; 2646 aValueStrings[ PDFWriter::UpperAlpha ] = "UpperAlpha"; 2647 aValueStrings[ PDFWriter::LowerAlpha ] = "LowerAlpha"; 2648 } 2649 2650 std::map< PDFWriter::StructAttributeValue, const char* >::const_iterator it = 2651 aValueStrings.find( eVal ); 2652 2653 #if OSL_DEBUG_LEVEL > 1 2654 if( it == aValueStrings.end() ) 2655 fprintf( stderr, "invalid PDFWriter::StructAttributeValue %d\n", eVal ); 2656 #endif 2657 2658 return it != aValueStrings.end() ? it->second : ""; 2659 } 2660 2661 static void appendStructureAttributeLine( PDFWriter::StructAttribute i_eAttr, const PDFWriterImpl::PDFStructureAttribute& i_rVal, OStringBuffer& o_rLine, bool i_bIsFixedInt ) 2662 { 2663 o_rLine.append( "/" ); 2664 o_rLine.append( PDFWriterImpl::getAttributeTag( i_eAttr ) ); 2665 2666 if( i_rVal.eValue != PDFWriter::Invalid ) 2667 { 2668 o_rLine.append( "/" ); 2669 o_rLine.append( PDFWriterImpl::getAttributeValueTag( i_rVal.eValue ) ); 2670 } 2671 else 2672 { 2673 // numerical value 2674 o_rLine.append( " " ); 2675 if( i_bIsFixedInt ) 2676 appendFixedInt( i_rVal.nValue, o_rLine ); 2677 else 2678 o_rLine.append( i_rVal.nValue ); 2679 } 2680 o_rLine.append( "\n" ); 2681 } 2682 2683 OString PDFWriterImpl::emitStructureAttributes( PDFStructureElement& i_rEle ) 2684 { 2685 // create layout, list and table attribute sets 2686 OStringBuffer aLayout(256), aList(64), aTable(64); 2687 for( PDFStructAttributes::const_iterator it = i_rEle.m_aAttributes.begin(); 2688 it != i_rEle.m_aAttributes.end(); ++it ) 2689 { 2690 if( it->first == PDFWriter::ListNumbering ) 2691 appendStructureAttributeLine( it->first, it->second, aList, true ); 2692 else if( it->first == PDFWriter::RowSpan || 2693 it->first == PDFWriter::ColSpan ) 2694 appendStructureAttributeLine( it->first, it->second, aTable, false ); 2695 else if( it->first == PDFWriter::LinkAnnotation ) 2696 { 2697 sal_Int32 nLink = it->second.nValue; 2698 std::map< sal_Int32, sal_Int32 >::const_iterator link_it = 2699 m_aLinkPropertyMap.find( nLink ); 2700 if( link_it != m_aLinkPropertyMap.end() ) 2701 nLink = link_it->second; 2702 if( nLink >= 0 && nLink < (sal_Int32)m_aLinks.size() ) 2703 { 2704 // update struct parent of link 2705 OStringBuffer aStructParentEntry( 32 ); 2706 aStructParentEntry.append( i_rEle.m_nObject ); 2707 aStructParentEntry.append( " 0 R" ); 2708 m_aStructParentTree.push_back( aStructParentEntry.makeStringAndClear() ); 2709 m_aLinks[ nLink ].m_nStructParent = m_aStructParentTree.size()-1; 2710 2711 sal_Int32 nRefObject = createObject(); 2712 OStringBuffer aRef( 256 ); 2713 aRef.append( nRefObject ); 2714 aRef.append( " 0 obj\n" 2715 "<</Type/OBJR/Obj " ); 2716 aRef.append( m_aLinks[ nLink ].m_nObject ); 2717 aRef.append( " 0 R>>\n" 2718 "endobj\n\n" 2719 ); 2720 updateObject( nRefObject ); 2721 writeBuffer( aRef.getStr(), aRef.getLength() ); 2722 2723 i_rEle.m_aKids.push_back( PDFStructureElementKid( nRefObject ) ); 2724 } 2725 else 2726 { 2727 DBG_ERROR( "unresolved link id for Link structure" ); 2728 #if OSL_DEBUG_LEVEL > 1 2729 fprintf( stderr, "unresolved link id %" SAL_PRIdINT32 " for Link structure\n", nLink ); 2730 { 2731 OStringBuffer aLine( "unresolved link id " ); 2732 aLine.append( nLink ); 2733 aLine.append( " for Link structure" ); 2734 emitComment( aLine.getStr() ); 2735 } 2736 #endif 2737 } 2738 } 2739 else 2740 appendStructureAttributeLine( it->first, it->second, aLayout, true ); 2741 } 2742 if( ! i_rEle.m_aBBox.IsEmpty() ) 2743 { 2744 aLayout.append( "/BBox[" ); 2745 appendFixedInt( i_rEle.m_aBBox.Left(), aLayout ); 2746 aLayout.append( " " ); 2747 appendFixedInt( i_rEle.m_aBBox.Top(), aLayout ); 2748 aLayout.append( " " ); 2749 appendFixedInt( i_rEle.m_aBBox.Right(), aLayout ); 2750 aLayout.append( " " ); 2751 appendFixedInt( i_rEle.m_aBBox.Bottom(), aLayout ); 2752 aLayout.append( "]\n" ); 2753 } 2754 2755 std::vector< sal_Int32 > aAttribObjects; 2756 if( aLayout.getLength() ) 2757 { 2758 aAttribObjects.push_back( createObject() ); 2759 updateObject( aAttribObjects.back() ); 2760 OStringBuffer aObj( 64 ); 2761 aObj.append( aAttribObjects.back() ); 2762 aObj.append( " 0 obj\n" 2763 "<</O/Layout\n" ); 2764 aLayout.append( ">>\nendobj\n\n" ); 2765 writeBuffer( aObj.getStr(), aObj.getLength() ); 2766 writeBuffer( aLayout.getStr(), aLayout.getLength() ); 2767 } 2768 if( aList.getLength() ) 2769 { 2770 aAttribObjects.push_back( createObject() ); 2771 updateObject( aAttribObjects.back() ); 2772 OStringBuffer aObj( 64 ); 2773 aObj.append( aAttribObjects.back() ); 2774 aObj.append( " 0 obj\n" 2775 "<</O/List\n" ); 2776 aList.append( ">>\nendobj\n\n" ); 2777 writeBuffer( aObj.getStr(), aObj.getLength() ); 2778 writeBuffer( aList.getStr(), aList.getLength() ); 2779 } 2780 if( aTable.getLength() ) 2781 { 2782 aAttribObjects.push_back( createObject() ); 2783 updateObject( aAttribObjects.back() ); 2784 OStringBuffer aObj( 64 ); 2785 aObj.append( aAttribObjects.back() ); 2786 aObj.append( " 0 obj\n" 2787 "<</O/Table\n" ); 2788 aTable.append( ">>\nendobj\n\n" ); 2789 writeBuffer( aObj.getStr(), aObj.getLength() ); 2790 writeBuffer( aTable.getStr(), aTable.getLength() ); 2791 } 2792 2793 OStringBuffer aRet( 64 ); 2794 if( aAttribObjects.size() > 1 ) 2795 aRet.append( " [" ); 2796 for( std::vector< sal_Int32 >::const_iterator at_it = aAttribObjects.begin(); 2797 at_it != aAttribObjects.end(); ++at_it ) 2798 { 2799 aRet.append( " " ); 2800 aRet.append( *at_it ); 2801 aRet.append( " 0 R" ); 2802 } 2803 if( aAttribObjects.size() > 1 ) 2804 aRet.append( " ]" ); 2805 return aRet.makeStringAndClear(); 2806 } 2807 2808 sal_Int32 PDFWriterImpl::emitStructure( PDFStructureElement& rEle ) 2809 { 2810 if( 2811 // do not emit NonStruct and its children 2812 rEle.m_eType == PDFWriter::NonStructElement && 2813 rEle.m_nOwnElement != rEle.m_nParentElement // but of course emit the struct tree root 2814 ) 2815 return 0; 2816 2817 for( std::list< sal_Int32 >::const_iterator it = rEle.m_aChildren.begin(); it != rEle.m_aChildren.end(); ++it ) 2818 { 2819 if( *it > 0 && *it < sal_Int32(m_aStructure.size()) ) 2820 { 2821 PDFStructureElement& rChild = m_aStructure[ *it ]; 2822 if( rChild.m_eType != PDFWriter::NonStructElement ) 2823 { 2824 if( rChild.m_nParentElement == rEle.m_nOwnElement ) 2825 emitStructure( rChild ); 2826 else 2827 { 2828 DBG_ERROR( "PDFWriterImpl::emitStructure: invalid child structure element" ); 2829 #if OSL_DEBUG_LEVEL > 1 2830 fprintf( stderr, "PDFWriterImpl::emitStructure: invalid child structure elemnt with id %" SAL_PRIdINT32 "\n", *it ); 2831 #endif 2832 } 2833 } 2834 } 2835 else 2836 { 2837 DBG_ERROR( "PDFWriterImpl::emitStructure: invalid child structure id" ); 2838 #if OSL_DEBUG_LEVEL > 1 2839 fprintf( stderr, "PDFWriterImpl::emitStructure: invalid child structure id %" SAL_PRIdINT32 "\n", *it ); 2840 #endif 2841 } 2842 } 2843 2844 OStringBuffer aLine( 512 ); 2845 aLine.append( rEle.m_nObject ); 2846 aLine.append( " 0 obj\n" 2847 "<</Type" ); 2848 sal_Int32 nParentTree = -1; 2849 if( rEle.m_nOwnElement == rEle.m_nParentElement ) 2850 { 2851 nParentTree = createObject(); 2852 CHECK_RETURN( nParentTree ); 2853 aLine.append( "/StructTreeRoot\n" ); 2854 aLine.append( "/ParentTree " ); 2855 aLine.append( nParentTree ); 2856 aLine.append( " 0 R\n" ); 2857 if( ! m_aRoleMap.empty() ) 2858 { 2859 aLine.append( "/RoleMap<<" ); 2860 for( std::hash_map<OString,OString,OStringHash>::const_iterator 2861 it = m_aRoleMap.begin(); it != m_aRoleMap.end(); ++it ) 2862 { 2863 aLine.append( '/' ); 2864 aLine.append(it->first); 2865 aLine.append( '/' ); 2866 aLine.append( it->second ); 2867 aLine.append( '\n' ); 2868 } 2869 aLine.append( ">>\n" ); 2870 } 2871 } 2872 else 2873 { 2874 aLine.append( "/StructElem\n" 2875 "/S/" ); 2876 if( rEle.m_aAlias.getLength() > 0 ) 2877 aLine.append( rEle.m_aAlias ); 2878 else 2879 aLine.append( getStructureTag( rEle.m_eType ) ); 2880 aLine.append( "\n" 2881 "/P " ); 2882 aLine.append( m_aStructure[ rEle.m_nParentElement ].m_nObject ); 2883 aLine.append( " 0 R\n" 2884 "/Pg " ); 2885 aLine.append( rEle.m_nFirstPageObject ); 2886 aLine.append( " 0 R\n" ); 2887 if( rEle.m_aActualText.getLength() ) 2888 { 2889 aLine.append( "/ActualText" ); 2890 appendUnicodeTextStringEncrypt( rEle.m_aActualText, rEle.m_nObject, aLine ); 2891 aLine.append( "\n" ); 2892 } 2893 if( rEle.m_aAltText.getLength() ) 2894 { 2895 aLine.append( "/Alt" ); 2896 appendUnicodeTextStringEncrypt( rEle.m_aAltText, rEle.m_nObject, aLine ); 2897 aLine.append( "\n" ); 2898 } 2899 } 2900 if( ! rEle.m_aBBox.IsEmpty() || rEle.m_aAttributes.size() ) 2901 { 2902 OString aAttribs = emitStructureAttributes( rEle ); 2903 if( aAttribs.getLength() ) 2904 { 2905 aLine.append( "/A" ); 2906 aLine.append( aAttribs ); 2907 aLine.append( "\n" ); 2908 } 2909 } 2910 if( rEle.m_aLocale.Language.getLength() > 0 ) 2911 { 2912 OUStringBuffer aLocBuf( 16 ); 2913 aLocBuf.append( rEle.m_aLocale.Language.toAsciiLowerCase() ); 2914 if( rEle.m_aLocale.Country.getLength() > 0 ) 2915 { 2916 aLocBuf.append( sal_Unicode('-') ); 2917 aLocBuf.append( rEle.m_aLocale.Country ); 2918 } 2919 aLine.append( "/Lang" ); 2920 appendLiteralStringEncrypt( aLocBuf.makeStringAndClear(), rEle.m_nObject, aLine ); 2921 aLine.append( "\n" ); 2922 } 2923 if( ! rEle.m_aKids.empty() ) 2924 { 2925 unsigned int i = 0; 2926 aLine.append( "/K[" ); 2927 for( std::list< PDFStructureElementKid >::const_iterator it = 2928 rEle.m_aKids.begin(); it != rEle.m_aKids.end(); ++it, i++ ) 2929 { 2930 if( it->nMCID == -1 ) 2931 { 2932 aLine.append( it->nObject ); 2933 aLine.append( " 0 R" ); 2934 aLine.append( ( (i & 15) == 15 ) ? "\n" : " " ); 2935 } 2936 else 2937 { 2938 if( it->nObject == rEle.m_nFirstPageObject ) 2939 { 2940 aLine.append( it->nMCID ); 2941 aLine.append( " " ); 2942 } 2943 else 2944 { 2945 aLine.append( "<</Type/MCR/Pg " ); 2946 aLine.append( it->nObject ); 2947 aLine.append( " 0 R /MCID " ); 2948 aLine.append( it->nMCID ); 2949 aLine.append( ">>\n" ); 2950 } 2951 } 2952 } 2953 aLine.append( "]\n" ); 2954 } 2955 aLine.append( ">>\nendobj\n\n" ); 2956 2957 CHECK_RETURN( updateObject( rEle.m_nObject ) ); 2958 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 2959 2960 CHECK_RETURN( emitStructParentTree( nParentTree ) ); 2961 2962 return rEle.m_nObject; 2963 } 2964 2965 bool PDFWriterImpl::emitGradients() 2966 { 2967 for( std::list<GradientEmit>::iterator it = m_aGradients.begin(); 2968 it != m_aGradients.end(); ++it ) 2969 { 2970 CHECK_RETURN( writeGradientFunction( *it ) ); 2971 } 2972 return true; 2973 } 2974 2975 bool PDFWriterImpl::emitTilings() 2976 { 2977 OStringBuffer aTilingObj( 1024 ); 2978 2979 for( std::vector<TilingEmit>::iterator it = m_aTilings.begin(); it != m_aTilings.end(); ++it ) 2980 { 2981 DBG_ASSERT( it->m_pTilingStream, "tiling without stream" ); 2982 if( ! it->m_pTilingStream ) 2983 continue; 2984 2985 aTilingObj.setLength( 0 ); 2986 2987 #if OSL_DEBUG_LEVEL > 1 2988 emitComment( "PDFWriterImpl::emitTilings" ); 2989 #endif 2990 2991 sal_Int32 nX = (sal_Int32)it->m_aRectangle.Left(); 2992 sal_Int32 nY = (sal_Int32)it->m_aRectangle.Top(); 2993 sal_Int32 nW = (sal_Int32)it->m_aRectangle.GetWidth(); 2994 sal_Int32 nH = (sal_Int32)it->m_aRectangle.GetHeight(); 2995 if( it->m_aCellSize.Width() == 0 ) 2996 it->m_aCellSize.Width() = nW; 2997 if( it->m_aCellSize.Height() == 0 ) 2998 it->m_aCellSize.Height() = nH; 2999 3000 bool bDeflate = compressStream( it->m_pTilingStream ); 3001 it->m_pTilingStream->Seek( STREAM_SEEK_TO_END ); 3002 sal_Size nTilingStreamSize = it->m_pTilingStream->Tell(); 3003 it->m_pTilingStream->Seek( STREAM_SEEK_TO_BEGIN ); 3004 3005 // write pattern object 3006 aTilingObj.append( it->m_nObject ); 3007 aTilingObj.append( " 0 obj\n" ); 3008 aTilingObj.append( "<</Type/Pattern/PatternType 1\n" 3009 "/PaintType 1\n" 3010 "/TilingType 2\n" 3011 "/BBox[" ); 3012 appendFixedInt( nX, aTilingObj ); 3013 aTilingObj.append( ' ' ); 3014 appendFixedInt( nY, aTilingObj ); 3015 aTilingObj.append( ' ' ); 3016 appendFixedInt( nX+nW, aTilingObj ); 3017 aTilingObj.append( ' ' ); 3018 appendFixedInt( nY+nH, aTilingObj ); 3019 aTilingObj.append( "]\n" 3020 "/XStep " ); 3021 appendFixedInt( it->m_aCellSize.Width(), aTilingObj ); 3022 aTilingObj.append( "\n" 3023 "/YStep " ); 3024 appendFixedInt( it->m_aCellSize.Height(), aTilingObj ); 3025 aTilingObj.append( "\n" ); 3026 if( it->m_aTransform.matrix[0] != 1.0 || 3027 it->m_aTransform.matrix[1] != 0.0 || 3028 it->m_aTransform.matrix[3] != 0.0 || 3029 it->m_aTransform.matrix[4] != 1.0 || 3030 it->m_aTransform.matrix[2] != 0.0 || 3031 it->m_aTransform.matrix[5] != 0.0 ) 3032 { 3033 aTilingObj.append( "/Matrix [" ); 3034 // TODO: scaling, mirroring on y, etc 3035 appendDouble( it->m_aTransform.matrix[0], aTilingObj ); 3036 aTilingObj.append( ' ' ); 3037 appendDouble( it->m_aTransform.matrix[1], aTilingObj ); 3038 aTilingObj.append( ' ' ); 3039 appendDouble( it->m_aTransform.matrix[3], aTilingObj ); 3040 aTilingObj.append( ' ' ); 3041 appendDouble( it->m_aTransform.matrix[4], aTilingObj ); 3042 aTilingObj.append( ' ' ); 3043 appendDouble( it->m_aTransform.matrix[2], aTilingObj ); 3044 aTilingObj.append( ' ' ); 3045 appendDouble( it->m_aTransform.matrix[5], aTilingObj ); 3046 aTilingObj.append( "]\n" ); 3047 } 3048 aTilingObj.append( "/Resources" ); 3049 it->m_aResources.append( aTilingObj, getFontDictObject() ); 3050 if( bDeflate ) 3051 aTilingObj.append( "/Filter/FlateDecode" ); 3052 aTilingObj.append( "/Length " ); 3053 aTilingObj.append( (sal_Int32)nTilingStreamSize ); 3054 aTilingObj.append( ">>\nstream\n" ); 3055 CHECK_RETURN( updateObject( it->m_nObject ) ); 3056 CHECK_RETURN( writeBuffer( aTilingObj.getStr(), aTilingObj.getLength() ) ); 3057 checkAndEnableStreamEncryption( it->m_nObject ); 3058 nTilingStreamSize = writeBuffer( it->m_pTilingStream->GetData(), nTilingStreamSize ); 3059 delete it->m_pTilingStream; 3060 it->m_pTilingStream = NULL; 3061 if( nTilingStreamSize == 0 ) 3062 return false; 3063 disableStreamEncryption(); 3064 aTilingObj.setLength( 0 ); 3065 aTilingObj.append( "\nendstream\nendobj\n\n" ); 3066 CHECK_RETURN( writeBuffer( aTilingObj.getStr(), aTilingObj.getLength() ) ); 3067 } 3068 return true; 3069 } 3070 3071 sal_Int32 PDFWriterImpl::emitBuiltinFont( const ImplFontData* pFont, sal_Int32 nFontObject ) 3072 { 3073 const ImplPdfBuiltinFontData* pFD = GetPdfFontData( pFont ); 3074 if( !pFD ) 3075 return 0; 3076 const BuiltinFont* pBuiltinFont = pFD->GetBuiltinFont(); 3077 3078 OStringBuffer aLine( 1024 ); 3079 3080 if( nFontObject <= 0 ) 3081 nFontObject = createObject(); 3082 CHECK_RETURN( updateObject( nFontObject ) ); 3083 aLine.append( nFontObject ); 3084 aLine.append( " 0 obj\n" 3085 "<</Type/Font/Subtype/Type1/BaseFont/" ); 3086 appendName( pBuiltinFont->m_pPSName, aLine ); 3087 aLine.append( "\n" ); 3088 if( pBuiltinFont->m_eCharSet == RTL_TEXTENCODING_MS_1252 ) 3089 aLine.append( "/Encoding/WinAnsiEncoding\n" ); 3090 aLine.append( ">>\nendobj\n\n" ); 3091 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 3092 return nFontObject; 3093 } 3094 3095 std::map< sal_Int32, sal_Int32 > PDFWriterImpl::emitSystemFont( const ImplFontData* pFont, EmbedFont& rEmbed ) 3096 { 3097 std::map< sal_Int32, sal_Int32 > aRet; 3098 if( isBuiltinFont( pFont ) ) 3099 { 3100 aRet[ rEmbed.m_nNormalFontID ] = emitBuiltinFont( pFont ); 3101 return aRet; 3102 } 3103 3104 sal_Int32 nFontObject = 0; 3105 sal_Int32 nFontDescriptor = 0; 3106 rtl::OString aSubType( "/Type1" ); 3107 FontSubsetInfo aInfo; 3108 // fill in dummy values 3109 aInfo.m_nAscent = 1000; 3110 aInfo.m_nDescent = 200; 3111 aInfo.m_nCapHeight = 1000; 3112 aInfo.m_aFontBBox = Rectangle( Point( -200, -200 ), Size( 1700, 1700 ) ); 3113 aInfo.m_aPSName = pFont->maName; 3114 sal_Int32 pWidths[256]; 3115 rtl_zeroMemory( pWidths, sizeof(pWidths) ); 3116 if( pFont->IsEmbeddable() ) 3117 { 3118 const unsigned char* pFontData = NULL; 3119 long nFontLen = 0; 3120 sal_Ucs nEncodedCodes[256]; 3121 sal_Int32 pEncWidths[256]; 3122 if( (pFontData = (const unsigned char*)m_pReferenceDevice->mpGraphics->GetEmbedFontData( pFont, nEncodedCodes, pEncWidths, aInfo, &nFontLen )) != NULL ) 3123 { 3124 m_pReferenceDevice->mpGraphics->FreeEmbedFontData( pFontData, nFontLen ); 3125 for( int i = 0; i < 256; i++ ) 3126 { 3127 if( nEncodedCodes[i] >= 32 && nEncodedCodes[i] < 256 ) 3128 { 3129 pWidths[i] = pEncWidths[ i ]; 3130 } 3131 } 3132 } 3133 } 3134 else if( pFont->mbSubsettable ) 3135 { 3136 aSubType = rtl::OString( "/TrueType" ); 3137 Int32Vector aGlyphWidths; 3138 Ucs2UIntMap aUnicodeMap; 3139 m_pReferenceDevice->mpGraphics->GetGlyphWidths( pFont, false, aGlyphWidths, aUnicodeMap ); 3140 3141 OUString aTmpName; 3142 osl_createTempFile( NULL, NULL, &aTmpName.pData ); 3143 sal_Int32 pGlyphIDs[ 256 ]; 3144 sal_uInt8 pEncoding[ 256 ]; 3145 sal_Ucs pUnicodes[ 256 ]; 3146 sal_Int32 pDuWidths[ 256 ]; 3147 3148 memset( pGlyphIDs, 0, sizeof( pGlyphIDs ) ); 3149 memset( pEncoding, 0, sizeof( pEncoding ) ); 3150 memset( pUnicodes, 0, sizeof( pUnicodes ) ); 3151 memset( pDuWidths, 0, sizeof( pDuWidths ) ); 3152 3153 for( sal_Ucs c = 32; c < 256; c++ ) 3154 { 3155 pUnicodes[c] = c; 3156 pEncoding[c] = c; 3157 pGlyphIDs[c] = 0; 3158 if( aUnicodeMap.find( c ) != aUnicodeMap.end() ) 3159 pWidths[ c ] = aGlyphWidths[ aUnicodeMap[ c ] ]; 3160 } 3161 3162 m_pReferenceDevice->mpGraphics->CreateFontSubset( aTmpName, pFont, pGlyphIDs, pEncoding, pDuWidths, 256, aInfo ); 3163 osl_removeFile( aTmpName.pData ); 3164 } 3165 else 3166 { 3167 DBG_ERROR( "system font neither embeddable nor subsettable" ); 3168 } 3169 3170 // write font descriptor 3171 nFontDescriptor = emitFontDescriptor( pFont, aInfo, 0, 0 ); 3172 if( nFontDescriptor ) 3173 { 3174 // write font object 3175 sal_Int32 nObject = createObject(); 3176 if( updateObject( nObject ) ) 3177 { 3178 OStringBuffer aLine( 1024 ); 3179 aLine.append( nObject ); 3180 aLine.append( " 0 obj\n" 3181 "<</Type/Font/Subtype" ); 3182 aLine.append( aSubType ); 3183 aLine.append( "/BaseFont/" ); 3184 appendName( aInfo.m_aPSName, aLine ); 3185 aLine.append( "\n" ); 3186 if( !pFont->mbSymbolFlag ) 3187 aLine.append( "/Encoding/WinAnsiEncoding\n" ); 3188 aLine.append( "/FirstChar 32 /LastChar 255\n" 3189 "/Widths[" ); 3190 for( int i = 32; i < 256; i++ ) 3191 { 3192 aLine.append( pWidths[i] ); 3193 aLine.append( ((i&15) == 15) ? "\n" : " " ); 3194 } 3195 aLine.append( "]\n" 3196 "/FontDescriptor " ); 3197 aLine.append( nFontDescriptor ); 3198 aLine.append( " 0 R>>\n" 3199 "endobj\n\n" ); 3200 writeBuffer( aLine.getStr(), aLine.getLength() ); 3201 3202 nFontObject = nObject; 3203 aRet[ rEmbed.m_nNormalFontID ] = nObject; 3204 } 3205 } 3206 3207 return aRet; 3208 } 3209 3210 typedef int ThreeInts[3]; 3211 static bool getPfbSegmentLengths( const unsigned char* pFontBytes, int nByteLen, 3212 ThreeInts& rSegmentLengths ) 3213 { 3214 if( !pFontBytes || (nByteLen < 0) ) 3215 return false; 3216 const unsigned char* pPtr = pFontBytes; 3217 const unsigned char* pEnd = pFontBytes + nByteLen; 3218 3219 for( int i = 0; i < 3; ++i) { 3220 // read segment1 header 3221 if( pPtr+6 >= pEnd ) 3222 return false; 3223 if( (pPtr[0] != 0x80) || (pPtr[1] >= 0x03) ) 3224 return false; 3225 const int nLen = (pPtr[5]<<24) + (pPtr[4]<<16) + (pPtr[3]<<8) + pPtr[2]; 3226 if( nLen <= 0) 3227 return false; 3228 rSegmentLengths[i] = nLen; 3229 pPtr += nLen + 6; 3230 } 3231 3232 // read segment-end header 3233 if( pPtr+2 >= pEnd ) 3234 return false; 3235 if( (pPtr[0] != 0x80) || (pPtr[1] != 0x03) ) 3236 return false; 3237 3238 return true; 3239 } 3240 3241 struct FontException : public std::exception 3242 { 3243 }; 3244 3245 // TODO: always subset instead of embedding the full font => this method becomes obsolete then 3246 std::map< sal_Int32, sal_Int32 > PDFWriterImpl::emitEmbeddedFont( const ImplFontData* pFont, EmbedFont& rEmbed ) 3247 { 3248 std::map< sal_Int32, sal_Int32 > aRet; 3249 if( isBuiltinFont( pFont ) ) 3250 { 3251 aRet[ rEmbed.m_nNormalFontID ] = emitBuiltinFont( pFont ); 3252 return aRet; 3253 } 3254 3255 sal_Int32 nFontObject = 0; 3256 sal_Int32 nStreamObject = 0; 3257 sal_Int32 nFontDescriptor = 0; 3258 3259 // prepare font encoding 3260 const Ucs2SIntMap* pEncoding = m_pReferenceDevice->mpGraphics->GetFontEncodingVector( pFont, NULL ); 3261 sal_Int32 nToUnicodeStream = 0; 3262 sal_uInt8 nEncoding[256]; 3263 sal_Ucs nEncodedCodes[256]; 3264 std::vector<sal_Ucs> aUnicodes; 3265 aUnicodes.reserve( 256 ); 3266 sal_Int32 pUnicodesPerGlyph[256]; 3267 sal_Int32 pEncToUnicodeIndex[256]; 3268 if( pEncoding ) 3269 { 3270 rtl_zeroMemory( nEncoding, sizeof(nEncoding) ); 3271 rtl_zeroMemory( nEncodedCodes, sizeof(nEncodedCodes) ); 3272 rtl_zeroMemory( pUnicodesPerGlyph, sizeof(pUnicodesPerGlyph) ); 3273 rtl_zeroMemory( pEncToUnicodeIndex, sizeof(pEncToUnicodeIndex) ); 3274 for( Ucs2SIntMap::const_iterator it = pEncoding->begin(); it != pEncoding->end(); ++it ) 3275 { 3276 if( it->second != -1 ) 3277 { 3278 sal_Int32 nCode = (sal_Int32)(it->second & 0x000000ff); 3279 nEncoding[ nCode ] = static_cast<sal_uInt8>( nCode ); 3280 nEncodedCodes[ nCode ] = it->first; 3281 pEncToUnicodeIndex[ nCode ] = static_cast<sal_Int32>(aUnicodes.size()); 3282 aUnicodes.push_back( it->first ); 3283 pUnicodesPerGlyph[ nCode ] = 1; 3284 } 3285 } 3286 } 3287 3288 FontSubsetInfo aInfo; 3289 sal_Int32 pWidths[256]; 3290 const unsigned char* pFontData = NULL; 3291 long nFontLen = 0; 3292 sal_Int32 nLength1, nLength2; 3293 try 3294 { 3295 if( (pFontData = (const unsigned char*)m_pReferenceDevice->mpGraphics->GetEmbedFontData( pFont, nEncodedCodes, pWidths, aInfo, &nFontLen )) != NULL ) 3296 { 3297 if( (aInfo.m_nFontType & FontSubsetInfo::ANY_TYPE1) == 0 ) 3298 throw FontException(); 3299 // see whether it is pfb or pfa; if it is a pfb, fill ranges 3300 // of 6 bytes that are not part of the font program 3301 std::list< int > aSections; 3302 std::list< int >::const_iterator it; 3303 int nIndex = 0; 3304 while( pFontData[nIndex] == 0x80 && nIndex < nFontLen-1 ) 3305 { 3306 aSections.push_back( nIndex ); 3307 if( pFontData[nIndex+1] == 0x03 ) 3308 break; 3309 sal_Int32 nBytes = 3310 ((sal_Int32)pFontData[nIndex+2]) | 3311 ((sal_Int32)pFontData[nIndex+3]) << 8 | 3312 ((sal_Int32)pFontData[nIndex+4]) << 16 | 3313 ((sal_Int32)pFontData[nIndex+5]) << 24; 3314 nIndex += nBytes+6; 3315 } 3316 3317 // search for eexec 3318 // TODO: use getPfbSegmentLengths() if possible to skip the search thingies below 3319 nIndex = 0; 3320 int nEndAsciiIndex; 3321 int nBeginBinaryIndex; 3322 int nEndBinaryIndex; 3323 do 3324 { 3325 while( nIndex < nFontLen-4 && 3326 ( pFontData[nIndex] != 'e' || 3327 pFontData[nIndex+1] != 'e' || 3328 pFontData[nIndex+2] != 'x' || 3329 pFontData[nIndex+3] != 'e' || 3330 pFontData[nIndex+4] != 'c' 3331 ) 3332 ) 3333 nIndex++; 3334 // check whether we are in a excluded section 3335 for( it = aSections.begin(); it != aSections.end() && (nIndex < *it || nIndex > ((*it) + 5) ); ++it ) 3336 ; 3337 } while( it != aSections.end() && nIndex < nFontLen-4 ); 3338 // this should end the ascii part 3339 if( nIndex > nFontLen-5 ) 3340 throw FontException(); 3341 3342 nEndAsciiIndex = nIndex+4; 3343 // now count backwards until we can account for 512 '0' 3344 // which is the endmarker of the (hopefully) binary data 3345 // do not count the pfb header sections 3346 int nFound = 0; 3347 nIndex = nFontLen-1; 3348 while( nIndex > 0 && nFound < 512 ) 3349 { 3350 for( it = aSections.begin(); it != aSections.end() && (nIndex < *it || nIndex > ((*it) + 5) ); ++it ) 3351 ; 3352 if( it == aSections.end() ) 3353 { 3354 // inside the 512 '0' block there may only be whitespace 3355 // according to T1 spec; probably it would be to simple 3356 // if all fonts complied 3357 if( pFontData[nIndex] == '0' ) 3358 nFound++; 3359 else if( nFound > 0 && 3360 pFontData[nIndex] != '\r' && 3361 pFontData[nIndex] != '\t' && 3362 pFontData[nIndex] != '\n' && 3363 pFontData[nIndex] != ' ' ) 3364 break; 3365 } 3366 nIndex--; 3367 } 3368 3369 if( nIndex < 1 || nIndex <= nEndAsciiIndex ) 3370 throw FontException(); 3371 3372 // nLength3 is the rest of the file - excluding any section headers 3373 // nIndex now points to the first of the 512 '0' characters marking the 3374 // fixed content portion 3375 sal_Int32 nLength3 = nFontLen - nIndex; 3376 for( it = aSections.begin(); it != aSections.end(); ++it ) 3377 { 3378 if( *it >= nIndex ) 3379 { 3380 // special case: nIndex inside a section marker 3381 if( nIndex >= (*it) && (*it)+5 > nIndex ) 3382 nLength3 -= (*it)+5 - nIndex; 3383 else 3384 { 3385 if( *it < nFontLen - 6 ) 3386 nLength3 -= 6; 3387 else // the last section 0x8003 is only 2 bytes after all 3388 nLength3 -= (nFontLen - *it); 3389 } 3390 } 3391 } 3392 3393 // there may be whitespace to ignore before the 512 '0' 3394 while( pFontData[nIndex] == '\r' || pFontData[nIndex] == '\n' ) 3395 { 3396 nIndex--; 3397 for( it = aSections.begin(); it != aSections.end() && (nIndex < *it || nIndex > ((*it) + 5) ); ++it ) 3398 ; 3399 if( it != aSections.end() ) 3400 { 3401 nIndex = (*it)-1; 3402 break; // this is surely a binary boundary, in ascii case it wouldn't matter 3403 } 3404 } 3405 nEndBinaryIndex = nIndex; 3406 3407 // search for beginning of binary section 3408 nBeginBinaryIndex = nEndAsciiIndex; 3409 do 3410 { 3411 nBeginBinaryIndex++; 3412 for( it = aSections.begin(); it != aSections.end() && (nBeginBinaryIndex < *it || nBeginBinaryIndex > ((*it) + 5) ); ++it ) 3413 ; 3414 } while( nBeginBinaryIndex < nEndBinaryIndex && 3415 ( pFontData[nBeginBinaryIndex] == '\r' || 3416 pFontData[nBeginBinaryIndex] == '\n' || 3417 it != aSections.end() ) ); 3418 3419 // it seems to be vital to copy the exact whitespace between binary data 3420 // and eexec, else a invalid font results. so make nEndAsciiIndex 3421 // always immediate in front of nBeginBinaryIndex 3422 nEndAsciiIndex = nBeginBinaryIndex-1; 3423 for( it = aSections.begin(); it != aSections.end() && (nEndAsciiIndex < *it || nEndAsciiIndex > ((*it)+5)); ++it ) 3424 ; 3425 if( it != aSections.end() ) 3426 nEndAsciiIndex = (*it)-1; 3427 3428 nLength1 = nEndAsciiIndex+1; // including the last character 3429 for( it = aSections.begin(); it != aSections.end() && *it < nEndAsciiIndex; ++it ) 3430 nLength1 -= 6; // decrease by pfb section size 3431 3432 // if the first four bytes are all ascii hex characters, then binary data 3433 // has to be converted to real binary data 3434 for( nIndex = 0; nIndex < 4 && 3435 ( ( pFontData[ nBeginBinaryIndex+nIndex ] >= '0' && pFontData[ nBeginBinaryIndex+nIndex ] <= '9' ) || 3436 ( pFontData[ nBeginBinaryIndex+nIndex ] >= 'a' && pFontData[ nBeginBinaryIndex+nIndex ] <= 'f' ) || 3437 ( pFontData[ nBeginBinaryIndex+nIndex ] >= 'A' && pFontData[ nBeginBinaryIndex+nIndex ] <= 'F' ) 3438 ); ++nIndex ) 3439 ; 3440 bool bConvertHexData = true; 3441 if( nIndex < 4 ) 3442 { 3443 bConvertHexData = false; 3444 nLength2 = nEndBinaryIndex - nBeginBinaryIndex + 1; // include the last byte 3445 for( it = aSections.begin(); it != aSections.end(); ++it ) 3446 if( *it > nBeginBinaryIndex && *it < nEndBinaryIndex ) 3447 nLength2 -= 6; 3448 } 3449 else 3450 { 3451 // count the hex ascii characters to get nLength2 3452 nLength2 = 0; 3453 int nNextSectionIndex = 0; 3454 for( it = aSections.begin(); it != aSections.end() && *it < nBeginBinaryIndex; ++it ) 3455 ; 3456 if( it != aSections.end() ) 3457 nNextSectionIndex = *it; 3458 for( nIndex = nBeginBinaryIndex; nIndex <= nEndBinaryIndex; nIndex++ ) 3459 { 3460 if( nIndex == nNextSectionIndex ) 3461 { 3462 nIndex += 6; 3463 ++it; 3464 nNextSectionIndex = (it == aSections.end() ? 0 : *it ); 3465 } 3466 if( ( pFontData[ nIndex ] >= '0' && pFontData[ nIndex ] <= '9' ) || 3467 ( pFontData[ nIndex ] >= 'a' && pFontData[ nIndex ] <= 'f' ) || 3468 ( pFontData[ nIndex ] >= 'A' && pFontData[ nIndex ] <= 'F' ) ) 3469 nLength2++; 3470 } 3471 DBG_ASSERT( !(nLength2 & 1), "uneven number of hex chars in binary pfa section" ); 3472 nLength2 /= 2; 3473 } 3474 3475 // now we can actually write the font stream ! 3476 #if OSL_DEBUG_LEVEL > 1 3477 emitComment( " PDFWriterImpl::emitEmbeddedFont" ); 3478 #endif 3479 OStringBuffer aLine( 512 ); 3480 nStreamObject = createObject(); 3481 if( !updateObject(nStreamObject)) 3482 throw FontException(); 3483 sal_Int32 nStreamLengthObject = createObject(); 3484 aLine.append( nStreamObject ); 3485 aLine.append( " 0 obj\n" 3486 "<</Length " ); 3487 aLine.append( nStreamLengthObject ); 3488 aLine.append( " 0 R" 3489 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 3490 "/Filter/FlateDecode" 3491 #endif 3492 "/Length1 " ); 3493 aLine.append( nLength1 ); 3494 aLine.append( " /Length2 " ); 3495 aLine.append( nLength2 ); 3496 aLine.append( " /Length3 "); 3497 aLine.append( nLength3 ); 3498 aLine.append( ">>\n" 3499 "stream\n" ); 3500 if( !writeBuffer( aLine.getStr(), aLine.getLength() ) ) 3501 throw FontException(); 3502 3503 sal_uInt64 nBeginStreamPos = 0; 3504 osl_getFilePos( m_aFile, &nBeginStreamPos ); 3505 3506 beginCompression(); 3507 checkAndEnableStreamEncryption( nStreamObject ); 3508 3509 // write ascii section 3510 if( aSections.begin() == aSections.end() ) 3511 { 3512 if( ! writeBuffer( pFontData, nEndAsciiIndex+1 ) ) 3513 throw FontException(); 3514 } 3515 else 3516 { 3517 // first section always starts at 0 3518 it = aSections.begin(); 3519 nIndex = (*it)+6; 3520 ++it; 3521 while( *it < nEndAsciiIndex ) 3522 { 3523 if( ! writeBuffer( pFontData+nIndex, (*it)-nIndex ) ) 3524 throw FontException(); 3525 nIndex = (*it)+6; 3526 ++it; 3527 } 3528 // write partial last section 3529 if( ! writeBuffer( pFontData+nIndex, nEndAsciiIndex-nIndex+1 ) ) 3530 throw FontException(); 3531 } 3532 3533 // write binary section 3534 if( ! bConvertHexData ) 3535 { 3536 if( aSections.begin() == aSections.end() ) 3537 { 3538 if( ! writeBuffer( pFontData+nBeginBinaryIndex, nFontLen-nBeginBinaryIndex ) ) 3539 throw FontException(); 3540 } 3541 else 3542 { 3543 for( it = aSections.begin(); *it < nBeginBinaryIndex; ++it ) 3544 ; 3545 // write first partial section 3546 if( ! writeBuffer( pFontData+nBeginBinaryIndex, (*it) - nBeginBinaryIndex ) ) 3547 throw FontException(); 3548 // write following sections 3549 while( it != aSections.end() ) 3550 { 3551 nIndex = (*it)+6; 3552 ++it; 3553 if( nIndex < nFontLen ) // last section marker is usually the EOF which has only 2 bytes 3554 { 3555 sal_Int32 nSectionLen = (it == aSections.end()) ? nFontLen - nIndex : (*it) - nIndex; 3556 if( ! writeBuffer( pFontData+nIndex, nSectionLen ) ) 3557 throw FontException(); 3558 } 3559 } 3560 } 3561 } 3562 else 3563 { 3564 boost::shared_array<unsigned char> pWriteBuffer( new unsigned char[ nLength2 ] ); 3565 rtl_zeroMemory( pWriteBuffer.get(), nLength2 ); 3566 int nWriteIndex = 0; 3567 3568 int nNextSectionIndex = 0; 3569 for( it = aSections.begin(); it != aSections.end() && *it < nBeginBinaryIndex; ++it ) 3570 ; 3571 if( it != aSections.end() ) 3572 nNextSectionIndex = *it; 3573 for( nIndex = nBeginBinaryIndex; nIndex <= nEndBinaryIndex; nIndex++ ) 3574 { 3575 if( nIndex == nNextSectionIndex ) 3576 { 3577 nIndex += 6; 3578 ++it; 3579 nNextSectionIndex = (it == aSections.end() ? nFontLen : *it ); 3580 } 3581 unsigned char cNibble = 0x80; 3582 if( pFontData[ nIndex ] >= '0' && pFontData[ nIndex ] <= '9' ) 3583 cNibble = pFontData[nIndex] - '0'; 3584 else if( pFontData[ nIndex ] >= 'a' && pFontData[ nIndex ] <= 'f' ) 3585 cNibble = pFontData[nIndex] - 'a' + 10; 3586 else if( pFontData[ nIndex ] >= 'A' && pFontData[ nIndex ] <= 'F' ) 3587 cNibble = pFontData[nIndex] - 'A' + 10; 3588 if( cNibble != 0x80 ) 3589 { 3590 if( !(nWriteIndex & 1 ) ) 3591 cNibble <<= 4; 3592 pWriteBuffer.get()[ nWriteIndex/2 ] |= cNibble; 3593 nWriteIndex++; 3594 } 3595 } 3596 if( ! writeBuffer( pWriteBuffer.get(), nLength2 ) ) 3597 throw FontException(); 3598 if( aSections.empty() ) 3599 { 3600 if( ! writeBuffer( pFontData+nIndex, nFontLen-nIndex ) ) 3601 throw FontException(); 3602 } 3603 else 3604 { 3605 // write rest of this section 3606 if( nIndex < nNextSectionIndex ) 3607 { 3608 if( ! writeBuffer( pFontData+nIndex, nNextSectionIndex - nIndex ) ) 3609 throw FontException(); 3610 } 3611 // write following sections 3612 while( it != aSections.end() ) 3613 { 3614 nIndex = (*it)+6; 3615 ++it; 3616 if( nIndex < nFontLen ) // last section marker is usually the EOF which has only 2 bytes 3617 { 3618 sal_Int32 nSectionLen = (it == aSections.end()) ? nFontLen - nIndex : (*it) - nIndex; 3619 if( ! writeBuffer( pFontData+nIndex, nSectionLen ) ) 3620 throw FontException(); 3621 } 3622 } 3623 } 3624 } 3625 endCompression(); 3626 disableStreamEncryption(); 3627 3628 3629 sal_uInt64 nEndStreamPos = 0; 3630 osl_getFilePos( m_aFile, &nEndStreamPos ); 3631 3632 // and finally close the stream 3633 aLine.setLength( 0 ); 3634 aLine.append( "\nendstream\nendobj\n\n" ); 3635 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 3636 throw FontException(); 3637 3638 // write stream length object 3639 aLine.setLength( 0 ); 3640 if( ! updateObject( nStreamLengthObject ) ) 3641 throw FontException(); 3642 aLine.append( nStreamLengthObject ); 3643 aLine.append( " 0 obj\n" ); 3644 aLine.append( (sal_Int64)(nEndStreamPos-nBeginStreamPos ) ); 3645 aLine.append( "\nendobj\n\n" ); 3646 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 3647 throw FontException(); 3648 } 3649 else 3650 { 3651 rtl::OStringBuffer aErrorComment( 256 ); 3652 aErrorComment.append( "GetEmbedFontData failed for font \"" ); 3653 aErrorComment.append( OUStringToOString( pFont->GetFamilyName(), RTL_TEXTENCODING_UTF8 ) ); 3654 aErrorComment.append( '\"' ); 3655 if( pFont->GetSlant() == ITALIC_NORMAL ) 3656 aErrorComment.append( " italic" ); 3657 else if( pFont->GetSlant() == ITALIC_OBLIQUE ) 3658 aErrorComment.append( " oblique" ); 3659 aErrorComment.append( " weight=" ); 3660 aErrorComment.append( sal_Int32(pFont->GetWeight()) ); 3661 emitComment( aErrorComment.getStr() ); 3662 } 3663 3664 if( nStreamObject ) 3665 // write font descriptor 3666 nFontDescriptor = emitFontDescriptor( pFont, aInfo, 0, nStreamObject ); 3667 3668 if( nFontDescriptor ) 3669 { 3670 if( pEncoding ) 3671 nToUnicodeStream = createToUnicodeCMap( nEncoding, &aUnicodes[0], pUnicodesPerGlyph, pEncToUnicodeIndex, sizeof(nEncoding)/sizeof(nEncoding[0]) ); 3672 3673 // write font object 3674 sal_Int32 nObject = createObject(); 3675 if( ! updateObject( nObject ) ) 3676 throw FontException(); 3677 3678 OStringBuffer aLine( 1024 ); 3679 aLine.append( nObject ); 3680 aLine.append( " 0 obj\n" 3681 "<</Type/Font/Subtype/Type1/BaseFont/" ); 3682 appendName( aInfo.m_aPSName, aLine ); 3683 aLine.append( "\n" ); 3684 if( !pFont->mbSymbolFlag && pEncoding == 0 ) 3685 aLine.append( "/Encoding/WinAnsiEncoding\n" ); 3686 if( nToUnicodeStream ) 3687 { 3688 aLine.append( "/ToUnicode " ); 3689 aLine.append( nToUnicodeStream ); 3690 aLine.append( " 0 R\n" ); 3691 } 3692 aLine.append( "/FirstChar 0 /LastChar 255\n" 3693 "/Widths[" ); 3694 for( int i = 0; i < 256; i++ ) 3695 { 3696 aLine.append( pWidths[i] ); 3697 aLine.append( ((i&15) == 15) ? "\n" : " " ); 3698 } 3699 aLine.append( "]\n" 3700 "/FontDescriptor " ); 3701 aLine.append( nFontDescriptor ); 3702 aLine.append( " 0 R>>\n" 3703 "endobj\n\n" ); 3704 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 3705 throw FontException(); 3706 3707 nFontObject = nObject; 3708 3709 aRet[ rEmbed.m_nNormalFontID ] = nObject; 3710 3711 // write additional encodings 3712 for( std::list< EmbedEncoding >::iterator enc_it = rEmbed.m_aExtendedEncodings.begin(); enc_it != rEmbed.m_aExtendedEncodings.end(); ++enc_it ) 3713 { 3714 sal_Int32 aEncWidths[ 256 ]; 3715 // emit encoding dict 3716 sal_Int32 nEncObject = createObject(); 3717 if( ! updateObject( nEncObject ) ) 3718 throw FontException(); 3719 3720 OutputDevice* pRef = getReferenceDevice(); 3721 pRef->Push( PUSH_FONT | PUSH_MAPMODE ); 3722 pRef->SetMapMode( MapMode( MAP_PIXEL ) ); 3723 Font aFont( pFont->GetFamilyName(), pFont->GetStyleName(), Size( 0, 1000 ) ); 3724 aFont.SetWeight( pFont->GetWeight() ); 3725 aFont.SetItalic( pFont->GetSlant() ); 3726 aFont.SetPitch( pFont->GetPitch() ); 3727 pRef->SetFont( aFont ); 3728 pRef->ImplNewFont(); 3729 3730 aLine.setLength( 0 ); 3731 aLine.append( nEncObject ); 3732 aLine.append( " 0 obj\n" 3733 "<</Type/Encoding/Differences[ 0\n" ); 3734 int nEncoded = 0; 3735 aUnicodes.clear(); 3736 for( std::vector< EmbedCode >::iterator str_it = enc_it->m_aEncVector.begin(); str_it != enc_it->m_aEncVector.end(); ++str_it ) 3737 { 3738 String aStr( str_it->m_aUnicode ); 3739 aEncWidths[nEncoded] = pRef->GetTextWidth( aStr ); 3740 nEncodedCodes[nEncoded] = str_it->m_aUnicode; 3741 nEncoding[nEncoded] = sal::static_int_cast<sal_uInt8>(nEncoded); 3742 pEncToUnicodeIndex[nEncoded] = static_cast<sal_Int32>(aUnicodes.size()); 3743 aUnicodes.push_back( nEncodedCodes[nEncoded] ); 3744 pUnicodesPerGlyph[nEncoded] = 1; 3745 3746 aLine.append( " /" ); 3747 aLine.append( str_it->m_aName ); 3748 if( !((++nEncoded) & 15) ) 3749 aLine.append( "\n" ); 3750 } 3751 aLine.append( "]>>\n" 3752 "endobj\n\n" ); 3753 3754 pRef->Pop(); 3755 3756 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 3757 throw FontException(); 3758 3759 nToUnicodeStream = createToUnicodeCMap( nEncoding, &aUnicodes[0], pUnicodesPerGlyph, pEncToUnicodeIndex, nEncoded ); 3760 3761 nObject = createObject(); 3762 if( ! updateObject( nObject ) ) 3763 throw FontException(); 3764 3765 aLine.setLength( 0 ); 3766 aLine.append( nObject ); 3767 aLine.append( " 0 obj\n" 3768 "<</Type/Font/Subtype/Type1/BaseFont/" ); 3769 appendName( aInfo.m_aPSName, aLine ); 3770 aLine.append( "\n" ); 3771 aLine.append( "/Encoding " ); 3772 aLine.append( nEncObject ); 3773 aLine.append( " 0 R\n" ); 3774 if( nToUnicodeStream ) 3775 { 3776 aLine.append( "/ToUnicode " ); 3777 aLine.append( nToUnicodeStream ); 3778 aLine.append( " 0 R\n" ); 3779 } 3780 aLine.append( "/FirstChar 0\n" 3781 "/LastChar " ); 3782 aLine.append( (sal_Int32)(nEncoded-1) ); 3783 aLine.append( "\n" 3784 "/Widths[" ); 3785 for( int i = 0; i < nEncoded; i++ ) 3786 { 3787 aLine.append( aEncWidths[i] ); 3788 aLine.append( ((i&15) == 15) ? "\n" : " " ); 3789 } 3790 aLine.append( " ]\n" 3791 "/FontDescriptor " ); 3792 aLine.append( nFontDescriptor ); 3793 aLine.append( " 0 R>>\n" 3794 "endobj\n\n" ); 3795 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 3796 throw FontException(); 3797 3798 aRet[ enc_it->m_nFontID ] = nObject; 3799 } 3800 } 3801 } 3802 catch( FontException& ) 3803 { 3804 // these do nothing in case there was no compression or encryption ongoing 3805 endCompression(); 3806 disableStreamEncryption(); 3807 } 3808 3809 if( pFontData ) 3810 m_pReferenceDevice->mpGraphics->FreeEmbedFontData( pFontData, nFontLen ); 3811 3812 return aRet; 3813 } 3814 3815 static void appendSubsetName( int nSubsetID, const OUString& rPSName, OStringBuffer& rBuffer ) 3816 { 3817 if( nSubsetID ) 3818 { 3819 for( int i = 0; i < 6; i++ ) 3820 { 3821 int nOffset = (nSubsetID % 26); 3822 nSubsetID /= 26; 3823 rBuffer.append( (sal_Char)('A'+nOffset) ); 3824 } 3825 rBuffer.append( '+' ); 3826 } 3827 appendName( rPSName, rBuffer ); 3828 } 3829 3830 sal_Int32 PDFWriterImpl::createToUnicodeCMap( sal_uInt8* pEncoding, 3831 sal_Ucs* pUnicodes, 3832 sal_Int32* pUnicodesPerGlyph, 3833 sal_Int32* pEncToUnicodeIndex, 3834 int nGlyphs ) 3835 { 3836 int nMapped = 0, n = 0; 3837 for( n = 0; n < nGlyphs; n++ ) 3838 if( pUnicodes[pEncToUnicodeIndex[n]] && pUnicodesPerGlyph[n] ) 3839 nMapped++; 3840 3841 if( nMapped == 0 ) 3842 return 0; 3843 3844 sal_Int32 nStream = createObject(); 3845 CHECK_RETURN( updateObject( nStream ) ); 3846 3847 OStringBuffer aContents( 1024 ); 3848 aContents.append( 3849 "/CIDInit/ProcSet findresource begin\n" 3850 "12 dict begin\n" 3851 "begincmap\n" 3852 "/CIDSystemInfo<<\n" 3853 "/Registry (Adobe)\n" 3854 "/Ordering (UCS)\n" 3855 "/Supplement 0\n" 3856 ">> def\n" 3857 "/CMapName/Adobe-Identity-UCS def\n" 3858 "/CMapType 2 def\n" 3859 "1 begincodespacerange\n" 3860 "<00> <FF>\n" 3861 "endcodespacerange\n" 3862 ); 3863 int nCount = 0; 3864 for( n = 0; n < nGlyphs; n++ ) 3865 { 3866 if( pUnicodes[pEncToUnicodeIndex[n]] && pUnicodesPerGlyph[n] ) 3867 { 3868 if( (nCount % 100) == 0 ) 3869 { 3870 if( nCount ) 3871 aContents.append( "endbfchar\n" ); 3872 aContents.append( (sal_Int32)((nMapped-nCount > 100) ? 100 : nMapped-nCount ) ); 3873 aContents.append( " beginbfchar\n" ); 3874 } 3875 aContents.append( '<' ); 3876 appendHex( (sal_Int8)pEncoding[n], aContents ); 3877 aContents.append( "> <" ); 3878 // TODO: handle unicodes>U+FFFF 3879 sal_Int32 nIndex = pEncToUnicodeIndex[n]; 3880 for( sal_Int32 j = 0; j < pUnicodesPerGlyph[n]; j++ ) 3881 { 3882 appendHex( (sal_Int8)(pUnicodes[nIndex + j] / 256), aContents ); 3883 appendHex( (sal_Int8)(pUnicodes[nIndex + j] & 255), aContents ); 3884 } 3885 aContents.append( ">\n" ); 3886 nCount++; 3887 } 3888 } 3889 aContents.append( "endbfchar\n" 3890 "endcmap\n" 3891 "CMapName currentdict /CMap defineresource pop\n" 3892 "end\n" 3893 "end\n" ); 3894 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 3895 ZCodec* pCodec = new ZCodec( 0x4000, 0x4000 ); 3896 SvMemoryStream aStream; 3897 pCodec->BeginCompression(); 3898 pCodec->Write( aStream, (const sal_uInt8*)aContents.getStr(), aContents.getLength() ); 3899 pCodec->EndCompression(); 3900 delete pCodec; 3901 #endif 3902 3903 #if OSL_DEBUG_LEVEL > 1 3904 emitComment( "PDFWriterImpl::createToUnicodeCMap" ); 3905 #endif 3906 OStringBuffer aLine( 40 ); 3907 3908 aLine.append( nStream ); 3909 aLine.append( " 0 obj\n<</Length " ); 3910 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 3911 sal_Int32 nLen = (sal_Int32)aStream.Tell(); 3912 aStream.Seek( 0 ); 3913 aLine.append( nLen ); 3914 aLine.append( "/Filter/FlateDecode" ); 3915 #else 3916 aLine.append( aContents.getLength() ); 3917 #endif 3918 aLine.append( ">>\nstream\n" ); 3919 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 3920 checkAndEnableStreamEncryption( nStream ); 3921 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 3922 CHECK_RETURN( writeBuffer( aStream.GetData(), nLen ) ); 3923 #else 3924 CHECK_RETURN( writeBuffer( aContents.getStr(), aContents.getLength() ) ); 3925 #endif 3926 disableStreamEncryption(); 3927 aLine.setLength( 0 ); 3928 aLine.append( "\nendstream\n" 3929 "endobj\n\n" ); 3930 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 3931 return nStream; 3932 } 3933 3934 sal_Int32 PDFWriterImpl::emitFontDescriptor( const ImplFontData* pFont, FontSubsetInfo& rInfo, sal_Int32 nSubsetID, sal_Int32 nFontStream ) 3935 { 3936 OStringBuffer aLine( 1024 ); 3937 // get font flags, see PDF reference 1.4 p. 358 3938 // possibly characters outside Adobe standard encoding 3939 // so set Symbolic flag 3940 sal_Int32 nFontFlags = (1<<2); 3941 if( pFont->GetSlant() == ITALIC_NORMAL || pFont->GetSlant() == ITALIC_OBLIQUE ) 3942 nFontFlags |= (1 << 6); 3943 if( pFont->GetPitch() == PITCH_FIXED ) 3944 nFontFlags |= 1; 3945 if( pFont->GetFamilyType() == FAMILY_SCRIPT ) 3946 nFontFlags |= (1 << 3); 3947 else if( pFont->GetFamilyType() == FAMILY_ROMAN ) 3948 nFontFlags |= (1 << 1); 3949 3950 sal_Int32 nFontDescriptor = createObject(); 3951 CHECK_RETURN( updateObject( nFontDescriptor ) ); 3952 aLine.setLength( 0 ); 3953 aLine.append( nFontDescriptor ); 3954 aLine.append( " 0 obj\n" 3955 "<</Type/FontDescriptor/FontName/" ); 3956 appendSubsetName( nSubsetID, rInfo.m_aPSName, aLine ); 3957 aLine.append( "\n" 3958 "/Flags " ); 3959 aLine.append( nFontFlags ); 3960 aLine.append( "\n" 3961 "/FontBBox[" ); 3962 // note: Top and Bottom are reversed in VCL and PDF rectangles 3963 aLine.append( (sal_Int32)rInfo.m_aFontBBox.TopLeft().X() ); 3964 aLine.append( ' ' ); 3965 aLine.append( (sal_Int32)rInfo.m_aFontBBox.TopLeft().Y() ); 3966 aLine.append( ' ' ); 3967 aLine.append( (sal_Int32)rInfo.m_aFontBBox.BottomRight().X() ); 3968 aLine.append( ' ' ); 3969 aLine.append( (sal_Int32)(rInfo.m_aFontBBox.BottomRight().Y()+1) ); 3970 aLine.append( "]/ItalicAngle " ); 3971 if( pFont->GetSlant() == ITALIC_OBLIQUE || pFont->GetSlant() == ITALIC_NORMAL ) 3972 aLine.append( "-30" ); 3973 else 3974 aLine.append( "0" ); 3975 aLine.append( "\n" 3976 "/Ascent " ); 3977 aLine.append( (sal_Int32)rInfo.m_nAscent ); 3978 aLine.append( "\n" 3979 "/Descent " ); 3980 aLine.append( (sal_Int32)-rInfo.m_nDescent ); 3981 aLine.append( "\n" 3982 "/CapHeight " ); 3983 aLine.append( (sal_Int32)rInfo.m_nCapHeight ); 3984 // According to PDF reference 1.4 StemV is required 3985 // seems a tad strange to me, but well ... 3986 aLine.append( "\n" 3987 "/StemV 80\n" ); 3988 if( nFontStream ) 3989 { 3990 aLine.append( "/FontFile" ); 3991 switch( rInfo.m_nFontType ) 3992 { 3993 case FontSubsetInfo::SFNT_TTF: 3994 aLine.append( '2' ); 3995 break; 3996 case FontSubsetInfo::TYPE1_PFA: 3997 case FontSubsetInfo::TYPE1_PFB: 3998 case FontSubsetInfo::ANY_TYPE1: 3999 break; 4000 default: 4001 DBG_ERROR( "unknown fonttype in PDF font descriptor" ); 4002 return 0; 4003 } 4004 aLine.append( ' ' ); 4005 aLine.append( nFontStream ); 4006 aLine.append( " 0 R\n" ); 4007 } 4008 aLine.append( ">>\n" 4009 "endobj\n\n" ); 4010 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4011 4012 return nFontDescriptor; 4013 } 4014 4015 void PDFWriterImpl::appendBuiltinFontsToDict( OStringBuffer& rDict ) const 4016 { 4017 for( std::map< sal_Int32, sal_Int32 >::const_iterator it = 4018 m_aBuiltinFontToObjectMap.begin(); it != m_aBuiltinFontToObjectMap.end(); ++it ) 4019 { 4020 rDict.append( m_aBuiltinFonts[it->first].getNameObject() ); 4021 rDict.append( ' ' ); 4022 rDict.append( it->second ); 4023 rDict.append( " 0 R" ); 4024 } 4025 } 4026 4027 bool PDFWriterImpl::emitFonts() 4028 { 4029 if( ! m_pReferenceDevice->ImplGetGraphics() ) 4030 return false; 4031 4032 OStringBuffer aLine( 1024 ); 4033 4034 std::map< sal_Int32, sal_Int32 > aFontIDToObject; 4035 4036 OUString aTmpName; 4037 osl_createTempFile( NULL, NULL, &aTmpName.pData ); 4038 for( FontSubsetData::iterator it = m_aSubsets.begin(); it != m_aSubsets.end(); ++it ) 4039 { 4040 for( FontEmitList::iterator lit = it->second.m_aSubsets.begin(); lit != it->second.m_aSubsets.end(); ++lit ) 4041 { 4042 sal_Int32 pGlyphIDs[ 256 ]; 4043 sal_Int32 pWidths[ 256 ]; 4044 sal_uInt8 pEncoding[ 256 ]; 4045 sal_Int32 pEncToUnicodeIndex[ 256 ]; 4046 sal_Int32 pUnicodesPerGlyph[ 256 ]; 4047 std::vector<sal_Ucs> aUnicodes; 4048 aUnicodes.reserve( 256 ); 4049 int nGlyphs = 1; 4050 // fill arrays and prepare encoding index map 4051 sal_Int32 nToUnicodeStream = 0; 4052 4053 rtl_zeroMemory( pGlyphIDs, sizeof( pGlyphIDs ) ); 4054 rtl_zeroMemory( pEncoding, sizeof( pEncoding ) ); 4055 rtl_zeroMemory( pUnicodesPerGlyph, sizeof( pUnicodesPerGlyph ) ); 4056 rtl_zeroMemory( pEncToUnicodeIndex, sizeof( pEncToUnicodeIndex ) ); 4057 for( FontEmitMapping::iterator fit = lit->m_aMapping.begin(); fit != lit->m_aMapping.end();++fit ) 4058 { 4059 sal_uInt8 nEnc = fit->second.getGlyphId(); 4060 4061 DBG_ASSERT( pGlyphIDs[nEnc] == 0 && pEncoding[nEnc] == 0, "duplicate glyph" ); 4062 DBG_ASSERT( nEnc <= lit->m_aMapping.size(), "invalid glyph encoding" ); 4063 4064 pGlyphIDs[ nEnc ] = fit->first; 4065 pEncoding[ nEnc ] = nEnc; 4066 pEncToUnicodeIndex[ nEnc ] = static_cast<sal_Int32>(aUnicodes.size()); 4067 pUnicodesPerGlyph[ nEnc ] = fit->second.countCodes(); 4068 for( sal_Int32 n = 0; n < pUnicodesPerGlyph[ nEnc ]; n++ ) 4069 aUnicodes.push_back( fit->second.getCode( n ) ); 4070 if( fit->second.getCode(0) ) 4071 nToUnicodeStream = 1; 4072 if( nGlyphs < 256 ) 4073 nGlyphs++; 4074 else 4075 { 4076 DBG_ERROR( "too many glyphs for subset" ); 4077 } 4078 } 4079 FontSubsetInfo aSubsetInfo; 4080 if( m_pReferenceDevice->mpGraphics->CreateFontSubset( aTmpName, it->first, pGlyphIDs, pEncoding, pWidths, nGlyphs, aSubsetInfo ) ) 4081 { 4082 // create font stream 4083 oslFileHandle aFontFile; 4084 CHECK_RETURN( (osl_File_E_None == osl_openFile( aTmpName.pData, &aFontFile, osl_File_OpenFlag_Read ) ) ); 4085 // get file size 4086 sal_uInt64 nLength1; 4087 CHECK_RETURN( (osl_File_E_None == osl_setFilePos( aFontFile, osl_Pos_End, 0 ) ) ); 4088 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( aFontFile, &nLength1 ) ) ); 4089 CHECK_RETURN( (osl_File_E_None == osl_setFilePos( aFontFile, osl_Pos_Absolut, 0 ) ) ); 4090 4091 #if OSL_DEBUG_LEVEL > 1 4092 emitComment( "PDFWriterImpl::emitFonts" ); 4093 #endif 4094 sal_Int32 nFontStream = createObject(); 4095 sal_Int32 nStreamLengthObject = createObject(); 4096 CHECK_RETURN( updateObject( nFontStream ) ); 4097 aLine.setLength( 0 ); 4098 aLine.append( nFontStream ); 4099 aLine.append( " 0 obj\n" 4100 "<</Length " ); 4101 aLine.append( (sal_Int32)nStreamLengthObject ); 4102 aLine.append( " 0 R" 4103 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 4104 "/Filter/FlateDecode" 4105 #endif 4106 "/Length1 " ); 4107 4108 sal_uInt64 nStartPos = 0; 4109 if( aSubsetInfo.m_nFontType == FontSubsetInfo::SFNT_TTF ) 4110 { 4111 aLine.append( (sal_Int32)nLength1 ); 4112 4113 aLine.append( ">>\n" 4114 "stream\n" ); 4115 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4116 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nStartPos ) ) ); 4117 4118 // copy font file 4119 beginCompression(); 4120 checkAndEnableStreamEncryption( nFontStream ); 4121 sal_Bool bEOF = sal_False; 4122 do 4123 { 4124 char buf[8192]; 4125 sal_uInt64 nRead; 4126 CHECK_RETURN( (osl_File_E_None == osl_readFile( aFontFile, buf, sizeof( buf ), &nRead ) ) ); 4127 CHECK_RETURN( writeBuffer( buf, nRead ) ); 4128 CHECK_RETURN( (osl_File_E_None == osl_isEndOfFile( aFontFile, &bEOF ) ) ); 4129 } while( ! bEOF ); 4130 } 4131 else if( (aSubsetInfo.m_nFontType & FontSubsetInfo::CFF_FONT) != 0 ) 4132 { 4133 // TODO: implement 4134 DBG_ERROR( "PDFWriterImpl does not support CFF-font subsets yet!" ); 4135 } 4136 else if( (aSubsetInfo.m_nFontType & FontSubsetInfo::TYPE1_PFB) != 0 ) // TODO: also support PFA? 4137 { 4138 boost::shared_array<unsigned char> pBuffer( new unsigned char[ nLength1 ] ); 4139 4140 sal_uInt64 nBytesRead = 0; 4141 CHECK_RETURN( (osl_File_E_None == osl_readFile( aFontFile, pBuffer.get(), nLength1, &nBytesRead ) ) ); 4142 DBG_ASSERT( nBytesRead==nLength1, "PDF-FontSubset read incomplete!" ); 4143 CHECK_RETURN( (osl_File_E_None == osl_setFilePos( aFontFile, osl_Pos_Absolut, 0 ) ) ); 4144 // get the PFB-segment lengths 4145 ThreeInts aSegmentLengths = {0,0,0}; 4146 getPfbSegmentLengths( pBuffer.get(), (int)nBytesRead, aSegmentLengths ); 4147 // the lengths below are mandatory for PDF-exported Type1 fonts 4148 // because the PFB segment headers get stripped! WhyOhWhy. 4149 aLine.append( (sal_Int32)aSegmentLengths[0] ); 4150 aLine.append( "/Length2 " ); 4151 aLine.append( (sal_Int32)aSegmentLengths[1] ); 4152 aLine.append( "/Length3 " ); 4153 aLine.append( (sal_Int32)aSegmentLengths[2] ); 4154 4155 aLine.append( ">>\n" 4156 "stream\n" ); 4157 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4158 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nStartPos ) ) ); 4159 4160 // emit PFB-sections without section headers 4161 beginCompression(); 4162 checkAndEnableStreamEncryption( nFontStream ); 4163 CHECK_RETURN( writeBuffer( &pBuffer[6], aSegmentLengths[0] ) ); 4164 CHECK_RETURN( writeBuffer( &pBuffer[12] + aSegmentLengths[0], aSegmentLengths[1] ) ); 4165 CHECK_RETURN( writeBuffer( &pBuffer[18] + aSegmentLengths[0] + aSegmentLengths[1], aSegmentLengths[2] ) ); 4166 } 4167 else 4168 { 4169 fprintf( stderr, "PDF: CreateFontSubset result in not yet supported format=%d\n",aSubsetInfo.m_nFontType); 4170 aLine.append( "0 >>\nstream\n" ); 4171 } 4172 4173 endCompression(); 4174 disableStreamEncryption(); 4175 // close the file 4176 osl_closeFile( aFontFile ); 4177 4178 sal_uInt64 nEndPos = 0; 4179 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nEndPos ) ) ); 4180 // end the stream 4181 aLine.setLength( 0 ); 4182 aLine.append( "\nendstream\nendobj\n\n" ); 4183 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4184 4185 // emit stream length object 4186 CHECK_RETURN( updateObject( nStreamLengthObject ) ); 4187 aLine.setLength( 0 ); 4188 aLine.append( nStreamLengthObject ); 4189 aLine.append( " 0 obj\n" ); 4190 aLine.append( (sal_Int64)(nEndPos-nStartPos) ); 4191 aLine.append( "\nendobj\n\n" ); 4192 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4193 4194 // write font descriptor 4195 sal_Int32 nFontDescriptor = emitFontDescriptor( it->first, aSubsetInfo, lit->m_nFontID, nFontStream ); 4196 4197 if( nToUnicodeStream ) 4198 nToUnicodeStream = createToUnicodeCMap( pEncoding, &aUnicodes[0], pUnicodesPerGlyph, pEncToUnicodeIndex, nGlyphs ); 4199 4200 sal_Int32 nFontObject = createObject(); 4201 CHECK_RETURN( updateObject( nFontObject ) ); 4202 aLine.setLength( 0 ); 4203 aLine.append( nFontObject ); 4204 4205 aLine.append( " 0 obj\n" ); 4206 aLine.append( ((aSubsetInfo.m_nFontType & FontSubsetInfo::ANY_TYPE1) != 0) ? 4207 "<</Type/Font/Subtype/Type1/BaseFont/" : 4208 "<</Type/Font/Subtype/TrueType/BaseFont/" ); 4209 appendSubsetName( lit->m_nFontID, aSubsetInfo.m_aPSName, aLine ); 4210 aLine.append( "\n" 4211 "/FirstChar 0\n" 4212 "/LastChar " ); 4213 aLine.append( (sal_Int32)(nGlyphs-1) ); 4214 aLine.append( "\n" 4215 "/Widths[" ); 4216 for( int i = 0; i < nGlyphs; i++ ) 4217 { 4218 aLine.append( pWidths[ i ] ); 4219 aLine.append( ((i & 15) == 15) ? "\n" : " " ); 4220 } 4221 aLine.append( "]\n" 4222 "/FontDescriptor " ); 4223 aLine.append( nFontDescriptor ); 4224 aLine.append( " 0 R\n" ); 4225 if( nToUnicodeStream ) 4226 { 4227 aLine.append( "/ToUnicode " ); 4228 aLine.append( nToUnicodeStream ); 4229 aLine.append( " 0 R\n" ); 4230 } 4231 aLine.append( ">>\n" 4232 "endobj\n\n" ); 4233 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4234 4235 aFontIDToObject[ lit->m_nFontID ] = nFontObject; 4236 } 4237 else 4238 { 4239 const ImplFontData* pFont = it->first; 4240 rtl::OStringBuffer aErrorComment( 256 ); 4241 aErrorComment.append( "CreateFontSubset failed for font \"" ); 4242 aErrorComment.append( OUStringToOString( pFont->GetFamilyName(), RTL_TEXTENCODING_UTF8 ) ); 4243 aErrorComment.append( '\"' ); 4244 if( pFont->GetSlant() == ITALIC_NORMAL ) 4245 aErrorComment.append( " italic" ); 4246 else if( pFont->GetSlant() == ITALIC_OBLIQUE ) 4247 aErrorComment.append( " oblique" ); 4248 aErrorComment.append( " weight=" ); 4249 aErrorComment.append( sal_Int32(pFont->GetWeight()) ); 4250 emitComment( aErrorComment.getStr() ); 4251 } 4252 } 4253 } 4254 osl_removeFile( aTmpName.pData ); 4255 4256 // emit embedded fonts 4257 for( FontEmbedData::iterator eit = m_aEmbeddedFonts.begin(); eit != m_aEmbeddedFonts.end(); ++eit ) 4258 { 4259 std::map< sal_Int32, sal_Int32 > aObjects = emitEmbeddedFont( eit->first, eit->second ); 4260 for( std::map< sal_Int32, sal_Int32 >::iterator fit = aObjects.begin(); fit != aObjects.end(); ++fit ) 4261 { 4262 CHECK_RETURN( fit->second ); 4263 aFontIDToObject[ fit->first ] = fit->second; 4264 } 4265 } 4266 4267 // emit system fonts 4268 for( FontEmbedData::iterator sit = m_aSystemFonts.begin(); sit != m_aSystemFonts.end(); ++sit ) 4269 { 4270 std::map< sal_Int32, sal_Int32 > aObjects = emitSystemFont( sit->first, sit->second ); 4271 for( std::map< sal_Int32, sal_Int32 >::iterator fit = aObjects.begin(); fit != aObjects.end(); ++fit ) 4272 { 4273 CHECK_RETURN( fit->second ); 4274 aFontIDToObject[ fit->first ] = fit->second; 4275 } 4276 } 4277 4278 OStringBuffer aFontDict( 1024 ); 4279 aFontDict.append( getFontDictObject() ); 4280 aFontDict.append( " 0 obj\n" 4281 "<<" ); 4282 int ni = 0; 4283 for( std::map< sal_Int32, sal_Int32 >::iterator mit = aFontIDToObject.begin(); mit != aFontIDToObject.end(); ++mit ) 4284 { 4285 aFontDict.append( "/F" ); 4286 aFontDict.append( mit->first ); 4287 aFontDict.append( ' ' ); 4288 aFontDict.append( mit->second ); 4289 aFontDict.append( " 0 R" ); 4290 if( ((++ni) & 7) == 0 ) 4291 aFontDict.append( '\n' ); 4292 } 4293 // emit builtin font for widget apperances / variable text 4294 for( std::map< sal_Int32, sal_Int32 >::iterator it = m_aBuiltinFontToObjectMap.begin(); 4295 it != m_aBuiltinFontToObjectMap.end(); ++it ) 4296 { 4297 ImplPdfBuiltinFontData aData(m_aBuiltinFonts[it->first]); 4298 it->second = emitBuiltinFont( &aData, it->second ); 4299 } 4300 appendBuiltinFontsToDict( aFontDict ); 4301 aFontDict.append( "\n>>\nendobj\n\n" ); 4302 4303 CHECK_RETURN( updateObject( getFontDictObject() ) ); 4304 CHECK_RETURN( writeBuffer( aFontDict.getStr(), aFontDict.getLength() ) ); 4305 return true; 4306 } 4307 4308 sal_Int32 PDFWriterImpl::emitResources() 4309 { 4310 // emit shadings 4311 if( ! m_aGradients.empty() ) 4312 CHECK_RETURN( emitGradients() ); 4313 // emit tilings 4314 if( ! m_aTilings.empty() ) 4315 CHECK_RETURN( emitTilings() ); 4316 4317 // emit font dict 4318 CHECK_RETURN( emitFonts() ); 4319 4320 // emit Resource dict 4321 OStringBuffer aLine( 512 ); 4322 sal_Int32 nResourceDict = getResourceDictObj(); 4323 CHECK_RETURN( updateObject( nResourceDict ) ); 4324 aLine.setLength( 0 ); 4325 aLine.append( nResourceDict ); 4326 aLine.append( " 0 obj\n" ); 4327 m_aGlobalResourceDict.append( aLine, getFontDictObject() ); 4328 aLine.append( "endobj\n\n" ); 4329 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4330 return nResourceDict; 4331 } 4332 4333 sal_Int32 PDFWriterImpl::updateOutlineItemCount( std::vector< sal_Int32 >& rCounts, 4334 sal_Int32 nItemLevel, 4335 sal_Int32 nCurrentItemId ) 4336 { 4337 /* The /Count number of an item is 4338 positive: the number of visible subitems 4339 negative: the negative number of subitems that will become visible if 4340 the item gets opened 4341 see PDF ref 1.4 p 478 4342 */ 4343 4344 sal_Int32 nCount = 0; 4345 4346 if( m_aContext.OpenBookmarkLevels < 0 || // all levels arevisible 4347 m_aContext.OpenBookmarkLevels >= nItemLevel // this level is visible 4348 ) 4349 { 4350 PDFOutlineEntry& rItem = m_aOutline[ nCurrentItemId ]; 4351 sal_Int32 nChildren = rItem.m_aChildren.size(); 4352 for( sal_Int32 i = 0; i < nChildren; i++ ) 4353 nCount += updateOutlineItemCount( rCounts, nItemLevel+1, rItem.m_aChildren[i] ); 4354 rCounts[nCurrentItemId] = nCount; 4355 // return 1 (this item) + visible sub items 4356 if( nCount < 0 ) 4357 nCount = 0; 4358 nCount++; 4359 } 4360 else 4361 { 4362 // this bookmark level is invisible 4363 PDFOutlineEntry& rItem = m_aOutline[ nCurrentItemId ]; 4364 sal_Int32 nChildren = rItem.m_aChildren.size(); 4365 rCounts[ nCurrentItemId ] = -sal_Int32(rItem.m_aChildren.size()); 4366 for( sal_Int32 i = 0; i < nChildren; i++ ) 4367 updateOutlineItemCount( rCounts, nItemLevel+1, rItem.m_aChildren[i] ); 4368 nCount = -1; 4369 } 4370 4371 return nCount; 4372 } 4373 4374 sal_Int32 PDFWriterImpl::emitOutline() 4375 { 4376 int i, nItems = m_aOutline.size(); 4377 4378 // do we have an outline at all ? 4379 if( nItems < 2 ) 4380 return 0; 4381 4382 // reserve object numbers for all outline items 4383 for( i = 0; i < nItems; ++i ) 4384 m_aOutline[i].m_nObject = createObject(); 4385 4386 // update all parent, next and prev object ids 4387 for( i = 0; i < nItems; ++i ) 4388 { 4389 PDFOutlineEntry& rItem = m_aOutline[i]; 4390 int nChildren = rItem.m_aChildren.size(); 4391 4392 if( nChildren ) 4393 { 4394 for( int n = 0; n < nChildren; ++n ) 4395 { 4396 PDFOutlineEntry& rChild = m_aOutline[ rItem.m_aChildren[n] ]; 4397 4398 rChild.m_nParentObject = rItem.m_nObject; 4399 rChild.m_nPrevObject = (n > 0) ? m_aOutline[ rItem.m_aChildren[n-1] ].m_nObject : 0; 4400 rChild.m_nNextObject = (n < nChildren-1) ? m_aOutline[ rItem.m_aChildren[n+1] ].m_nObject : 0; 4401 } 4402 4403 } 4404 } 4405 4406 // calculate Count entries for all items 4407 std::vector< sal_Int32 > aCounts( nItems ); 4408 updateOutlineItemCount( aCounts, 0, 0 ); 4409 4410 // emit hierarchy 4411 for( i = 0; i < nItems; ++i ) 4412 { 4413 PDFOutlineEntry& rItem = m_aOutline[i]; 4414 OStringBuffer aLine( 1024 ); 4415 4416 CHECK_RETURN( updateObject( rItem.m_nObject ) ); 4417 aLine.append( rItem.m_nObject ); 4418 aLine.append( " 0 obj\n" ); 4419 aLine.append( "<<" ); 4420 // number of visible children (all levels) 4421 if( i > 0 || aCounts[0] > 0 ) 4422 { 4423 aLine.append( "/Count " ); 4424 aLine.append( aCounts[i] ); 4425 } 4426 if( ! rItem.m_aChildren.empty() ) 4427 { 4428 // children list: First, Last 4429 aLine.append( "/First " ); 4430 aLine.append( m_aOutline[rItem.m_aChildren.front()].m_nObject ); 4431 aLine.append( " 0 R/Last " ); 4432 aLine.append( m_aOutline[rItem.m_aChildren.back()].m_nObject ); 4433 aLine.append( " 0 R\n" ); 4434 } 4435 if( i > 0 ) 4436 { 4437 // Title, Dest, Parent, Prev, Next 4438 aLine.append( "/Title" ); 4439 appendUnicodeTextStringEncrypt( rItem.m_aTitle, rItem.m_nObject, aLine ); 4440 aLine.append( "\n" ); 4441 // Dest is not required 4442 if( rItem.m_nDestID >= 0 && rItem.m_nDestID < (sal_Int32)m_aDests.size() ) 4443 { 4444 aLine.append( "/Dest" ); 4445 appendDest( rItem.m_nDestID, aLine ); 4446 } 4447 aLine.append( "/Parent " ); 4448 aLine.append( rItem.m_nParentObject ); 4449 aLine.append( " 0 R" ); 4450 if( rItem.m_nPrevObject ) 4451 { 4452 aLine.append( "/Prev " ); 4453 aLine.append( rItem.m_nPrevObject ); 4454 aLine.append( " 0 R" ); 4455 } 4456 if( rItem.m_nNextObject ) 4457 { 4458 aLine.append( "/Next " ); 4459 aLine.append( rItem.m_nNextObject ); 4460 aLine.append( " 0 R" ); 4461 } 4462 } 4463 aLine.append( ">>\nendobj\n\n" ); 4464 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4465 } 4466 4467 return m_aOutline[0].m_nObject; 4468 } 4469 4470 #undef CHECK_RETURN 4471 #define CHECK_RETURN( x ) if( !x ) return false 4472 4473 bool PDFWriterImpl::appendDest( sal_Int32 nDestID, OStringBuffer& rBuffer ) 4474 { 4475 if( nDestID < 0 || nDestID >= (sal_Int32)m_aDests.size() ) 4476 { 4477 #if OSL_DEBUG_LEVEL > 1 4478 fprintf( stderr, "ERROR: invalid dest %d requested\n", (int)nDestID ); 4479 #endif 4480 return false; 4481 } 4482 4483 4484 const PDFDest& rDest = m_aDests[ nDestID ]; 4485 const PDFPage& rDestPage = m_aPages[ rDest.m_nPage ]; 4486 4487 rBuffer.append( '[' ); 4488 rBuffer.append( rDestPage.m_nPageObject ); 4489 rBuffer.append( " 0 R" ); 4490 4491 switch( rDest.m_eType ) 4492 { 4493 case PDFWriter::XYZ: 4494 default: 4495 rBuffer.append( "/XYZ " ); 4496 appendFixedInt( rDest.m_aRect.Left(), rBuffer ); 4497 rBuffer.append( ' ' ); 4498 appendFixedInt( rDest.m_aRect.Bottom(), rBuffer ); 4499 rBuffer.append( " 0" ); 4500 break; 4501 case PDFWriter::Fit: 4502 rBuffer.append( "/Fit" ); 4503 break; 4504 case PDFWriter::FitRectangle: 4505 rBuffer.append( "/FitR " ); 4506 appendFixedInt( rDest.m_aRect.Left(), rBuffer ); 4507 rBuffer.append( ' ' ); 4508 appendFixedInt( rDest.m_aRect.Top(), rBuffer ); 4509 rBuffer.append( ' ' ); 4510 appendFixedInt( rDest.m_aRect.Right(), rBuffer ); 4511 rBuffer.append( ' ' ); 4512 appendFixedInt( rDest.m_aRect.Bottom(), rBuffer ); 4513 break; 4514 case PDFWriter::FitHorizontal: 4515 rBuffer.append( "/FitH " ); 4516 appendFixedInt( rDest.m_aRect.Bottom(), rBuffer ); 4517 break; 4518 case PDFWriter::FitVertical: 4519 rBuffer.append( "/FitV " ); 4520 appendFixedInt( rDest.m_aRect.Left(), rBuffer ); 4521 break; 4522 case PDFWriter::FitPageBoundingBox: 4523 rBuffer.append( "/FitB" ); 4524 break; 4525 case PDFWriter::FitPageBoundingBoxHorizontal: 4526 rBuffer.append( "/FitBH " ); 4527 appendFixedInt( rDest.m_aRect.Bottom(), rBuffer ); 4528 break; 4529 case PDFWriter::FitPageBoundingBoxVertical: 4530 rBuffer.append( "/FitBV " ); 4531 appendFixedInt( rDest.m_aRect.Left(), rBuffer ); 4532 break; 4533 } 4534 rBuffer.append( ']' ); 4535 4536 return true; 4537 } 4538 4539 bool PDFWriterImpl::emitLinkAnnotations() 4540 { 4541 int nAnnots = m_aLinks.size(); 4542 for( int i = 0; i < nAnnots; i++ ) 4543 { 4544 const PDFLink& rLink = m_aLinks[i]; 4545 if( ! updateObject( rLink.m_nObject ) ) 4546 continue; 4547 4548 OStringBuffer aLine( 1024 ); 4549 aLine.append( rLink.m_nObject ); 4550 aLine.append( " 0 obj\n" ); 4551 //i59651 key /F set bits Print to 1 rest to 0. We don't set NoZoom NoRotate to 1, since it's a 'should' 4552 // see PDF 8.4.2 and ISO 19005-1:2005 6.5.3 4553 aLine.append( "<</Type/Annot" ); 4554 if( m_bIsPDF_A1 ) 4555 aLine.append( "/F 4" ); 4556 aLine.append( "/Subtype/Link/Border[0 0 0]/Rect[" ); 4557 4558 appendFixedInt( rLink.m_aRect.Left()-7, aLine );//the +7 to have a better shape of the border rectangle 4559 aLine.append( ' ' ); 4560 appendFixedInt( rLink.m_aRect.Top(), aLine ); 4561 aLine.append( ' ' ); 4562 appendFixedInt( rLink.m_aRect.Right()+7, aLine );//the +7 to have a better shape of the border rectangle 4563 aLine.append( ' ' ); 4564 appendFixedInt( rLink.m_aRect.Bottom(), aLine ); 4565 aLine.append( "]" ); 4566 if( rLink.m_nDest >= 0 ) 4567 { 4568 aLine.append( "/Dest" ); 4569 appendDest( rLink.m_nDest, aLine ); 4570 } 4571 else 4572 { 4573 /*--->i56629 4574 destination is external to the document, so 4575 we check in the following sequence: 4576 4577 if target type is neither .pdf, nor .od[tpgs], then 4578 check if relative or absolute and act accordingly (use URI or 'launch application' as requested) 4579 end processing 4580 else if target is .od[tpgs]: then 4581 if conversion of type from od[tpgs] to pdf is requested, convert it and this becomes the new target file 4582 processing continue 4583 4584 if (new)target is .pdf : then 4585 if GotToR is requested, then 4586 convert the target in GoToR where the fragment of the URI is 4587 considered the named destination in the target file, set relative or absolute as requested 4588 else strip the fragment from URL and then set URI or 'launch application' as requested 4589 */ 4590 // 4591 // FIXME: check if the decode mechanisms for URL processing throughout this implementation 4592 // are the correct one!! 4593 // 4594 // extract target file type 4595 INetURLObject aDocumentURL( m_aContext.BaseURL ); 4596 INetURLObject aTargetURL( rLink.m_aURL ); 4597 sal_Int32 nChangeFileExtensionToPDF = 0; 4598 sal_Int32 nSetGoToRMode = 0; 4599 sal_Bool bTargetHasPDFExtension = sal_False; 4600 INetProtocol eTargetProtocol = aTargetURL.GetProtocol(); 4601 sal_Bool bIsUNCPath = sal_False; 4602 // check if the protocol is a known one, or if there is no protocol at all (on target only) 4603 // if there is no protocol, make the target relative to the current document directory 4604 // getting the needed URL information from the current document path 4605 if( eTargetProtocol == INET_PROT_NOT_VALID ) 4606 { 4607 if( rLink.m_aURL.getLength() > 4 && rLink.m_aURL.compareToAscii( "\\\\\\\\", 4 ) == 0) 4608 { 4609 bIsUNCPath = sal_True; 4610 } 4611 else 4612 { 4613 INetURLObject aNewBase( aDocumentURL );//duplicate document URL 4614 aNewBase.removeSegment(); //remove last segment from it, obtaining the base URL of the 4615 //target document 4616 aNewBase.insertName( rLink.m_aURL ); 4617 aTargetURL = aNewBase;//reassign the new target URL 4618 //recompute the target protocol, with the new URL 4619 //normal URL processing resumes 4620 eTargetProtocol = aTargetURL.GetProtocol(); 4621 } 4622 } 4623 4624 rtl::OUString aFileExtension = aTargetURL.GetFileExtension(); 4625 4626 // Check if the URL ends in '/': if yes it's a directory, 4627 // it will be forced to a URI link. 4628 // possibly a malformed URI, leave it as it is, force as URI 4629 if( aTargetURL.hasFinalSlash() ) 4630 m_aContext.DefaultLinkAction = PDFWriter::URIAction; 4631 4632 if( aFileExtension.getLength() > 0 ) 4633 { 4634 if( m_aContext.ConvertOOoTargetToPDFTarget ) 4635 { 4636 //examine the file type (.odm .odt. .odp, odg, ods) 4637 if( aFileExtension.equalsIgnoreAsciiCase(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "odm" ) ) ) ) 4638 nChangeFileExtensionToPDF++; 4639 if( aFileExtension.equalsIgnoreAsciiCase(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "odt" ) ) ) ) 4640 nChangeFileExtensionToPDF++; 4641 else if( aFileExtension.equalsIgnoreAsciiCase(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "odp" ) ) ) ) 4642 nChangeFileExtensionToPDF++; 4643 else if( aFileExtension.equalsIgnoreAsciiCase(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "odg" ) ) ) ) 4644 nChangeFileExtensionToPDF++; 4645 else if( aFileExtension.equalsIgnoreAsciiCase(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "ods" ) ) ) ) 4646 nChangeFileExtensionToPDF++; 4647 if( nChangeFileExtensionToPDF ) 4648 aTargetURL.setExtension(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "pdf" ) ) ); 4649 } 4650 //check if extension is pdf, see if GoToR should be forced 4651 bTargetHasPDFExtension = aTargetURL.GetFileExtension().equalsIgnoreAsciiCase(rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "pdf" ) ) ); 4652 if( m_aContext.ForcePDFAction && bTargetHasPDFExtension ) 4653 nSetGoToRMode++; 4654 } 4655 //prepare the URL, if relative or not 4656 INetProtocol eBaseProtocol = aDocumentURL.GetProtocol(); 4657 //queue the string common to all types of actions 4658 aLine.append( "/A<</Type/Action/S"); 4659 if( bIsUNCPath ) // handle Win UNC paths 4660 { 4661 aLine.append( "/Launch/Win<</F" ); 4662 // INetURLObject is not good with UNC paths, use original path 4663 appendLiteralStringEncrypt( rLink.m_aURL, rLink.m_nObject, aLine, osl_getThreadTextEncoding() ); 4664 aLine.append( ">>" ); 4665 } 4666 else 4667 { 4668 bool bSetRelative = false; 4669 bool bFileSpec = false; 4670 //check if relative file link is requested and if the protocol is 'file://' 4671 if( m_aContext.RelFsys && eBaseProtocol == eTargetProtocol && eTargetProtocol == INET_PROT_FILE ) 4672 bSetRelative = true; 4673 4674 rtl::OUString aFragment = aTargetURL.GetMark( INetURLObject::NO_DECODE /*DECODE_WITH_CHARSET*/ ); //fragment as is, 4675 if( nSetGoToRMode == 0 ) 4676 { 4677 switch( m_aContext.DefaultLinkAction ) 4678 { 4679 default: 4680 case PDFWriter::URIAction : 4681 case PDFWriter::URIActionDestination : 4682 aLine.append( "/URI/URI" ); 4683 break; 4684 case PDFWriter::LaunchAction: 4685 // now: 4686 // if a launch action is requested and the hyperlink target has a fragment 4687 // and the target file does not have a pdf extension, or it's not a 'file:://' protocol 4688 // then force the uri action on it 4689 // This code will permit the correct opening of application on web pages, the one that 4690 // normally have fragments (but I may be wrong...) 4691 // and will force the use of URI when the protocol is not file:// 4692 if( (aFragment.getLength() > 0 && !bTargetHasPDFExtension) || 4693 eTargetProtocol != INET_PROT_FILE ) 4694 aLine.append( "/URI/URI" ); 4695 else 4696 { 4697 aLine.append( "/Launch/F" ); 4698 bFileSpec = true; 4699 } 4700 break; 4701 } 4702 } 4703 //fragment are encoded in the same way as in the named destination processing 4704 if( nSetGoToRMode ) 4705 {//add the fragment 4706 rtl::OUString aURLNoMark = aTargetURL.GetURLNoMark( INetURLObject::DECODE_WITH_CHARSET ); 4707 aLine.append("/GoToR"); 4708 aLine.append("/F"); 4709 bFileSpec = true; 4710 appendLiteralStringEncrypt( bSetRelative ? INetURLObject::GetRelURL( m_aContext.BaseURL, aURLNoMark, 4711 INetURLObject::WAS_ENCODED, 4712 INetURLObject::DECODE_WITH_CHARSET ) : 4713 aURLNoMark, rLink.m_nObject, aLine, osl_getThreadTextEncoding() ); 4714 if( aFragment.getLength() > 0 ) 4715 { 4716 aLine.append("/D/"); 4717 appendDestinationName( aFragment , aLine ); 4718 } 4719 } 4720 else 4721 { 4722 // change the fragment to accomodate the bookmark (only if the file extension is PDF and 4723 // the requested action is of the correct type) 4724 if(m_aContext.DefaultLinkAction == PDFWriter::URIActionDestination && 4725 bTargetHasPDFExtension && aFragment.getLength() > 0 ) 4726 { 4727 OStringBuffer aLineLoc( 1024 ); 4728 appendDestinationName( aFragment , aLineLoc ); 4729 //substitute the fragment 4730 aTargetURL.SetMark( aLineLoc.getStr() ); 4731 } 4732 rtl::OUString aURL = aTargetURL.GetMainURL( bFileSpec ? INetURLObject::DECODE_WITH_CHARSET : INetURLObject::NO_DECODE ); 4733 // check if we have a URL available, if the string is empty, set it as the original one 4734 // if( aURL.getLength() == 0 ) 4735 // appendLiteralStringEncrypt( rLink.m_aURL , rLink.m_nObject, aLine ); 4736 // else 4737 appendLiteralStringEncrypt( bSetRelative ? INetURLObject::GetRelURL( m_aContext.BaseURL, aURL, 4738 INetURLObject::WAS_ENCODED, 4739 bFileSpec ? INetURLObject::DECODE_WITH_CHARSET : INetURLObject::NO_DECODE 4740 ) : 4741 aURL , rLink.m_nObject, aLine, osl_getThreadTextEncoding() ); 4742 } 4743 //<--- i56629 4744 } 4745 aLine.append( ">>\n" ); 4746 } 4747 if( rLink.m_nStructParent > 0 ) 4748 { 4749 aLine.append( "/StructParent " ); 4750 aLine.append( rLink.m_nStructParent ); 4751 } 4752 aLine.append( ">>\nendobj\n\n" ); 4753 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4754 } 4755 4756 return true; 4757 } 4758 4759 bool PDFWriterImpl::emitNoteAnnotations() 4760 { 4761 // emit note annotations 4762 int nAnnots = m_aNotes.size(); 4763 for( int i = 0; i < nAnnots; i++ ) 4764 { 4765 const PDFNoteEntry& rNote = m_aNotes[i]; 4766 if( ! updateObject( rNote.m_nObject ) ) 4767 return false; 4768 4769 OStringBuffer aLine( 1024 ); 4770 aLine.append( rNote.m_nObject ); 4771 aLine.append( " 0 obj\n" ); 4772 //i59651 key /F set bits Print to 1 rest to 0. We don't set NoZoom NoRotate to 1, since it's a 'should' 4773 // see PDF 8.4.2 and ISO 19005-1:2005 6.5.3 4774 aLine.append( "<</Type/Annot" ); 4775 if( m_bIsPDF_A1 ) 4776 aLine.append( "/F 4" ); 4777 aLine.append( "/Subtype/Text/Rect[" ); 4778 4779 appendFixedInt( rNote.m_aRect.Left(), aLine ); 4780 aLine.append( ' ' ); 4781 appendFixedInt( rNote.m_aRect.Top(), aLine ); 4782 aLine.append( ' ' ); 4783 appendFixedInt( rNote.m_aRect.Right(), aLine ); 4784 aLine.append( ' ' ); 4785 appendFixedInt( rNote.m_aRect.Bottom(), aLine ); 4786 aLine.append( "]" ); 4787 4788 // contents of the note (type text string) 4789 aLine.append( "/Contents\n" ); 4790 appendUnicodeTextStringEncrypt( rNote.m_aContents.Contents, rNote.m_nObject, aLine ); 4791 aLine.append( "\n" ); 4792 4793 // optional title 4794 if( rNote.m_aContents.Title.Len() ) 4795 { 4796 aLine.append( "/T" ); 4797 appendUnicodeTextStringEncrypt( rNote.m_aContents.Title, rNote.m_nObject, aLine ); 4798 aLine.append( "\n" ); 4799 } 4800 4801 aLine.append( ">>\nendobj\n\n" ); 4802 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 4803 } 4804 return true; 4805 } 4806 4807 Font PDFWriterImpl::replaceFont( const Font& rControlFont, const Font& rAppSetFont ) 4808 { 4809 bool bAdjustSize = false; 4810 4811 Font aFont( rControlFont ); 4812 if( ! aFont.GetName().Len() ) 4813 { 4814 aFont = rAppSetFont; 4815 if( rControlFont.GetHeight() ) 4816 aFont.SetSize( Size( 0, rControlFont.GetHeight() ) ); 4817 else 4818 bAdjustSize = true; 4819 if( rControlFont.GetItalic() != ITALIC_DONTKNOW ) 4820 aFont.SetItalic( rControlFont.GetItalic() ); 4821 if( rControlFont.GetWeight() != WEIGHT_DONTKNOW ) 4822 aFont.SetWeight( rControlFont.GetWeight() ); 4823 } 4824 else if( ! aFont.GetHeight() ) 4825 { 4826 aFont.SetSize( rAppSetFont.GetSize() ); 4827 bAdjustSize = true; 4828 } 4829 if( bAdjustSize ) 4830 { 4831 Size aFontSize = aFont.GetSize(); 4832 OutputDevice* pDefDev = Application::GetDefaultDevice(); 4833 aFontSize = OutputDevice::LogicToLogic( aFontSize, pDefDev->GetMapMode(), getMapMode() ); 4834 aFont.SetSize( aFontSize ); 4835 } 4836 return aFont; 4837 } 4838 4839 sal_Int32 PDFWriterImpl::getBestBuiltinFont( const Font& rFont ) 4840 { 4841 sal_Int32 nBest = 4; // default to Helvetica 4842 OUString aFontName( rFont.GetName() ); 4843 aFontName = aFontName.toAsciiLowerCase(); 4844 4845 if( aFontName.indexOf( OUString( RTL_CONSTASCII_USTRINGPARAM( "times" ) ) ) != -1 ) 4846 nBest = 8; 4847 else if( aFontName.indexOf( OUString( RTL_CONSTASCII_USTRINGPARAM( "courier" ) ) ) != -1 ) 4848 nBest = 0; 4849 else if( aFontName.indexOf( OUString( RTL_CONSTASCII_USTRINGPARAM( "dingbats" ) ) ) != -1 ) 4850 nBest = 13; 4851 else if( aFontName.indexOf( OUString( RTL_CONSTASCII_USTRINGPARAM( "symbol" ) ) ) != -1 ) 4852 nBest = 12; 4853 if( nBest < 12 ) 4854 { 4855 if( rFont.GetItalic() == ITALIC_OBLIQUE || rFont.GetItalic() == ITALIC_NORMAL ) 4856 nBest += 1; 4857 if( rFont.GetWeight() > WEIGHT_MEDIUM ) 4858 nBest += 2; 4859 } 4860 4861 if( m_aBuiltinFontToObjectMap.find( nBest ) == m_aBuiltinFontToObjectMap.end() ) 4862 m_aBuiltinFontToObjectMap[ nBest ] = createObject(); 4863 4864 return nBest; 4865 } 4866 4867 static inline const Color& replaceColor( const Color& rCol1, const Color& rCol2 ) 4868 { 4869 return (rCol1 == Color( COL_TRANSPARENT )) ? rCol2 : rCol1; 4870 } 4871 4872 void PDFWriterImpl::createDefaultPushButtonAppearance( PDFWidget& rButton, const PDFWriter::PushButtonWidget& rWidget ) 4873 { 4874 const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings(); 4875 4876 // save graphics state 4877 push( sal::static_int_cast<sal_uInt16>(~0U) ); 4878 4879 // transform relative to control's coordinates since an 4880 // appearance stream is a form XObject 4881 // this relies on the m_aRect member of rButton NOT already being transformed 4882 // to default user space 4883 if( rWidget.Background || rWidget.Border ) 4884 { 4885 setLineColor( rWidget.Border ? replaceColor( rWidget.BorderColor, rSettings.GetLightColor() ) : Color( COL_TRANSPARENT ) ); 4886 setFillColor( rWidget.Background ? replaceColor( rWidget.BackgroundColor, rSettings.GetDialogColor() ) : Color( COL_TRANSPARENT ) ); 4887 drawRectangle( rWidget.Location ); 4888 } 4889 // prepare font to use 4890 Font aFont = replaceFont( rWidget.TextFont, rSettings.GetPushButtonFont() ); 4891 setFont( aFont ); 4892 setTextColor( replaceColor( rWidget.TextColor, rSettings.GetButtonTextColor() ) ); 4893 4894 drawText( rButton.m_aRect, rButton.m_aText, rButton.m_nTextStyle ); 4895 4896 // create DA string while local mapmode is still in place 4897 // (that is before endRedirect()) 4898 OStringBuffer aDA( 256 ); 4899 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetButtonTextColor() ), aDA ); 4900 Font aDummyFont( String( RTL_CONSTASCII_USTRINGPARAM( "Helvetica" ) ), aFont.GetSize() ); 4901 sal_Int32 nDummyBuiltin = getBestBuiltinFont( aDummyFont ); 4902 aDA.append( ' ' ); 4903 aDA.append( m_aBuiltinFonts[nDummyBuiltin].getNameObject() ); 4904 aDA.append( ' ' ); 4905 m_aPages[m_nCurrentPage].appendMappedLength( sal_Int32( aFont.GetHeight() ), aDA ); 4906 aDA.append( " Tf" ); 4907 rButton.m_aDAString = aDA.makeStringAndClear(); 4908 4909 pop(); 4910 4911 rButton.m_aAppearances[ "N" ][ "Standard" ] = new SvMemoryStream(); 4912 4913 /* seems like a bad hack but at least works in both AR5 and 6: 4914 we draw the button ourselves and tell AR 4915 the button would be totally transparent with no text 4916 4917 One would expect that simply setting a normal appearance 4918 should suffice, but no, as soon as the user actually presses 4919 the button and an action is tied to it (gasp! a button that 4920 does something) the appearance gets replaced by some crap that AR 4921 creates on the fly even if no DA or MK is given. On AR6 at least 4922 the DA and MK work as expected, but on AR5 this creates a region 4923 filled with the background color but nor text. Urgh. 4924 */ 4925 rButton.m_aMKDict = "/BC [] /BG [] /CA"; 4926 rButton.m_aMKDictCAString = ""; 4927 } 4928 4929 Font PDFWriterImpl::drawFieldBorder( PDFWidget& rIntern, 4930 const PDFWriter::AnyWidget& rWidget, 4931 const StyleSettings& rSettings ) 4932 { 4933 Font aFont = replaceFont( rWidget.TextFont, rSettings.GetFieldFont() ); 4934 4935 if( rWidget.Background || rWidget.Border ) 4936 { 4937 if( rWidget.Border && rWidget.BorderColor == Color( COL_TRANSPARENT ) ) 4938 { 4939 sal_Int32 nDelta = getReferenceDevice()->ImplGetDPIX() / 500; 4940 if( nDelta < 1 ) 4941 nDelta = 1; 4942 setLineColor( Color( COL_TRANSPARENT ) ); 4943 Rectangle aRect = rIntern.m_aRect; 4944 setFillColor( rSettings.GetLightBorderColor() ); 4945 drawRectangle( aRect ); 4946 aRect.Left() += nDelta; aRect.Top() += nDelta; 4947 aRect.Right() -= nDelta; aRect.Bottom() -= nDelta; 4948 setFillColor( rSettings.GetFieldColor() ); 4949 drawRectangle( aRect ); 4950 setFillColor( rSettings.GetLightColor() ); 4951 drawRectangle( Rectangle( Point( aRect.Left(), aRect.Bottom()-nDelta ), aRect.BottomRight() ) ); 4952 drawRectangle( Rectangle( Point( aRect.Right()-nDelta, aRect.Top() ), aRect.BottomRight() ) ); 4953 setFillColor( rSettings.GetDarkShadowColor() ); 4954 drawRectangle( Rectangle( aRect.TopLeft(), Point( aRect.Left()+nDelta, aRect.Bottom() ) ) ); 4955 drawRectangle( Rectangle( aRect.TopLeft(), Point( aRect.Right(), aRect.Top()+nDelta ) ) ); 4956 } 4957 else 4958 { 4959 setLineColor( rWidget.Border ? replaceColor( rWidget.BorderColor, rSettings.GetShadowColor() ) : Color( COL_TRANSPARENT ) ); 4960 setFillColor( rWidget.Background ? replaceColor( rWidget.BackgroundColor, rSettings.GetFieldColor() ) : Color( COL_TRANSPARENT ) ); 4961 drawRectangle( rIntern.m_aRect ); 4962 } 4963 4964 if( rWidget.Border ) 4965 { 4966 // adjust edit area accounting for border 4967 sal_Int32 nDelta = aFont.GetHeight()/4; 4968 if( nDelta < 1 ) 4969 nDelta = 1; 4970 rIntern.m_aRect.Left() += nDelta; 4971 rIntern.m_aRect.Top() += nDelta; 4972 rIntern.m_aRect.Right() -= nDelta; 4973 rIntern.m_aRect.Bottom()-= nDelta; 4974 } 4975 } 4976 return aFont; 4977 } 4978 4979 void PDFWriterImpl::createDefaultEditAppearance( PDFWidget& rEdit, const PDFWriter::EditWidget& rWidget ) 4980 { 4981 const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings(); 4982 SvMemoryStream* pEditStream = new SvMemoryStream( 1024, 1024 ); 4983 4984 push( sal::static_int_cast<sal_uInt16>(~0U) ); 4985 4986 // prepare font to use, draw field border 4987 Font aFont = drawFieldBorder( rEdit, rWidget, rSettings ); 4988 sal_Int32 nBest = m_aContext.FieldsUseSystemFonts ? getSystemFont( aFont ): getBestBuiltinFont( aFont ); 4989 4990 // prepare DA string 4991 OStringBuffer aDA( 32 ); 4992 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetFieldTextColor() ), aDA ); 4993 aDA.append( ' ' ); 4994 if( m_aContext.FieldsUseSystemFonts ) 4995 { 4996 aDA.append( "/F" ); 4997 aDA.append( nBest ); 4998 4999 OStringBuffer aDR( 32 ); 5000 aDR.append( "/Font " ); 5001 aDR.append( getFontDictObject() ); 5002 aDR.append( " 0 R" ); 5003 rEdit.m_aDRDict = aDR.makeStringAndClear(); 5004 } 5005 else 5006 aDA.append( m_aBuiltinFonts[nBest].getNameObject() ); 5007 aDA.append( ' ' ); 5008 m_aPages[ m_nCurrentPage ].appendMappedLength( sal_Int32( aFont.GetHeight() ), aDA ); 5009 aDA.append( " Tf" ); 5010 5011 /* create an empty appearance stream, let the viewer create 5012 the appearance at runtime. This is because AR5 seems to 5013 paint the widget appearance always, and a dynamically created 5014 appearance on top of it. AR6 is well behaved in that regard, so 5015 that behaviour seems to be a bug. Anyway this empty appearance 5016 relies on /NeedAppearances in the AcroForm dictionary set to "true" 5017 */ 5018 beginRedirect( pEditStream, rEdit.m_aRect ); 5019 OStringBuffer aAppearance( 32 ); 5020 aAppearance.append( "/Tx BMC\nEMC\n" ); 5021 writeBuffer( aAppearance.getStr(), aAppearance.getLength() ); 5022 5023 endRedirect(); 5024 pop(); 5025 5026 rEdit.m_aAppearances[ "N" ][ "Standard" ] = pEditStream; 5027 5028 rEdit.m_aDAString = aDA.makeStringAndClear(); 5029 } 5030 5031 void PDFWriterImpl::createDefaultListBoxAppearance( PDFWidget& rBox, const PDFWriter::ListBoxWidget& rWidget ) 5032 { 5033 const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings(); 5034 SvMemoryStream* pListBoxStream = new SvMemoryStream( 1024, 1024 ); 5035 5036 push( sal::static_int_cast<sal_uInt16>(~0U) ); 5037 5038 // prepare font to use, draw field border 5039 Font aFont = drawFieldBorder( rBox, rWidget, rSettings ); 5040 sal_Int32 nBest = m_aContext.FieldsUseSystemFonts ? getSystemFont( aFont ): getBestBuiltinFont( aFont ); 5041 5042 beginRedirect( pListBoxStream, rBox.m_aRect ); 5043 OStringBuffer aAppearance( 64 ); 5044 5045 #if 0 5046 if( ! rWidget.DropDown ) 5047 { 5048 // prepare linewidth for DA string hack, see below 5049 Size aFontSize = lcl_convert( m_aGraphicsStack.front().m_aMapMode, 5050 m_aMapMode, 5051 getReferenceDevice(), 5052 Size( 0, aFont.GetHeight() ) ); 5053 sal_Int32 nLW = aFontSize.Height() / 40; 5054 appendFixedInt( nLW > 0 ? nLW : 1, aAppearance ); 5055 aAppearance.append( " w\n" ); 5056 writeBuffer( aAppearance.getStr(), aAppearance.getLength() ); 5057 aAppearance.setLength( 0 ); 5058 } 5059 #endif 5060 5061 setLineColor( Color( COL_TRANSPARENT ) ); 5062 setFillColor( replaceColor( rWidget.BackgroundColor, rSettings.GetFieldColor() ) ); 5063 drawRectangle( rBox.m_aRect ); 5064 5065 // empty appearance, see createDefaultEditAppearance for reference 5066 aAppearance.append( "/Tx BMC\nEMC\n" ); 5067 writeBuffer( aAppearance.getStr(), aAppearance.getLength() ); 5068 5069 endRedirect(); 5070 pop(); 5071 5072 rBox.m_aAppearances[ "N" ][ "Standard" ] = pListBoxStream; 5073 5074 // prepare DA string 5075 OStringBuffer aDA( 256 ); 5076 #if 0 5077 if( !rWidget.DropDown ) 5078 { 5079 /* another of AR5's peculiarities: the selected item of a choice 5080 field is highlighted using the non stroking color - same as the 5081 text color. so workaround that by using text rendering mode 2 5082 (fill, then stroke) and set the stroking color 5083 */ 5084 appendStrokingColor( replaceColor( rWidget.BackgroundColor, rSettings.GetFieldColor() ), aDA ); 5085 aDA.append( " 2 Tr " ); 5086 } 5087 #endif 5088 // prepare DA string 5089 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetFieldTextColor() ), aDA ); 5090 aDA.append( ' ' ); 5091 if( m_aContext.FieldsUseSystemFonts ) 5092 { 5093 aDA.append( "/F" ); 5094 aDA.append( nBest ); 5095 5096 OStringBuffer aDR( 32 ); 5097 aDR.append( "/Font " ); 5098 aDR.append( getFontDictObject() ); 5099 aDR.append( " 0 R" ); 5100 rBox.m_aDRDict = aDR.makeStringAndClear(); 5101 } 5102 else 5103 aDA.append( m_aBuiltinFonts[nBest].getNameObject() ); 5104 aDA.append( ' ' ); 5105 m_aPages[ m_nCurrentPage ].appendMappedLength( sal_Int32( aFont.GetHeight() ), aDA ); 5106 aDA.append( " Tf" ); 5107 rBox.m_aDAString = aDA.makeStringAndClear(); 5108 } 5109 5110 void PDFWriterImpl::createDefaultCheckBoxAppearance( PDFWidget& rBox, const PDFWriter::CheckBoxWidget& rWidget ) 5111 { 5112 const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings(); 5113 5114 // save graphics state 5115 push( sal::static_int_cast<sal_uInt16>(~0U) ); 5116 5117 if( rWidget.Background || rWidget.Border ) 5118 { 5119 setLineColor( rWidget.Border ? replaceColor( rWidget.BorderColor, rSettings.GetCheckedColor() ) : Color( COL_TRANSPARENT ) ); 5120 setFillColor( rWidget.Background ? replaceColor( rWidget.BackgroundColor, rSettings.GetFieldColor() ) : Color( COL_TRANSPARENT ) ); 5121 drawRectangle( rBox.m_aRect ); 5122 } 5123 5124 Font aFont = replaceFont( rWidget.TextFont, rSettings.GetRadioCheckFont() ); 5125 setFont( aFont ); 5126 Size aFontSize = aFont.GetSize(); 5127 if( aFontSize.Height() > rBox.m_aRect.GetHeight() ) 5128 aFontSize.Height() = rBox.m_aRect.GetHeight(); 5129 sal_Int32 nDelta = aFontSize.Height()/10; 5130 if( nDelta < 1 ) 5131 nDelta = 1; 5132 5133 Rectangle aCheckRect, aTextRect; 5134 if( rWidget.ButtonIsLeft ) 5135 { 5136 aCheckRect.Left() = rBox.m_aRect.Left() + nDelta; 5137 aCheckRect.Top() = rBox.m_aRect.Top() + (rBox.m_aRect.GetHeight()-aFontSize.Height())/2; 5138 aCheckRect.Right() = aCheckRect.Left() + aFontSize.Height(); 5139 aCheckRect.Bottom() = aCheckRect.Top() + aFontSize.Height(); 5140 5141 // #i74206# handle small controls without text area 5142 while( aCheckRect.GetWidth() > rBox.m_aRect.GetWidth() && aCheckRect.GetWidth() > nDelta ) 5143 { 5144 aCheckRect.Right() -= nDelta; 5145 aCheckRect.Top() += nDelta/2; 5146 aCheckRect.Bottom() -= nDelta - (nDelta/2); 5147 } 5148 5149 aTextRect.Left() = rBox.m_aRect.Left() + aCheckRect.GetWidth()+5*nDelta; 5150 aTextRect.Top() = rBox.m_aRect.Top(); 5151 aTextRect.Right() = aTextRect.Left() + rBox.m_aRect.GetWidth() - aCheckRect.GetWidth()-6*nDelta; 5152 aTextRect.Bottom() = rBox.m_aRect.Bottom(); 5153 } 5154 else 5155 { 5156 aCheckRect.Left() = rBox.m_aRect.Right() - nDelta - aFontSize.Height(); 5157 aCheckRect.Top() = rBox.m_aRect.Top() + (rBox.m_aRect.GetHeight()-aFontSize.Height())/2; 5158 aCheckRect.Right() = aCheckRect.Left() + aFontSize.Height(); 5159 aCheckRect.Bottom() = aCheckRect.Top() + aFontSize.Height(); 5160 5161 // #i74206# handle small controls without text area 5162 while( aCheckRect.GetWidth() > rBox.m_aRect.GetWidth() && aCheckRect.GetWidth() > nDelta ) 5163 { 5164 aCheckRect.Left() += nDelta; 5165 aCheckRect.Top() += nDelta/2; 5166 aCheckRect.Bottom() -= nDelta - (nDelta/2); 5167 } 5168 5169 aTextRect.Left() = rBox.m_aRect.Left(); 5170 aTextRect.Top() = rBox.m_aRect.Top(); 5171 aTextRect.Right() = aTextRect.Left() + rBox.m_aRect.GetWidth() - aCheckRect.GetWidth()-6*nDelta; 5172 aTextRect.Bottom() = rBox.m_aRect.Bottom(); 5173 } 5174 setLineColor( Color( COL_BLACK ) ); 5175 setFillColor( Color( COL_TRANSPARENT ) ); 5176 OStringBuffer aLW( 32 ); 5177 aLW.append( "q " ); 5178 m_aPages[m_nCurrentPage].appendMappedLength( nDelta, aLW ); 5179 aLW.append( " w " ); 5180 writeBuffer( aLW.getStr(), aLW.getLength() ); 5181 drawRectangle( aCheckRect ); 5182 writeBuffer( " Q\n", 3 ); 5183 setTextColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ) ); 5184 drawText( aTextRect, rBox.m_aText, rBox.m_nTextStyle ); 5185 5186 pop(); 5187 5188 OStringBuffer aDA( 256 ); 5189 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ), aDA ); 5190 sal_Int32 nBest = getBestBuiltinFont( Font( String( RTL_CONSTASCII_USTRINGPARAM( "ZapfDingbats" ) ), aFont.GetSize() ) ); 5191 aDA.append( ' ' ); 5192 aDA.append( m_aBuiltinFonts[nBest].getNameObject() ); 5193 aDA.append( " 0 Tf" ); 5194 rBox.m_aDAString = aDA.makeStringAndClear(); 5195 rBox.m_aMKDict = "/CA"; 5196 rBox.m_aMKDictCAString = "8"; 5197 rBox.m_aRect = aCheckRect; 5198 5199 // create appearance streams 5200 sal_Char cMark = '8'; 5201 sal_Int32 nCharXOffset = 1000-m_aBuiltinFonts[13].m_aWidths[sal_Int32(cMark)]; 5202 nCharXOffset *= aCheckRect.GetHeight(); 5203 nCharXOffset /= 2000; 5204 sal_Int32 nCharYOffset = 1000- 5205 (m_aBuiltinFonts[13].m_nAscent+m_aBuiltinFonts[13].m_nDescent); // descent is negative 5206 nCharYOffset *= aCheckRect.GetHeight(); 5207 nCharYOffset /= 2000; 5208 5209 SvMemoryStream* pCheckStream = new SvMemoryStream( 256, 256 ); 5210 beginRedirect( pCheckStream, aCheckRect ); 5211 aDA.append( "/Tx BMC\nq BT\n" ); 5212 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ), aDA ); 5213 aDA.append( ' ' ); 5214 aDA.append( m_aBuiltinFonts[nBest].getNameObject() ); 5215 aDA.append( ' ' ); 5216 m_aPages[ m_nCurrentPage ].appendMappedLength( sal_Int32( aCheckRect.GetHeight() ), aDA ); 5217 aDA.append( " Tf\n" ); 5218 m_aPages[ m_nCurrentPage ].appendMappedLength( nCharXOffset, aDA ); 5219 aDA.append( " " ); 5220 m_aPages[ m_nCurrentPage ].appendMappedLength( nCharYOffset, aDA ); 5221 aDA.append( " Td (" ); 5222 aDA.append( cMark ); 5223 aDA.append( ") Tj\nET\nQ\nEMC\n" ); 5224 writeBuffer( aDA.getStr(), aDA.getLength() ); 5225 endRedirect(); 5226 rBox.m_aAppearances[ "N" ][ "Yes" ] = pCheckStream; 5227 5228 SvMemoryStream* pUncheckStream = new SvMemoryStream( 256, 256 ); 5229 beginRedirect( pUncheckStream, aCheckRect ); 5230 writeBuffer( "/Tx BMC\nEMC\n", 12 ); 5231 endRedirect(); 5232 rBox.m_aAppearances[ "N" ][ "Off" ] = pUncheckStream; 5233 } 5234 5235 void PDFWriterImpl::createDefaultRadioButtonAppearance( PDFWidget& rBox, const PDFWriter::RadioButtonWidget& rWidget ) 5236 { 5237 const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings(); 5238 5239 // save graphics state 5240 push( sal::static_int_cast<sal_uInt16>(~0U) ); 5241 5242 if( rWidget.Background || rWidget.Border ) 5243 { 5244 setLineColor( rWidget.Border ? replaceColor( rWidget.BorderColor, rSettings.GetCheckedColor() ) : Color( COL_TRANSPARENT ) ); 5245 setFillColor( rWidget.Background ? replaceColor( rWidget.BackgroundColor, rSettings.GetFieldColor() ) : Color( COL_TRANSPARENT ) ); 5246 drawRectangle( rBox.m_aRect ); 5247 } 5248 5249 Font aFont = replaceFont( rWidget.TextFont, rSettings.GetRadioCheckFont() ); 5250 setFont( aFont ); 5251 Size aFontSize = aFont.GetSize(); 5252 if( aFontSize.Height() > rBox.m_aRect.GetHeight() ) 5253 aFontSize.Height() = rBox.m_aRect.GetHeight(); 5254 sal_Int32 nDelta = aFontSize.Height()/10; 5255 if( nDelta < 1 ) 5256 nDelta = 1; 5257 5258 Rectangle aCheckRect, aTextRect; 5259 if( rWidget.ButtonIsLeft ) 5260 { 5261 aCheckRect.Left() = rBox.m_aRect.Left() + nDelta; 5262 aCheckRect.Top() = rBox.m_aRect.Top() + (rBox.m_aRect.GetHeight()-aFontSize.Height())/2; 5263 aCheckRect.Right() = aCheckRect.Left() + aFontSize.Height(); 5264 aCheckRect.Bottom() = aCheckRect.Top() + aFontSize.Height(); 5265 5266 // #i74206# handle small controls without text area 5267 while( aCheckRect.GetWidth() > rBox.m_aRect.GetWidth() && aCheckRect.GetWidth() > nDelta ) 5268 { 5269 aCheckRect.Right() -= nDelta; 5270 aCheckRect.Top() += nDelta/2; 5271 aCheckRect.Bottom() -= nDelta - (nDelta/2); 5272 } 5273 5274 aTextRect.Left() = rBox.m_aRect.Left() + aCheckRect.GetWidth()+5*nDelta; 5275 aTextRect.Top() = rBox.m_aRect.Top(); 5276 aTextRect.Right() = aTextRect.Left() + rBox.m_aRect.GetWidth() - aCheckRect.GetWidth()-6*nDelta; 5277 aTextRect.Bottom() = rBox.m_aRect.Bottom(); 5278 } 5279 else 5280 { 5281 aCheckRect.Left() = rBox.m_aRect.Right() - nDelta - aFontSize.Height(); 5282 aCheckRect.Top() = rBox.m_aRect.Top() + (rBox.m_aRect.GetHeight()-aFontSize.Height())/2; 5283 aCheckRect.Right() = aCheckRect.Left() + aFontSize.Height(); 5284 aCheckRect.Bottom() = aCheckRect.Top() + aFontSize.Height(); 5285 5286 // #i74206# handle small controls without text area 5287 while( aCheckRect.GetWidth() > rBox.m_aRect.GetWidth() && aCheckRect.GetWidth() > nDelta ) 5288 { 5289 aCheckRect.Left() += nDelta; 5290 aCheckRect.Top() += nDelta/2; 5291 aCheckRect.Bottom() -= nDelta - (nDelta/2); 5292 } 5293 5294 aTextRect.Left() = rBox.m_aRect.Left(); 5295 aTextRect.Top() = rBox.m_aRect.Top(); 5296 aTextRect.Right() = aTextRect.Left() + rBox.m_aRect.GetWidth() - aCheckRect.GetWidth()-6*nDelta; 5297 aTextRect.Bottom() = rBox.m_aRect.Bottom(); 5298 } 5299 setLineColor( Color( COL_BLACK ) ); 5300 setFillColor( Color( COL_TRANSPARENT ) ); 5301 OStringBuffer aLW( 32 ); 5302 aLW.append( "q " ); 5303 m_aPages[ m_nCurrentPage ].appendMappedLength( nDelta, aLW ); 5304 aLW.append( " w " ); 5305 writeBuffer( aLW.getStr(), aLW.getLength() ); 5306 drawEllipse( aCheckRect ); 5307 writeBuffer( " Q\n", 3 ); 5308 setTextColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ) ); 5309 drawText( aTextRect, rBox.m_aText, rBox.m_nTextStyle ); 5310 5311 pop(); 5312 5313 OStringBuffer aDA( 256 ); 5314 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ), aDA ); 5315 sal_Int32 nBest = getBestBuiltinFont( Font( String( RTL_CONSTASCII_USTRINGPARAM( "ZapfDingbats" ) ), aFont.GetSize() ) ); 5316 aDA.append( ' ' ); 5317 aDA.append( m_aBuiltinFonts[nBest].getNameObject() ); 5318 aDA.append( " 0 Tf" ); 5319 rBox.m_aDAString = aDA.makeStringAndClear(); 5320 //to encrypt this (el) 5321 rBox.m_aMKDict = "/CA"; 5322 //after this assignement, to m_aMKDic cannot be added anything 5323 rBox.m_aMKDictCAString = "l"; 5324 5325 rBox.m_aRect = aCheckRect; 5326 5327 // create appearance streams 5328 push( sal::static_int_cast<sal_uInt16>(~0U) ); 5329 SvMemoryStream* pCheckStream = new SvMemoryStream( 256, 256 ); 5330 5331 beginRedirect( pCheckStream, aCheckRect ); 5332 aDA.append( "/Tx BMC\nq BT\n" ); 5333 appendNonStrokingColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ), aDA ); 5334 aDA.append( ' ' ); 5335 aDA.append( m_aBuiltinFonts[nBest].getNameObject() ); 5336 aDA.append( ' ' ); 5337 m_aPages[m_nCurrentPage].appendMappedLength( sal_Int32( aCheckRect.GetHeight() ), aDA ); 5338 aDA.append( " Tf\n0 0 Td\nET\nQ\n" ); 5339 writeBuffer( aDA.getStr(), aDA.getLength() ); 5340 setFillColor( replaceColor( rWidget.TextColor, rSettings.GetRadioCheckTextColor() ) ); 5341 setLineColor( Color( COL_TRANSPARENT ) ); 5342 aCheckRect.Left() += 3*nDelta; 5343 aCheckRect.Top() += 3*nDelta; 5344 aCheckRect.Bottom() -= 3*nDelta; 5345 aCheckRect.Right() -= 3*nDelta; 5346 drawEllipse( aCheckRect ); 5347 writeBuffer( "\nEMC\n", 5 ); 5348 endRedirect(); 5349 5350 pop(); 5351 rBox.m_aAppearances[ "N" ][ "Yes" ] = pCheckStream; 5352 5353 SvMemoryStream* pUncheckStream = new SvMemoryStream( 256, 256 ); 5354 beginRedirect( pUncheckStream, aCheckRect ); 5355 writeBuffer( "/Tx BMC\nEMC\n", 12 ); 5356 endRedirect(); 5357 rBox.m_aAppearances[ "N" ][ "Off" ] = pUncheckStream; 5358 } 5359 5360 bool PDFWriterImpl::emitAppearances( PDFWidget& rWidget, OStringBuffer& rAnnotDict ) 5361 { 5362 5363 // TODO: check and insert default streams 5364 rtl::OString aStandardAppearance; 5365 switch( rWidget.m_eType ) 5366 { 5367 case PDFWriter::CheckBox: 5368 aStandardAppearance = OUStringToOString( rWidget.m_aValue, RTL_TEXTENCODING_ASCII_US ); 5369 break; 5370 default: 5371 break; 5372 } 5373 5374 if( rWidget.m_aAppearances.size() ) 5375 { 5376 rAnnotDict.append( "/AP<<\n" ); 5377 for( PDFAppearanceMap::iterator dict_it = rWidget.m_aAppearances.begin(); dict_it != rWidget.m_aAppearances.end(); ++dict_it ) 5378 { 5379 rAnnotDict.append( "/" ); 5380 rAnnotDict.append( dict_it->first ); 5381 bool bUseSubDict = (dict_it->second.size() > 1); 5382 rAnnotDict.append( bUseSubDict ? "<<" : " " ); 5383 5384 for( PDFAppearanceStreams::const_iterator stream_it = dict_it->second.begin(); 5385 stream_it != dict_it->second.end(); ++stream_it ) 5386 { 5387 SvMemoryStream* pApppearanceStream = stream_it->second; 5388 dict_it->second[ stream_it->first ] = NULL; 5389 5390 bool bDeflate = compressStream( pApppearanceStream ); 5391 5392 pApppearanceStream->Seek( STREAM_SEEK_TO_END ); 5393 sal_Int64 nStreamLen = pApppearanceStream->Tell(); 5394 pApppearanceStream->Seek( STREAM_SEEK_TO_BEGIN ); 5395 sal_Int32 nObject = createObject(); 5396 CHECK_RETURN( updateObject( nObject ) ); 5397 #if OSL_DEBUG_LEVEL > 1 5398 emitComment( "PDFWriterImpl::emitAppearances" ); 5399 #endif 5400 OStringBuffer aLine; 5401 aLine.append( nObject ); 5402 5403 aLine.append( " 0 obj\n" 5404 "<</Type/XObject\n" 5405 "/Subtype/Form\n" 5406 "/BBox[0 0 " ); 5407 appendFixedInt( rWidget.m_aRect.GetWidth()-1, aLine ); 5408 aLine.append( " " ); 5409 appendFixedInt( rWidget.m_aRect.GetHeight()-1, aLine ); 5410 aLine.append( "]\n" 5411 "/Resources " ); 5412 aLine.append( getResourceDictObj() ); 5413 aLine.append( " 0 R\n" 5414 "/Length " ); 5415 aLine.append( nStreamLen ); 5416 aLine.append( "\n" ); 5417 if( bDeflate ) 5418 aLine.append( "/Filter/FlateDecode\n" ); 5419 aLine.append( ">>\nstream\n" ); 5420 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 5421 checkAndEnableStreamEncryption( nObject ); 5422 CHECK_RETURN( writeBuffer( pApppearanceStream->GetData(), nStreamLen ) ); 5423 disableStreamEncryption(); 5424 CHECK_RETURN( writeBuffer( "\nendstream\nendobj\n\n", 19 ) ); 5425 5426 if( bUseSubDict ) 5427 { 5428 rAnnotDict.append( " /" ); 5429 rAnnotDict.append( stream_it->first ); 5430 rAnnotDict.append( " " ); 5431 } 5432 rAnnotDict.append( nObject ); 5433 rAnnotDict.append( " 0 R" ); 5434 5435 delete pApppearanceStream; 5436 } 5437 5438 rAnnotDict.append( bUseSubDict ? ">>\n" : "\n" ); 5439 } 5440 rAnnotDict.append( ">>\n" ); 5441 if( aStandardAppearance.getLength() ) 5442 { 5443 rAnnotDict.append( "/AS /" ); 5444 rAnnotDict.append( aStandardAppearance ); 5445 rAnnotDict.append( "\n" ); 5446 } 5447 } 5448 5449 return true; 5450 } 5451 5452 bool PDFWriterImpl::emitWidgetAnnotations() 5453 { 5454 ensureUniqueRadioOnValues(); 5455 5456 int nAnnots = m_aWidgets.size(); 5457 for( int a = 0; a < nAnnots; a++ ) 5458 { 5459 PDFWidget& rWidget = m_aWidgets[a]; 5460 5461 OStringBuffer aLine( 1024 ); 5462 OStringBuffer aValue( 256 ); 5463 aLine.append( rWidget.m_nObject ); 5464 aLine.append( " 0 obj\n" 5465 "<<" ); 5466 if( rWidget.m_eType != PDFWriter::Hierarchy ) 5467 { 5468 // emit widget annotation only for terminal fields 5469 if( rWidget.m_aKids.empty() ) 5470 { 5471 aLine.append( "/Type/Annot/Subtype/Widget/F 4\n" 5472 "/Rect[" ); 5473 appendFixedInt( rWidget.m_aRect.Left()-1, aLine ); 5474 aLine.append( ' ' ); 5475 appendFixedInt( rWidget.m_aRect.Top()+1, aLine ); 5476 aLine.append( ' ' ); 5477 appendFixedInt( rWidget.m_aRect.Right()+1, aLine ); 5478 aLine.append( ' ' ); 5479 appendFixedInt( rWidget.m_aRect.Bottom()-1, aLine ); 5480 aLine.append( "]\n" ); 5481 } 5482 aLine.append( "/FT/" ); 5483 switch( rWidget.m_eType ) 5484 { 5485 case PDFWriter::RadioButton: 5486 case PDFWriter::CheckBox: 5487 // for radio buttons only the RadioButton field, not the 5488 // CheckBox children should have a value, else acrobat reader 5489 // does not always check the right button 5490 // of course real check boxes (not belonging to a readio group) 5491 // need their values, too 5492 if( rWidget.m_eType == PDFWriter::RadioButton || rWidget.m_nRadioGroup < 0 ) 5493 { 5494 aValue.append( "/" ); 5495 // check for radio group with all buttons unpressed 5496 if( rWidget.m_aValue.getLength() == 0 ) 5497 aValue.append( "Off" ); 5498 else 5499 appendName( rWidget.m_aValue, aValue ); 5500 } 5501 case PDFWriter::PushButton: 5502 aLine.append( "Btn" ); 5503 break; 5504 case PDFWriter::ListBox: 5505 if( rWidget.m_nFlags & 0x200000 ) // multiselect 5506 { 5507 aValue.append( "[" ); 5508 for( unsigned int i = 0; i < rWidget.m_aSelectedEntries.size(); i++ ) 5509 { 5510 sal_Int32 nEntry = rWidget.m_aSelectedEntries[i]; 5511 if( nEntry >= 0 && nEntry < sal_Int32(rWidget.m_aListEntries.size()) ) 5512 appendUnicodeTextStringEncrypt( rWidget.m_aListEntries[ nEntry ], rWidget.m_nObject, aValue ); 5513 } 5514 aValue.append( "]" ); 5515 } 5516 else if( rWidget.m_aSelectedEntries.size() > 0 && 5517 rWidget.m_aSelectedEntries[0] >= 0 && 5518 rWidget.m_aSelectedEntries[0] < sal_Int32(rWidget.m_aListEntries.size()) ) 5519 { 5520 appendUnicodeTextStringEncrypt( rWidget.m_aListEntries[ rWidget.m_aSelectedEntries[0] ], rWidget.m_nObject, aValue ); 5521 } 5522 else 5523 appendUnicodeTextStringEncrypt( rtl::OUString(), rWidget.m_nObject, aValue ); 5524 aLine.append( "Ch" ); 5525 break; 5526 case PDFWriter::ComboBox: 5527 appendUnicodeTextStringEncrypt( rWidget.m_aValue, rWidget.m_nObject, aValue ); 5528 aLine.append( "Ch" ); 5529 break; 5530 case PDFWriter::Edit: 5531 aLine.append( "Tx" ); 5532 appendUnicodeTextStringEncrypt( rWidget.m_aValue, rWidget.m_nObject, aValue ); 5533 break; 5534 case PDFWriter::Hierarchy: // make the compiler happy 5535 break; 5536 } 5537 aLine.append( "\n" ); 5538 aLine.append( "/P " ); 5539 aLine.append( m_aPages[ rWidget.m_nPage ].m_nPageObject ); 5540 aLine.append( " 0 R\n" ); 5541 } 5542 if( rWidget.m_nParent ) 5543 { 5544 aLine.append( "/Parent " ); 5545 aLine.append( rWidget.m_nParent ); 5546 aLine.append( " 0 R\n" ); 5547 } 5548 if( rWidget.m_aKids.size() ) 5549 { 5550 aLine.append( "/Kids[" ); 5551 for( unsigned int i = 0; i < rWidget.m_aKids.size(); i++ ) 5552 { 5553 aLine.append( rWidget.m_aKids[i] ); 5554 aLine.append( " 0 R" ); 5555 aLine.append( ( (i&15) == 15 ) ? "\n" : " " ); 5556 } 5557 aLine.append( "]\n" ); 5558 } 5559 if( rWidget.m_aName.getLength() ) 5560 { 5561 aLine.append( "/T" ); 5562 appendLiteralStringEncrypt( rWidget.m_aName, rWidget.m_nObject, aLine ); 5563 aLine.append( "\n" ); 5564 } 5565 if( m_aContext.Version > PDFWriter::PDF_1_2 && rWidget.m_aDescription.getLength() ) 5566 { 5567 // the alternate field name should be unicode able since it is 5568 // supposed to be used in UI 5569 aLine.append( "/TU" ); 5570 appendUnicodeTextStringEncrypt( rWidget.m_aDescription, rWidget.m_nObject, aLine ); 5571 aLine.append( "\n" ); 5572 } 5573 5574 if( rWidget.m_nFlags ) 5575 { 5576 aLine.append( "/Ff " ); 5577 aLine.append( rWidget.m_nFlags ); 5578 aLine.append( "\n" ); 5579 } 5580 if( aValue.getLength() ) 5581 { 5582 OString aVal = aValue.makeStringAndClear(); 5583 aLine.append( "/V " ); 5584 aLine.append( aVal ); 5585 aLine.append( "\n" 5586 "/DV " ); 5587 aLine.append( aVal ); 5588 aLine.append( "\n" ); 5589 } 5590 if( rWidget.m_eType == PDFWriter::ListBox || rWidget.m_eType == PDFWriter::ComboBox ) 5591 { 5592 sal_Int32 nTI = -1; 5593 aLine.append( "/Opt[\n" ); 5594 sal_Int32 i = 0; 5595 for( std::vector< OUString >::const_iterator it = rWidget.m_aListEntries.begin(); it != rWidget.m_aListEntries.end(); ++it, ++i ) 5596 { 5597 appendUnicodeTextStringEncrypt( *it, rWidget.m_nObject, aLine ); 5598 aLine.append( "\n" ); 5599 if( *it == rWidget.m_aValue ) 5600 nTI = i; 5601 } 5602 aLine.append( "]\n" ); 5603 if( nTI > 0 ) 5604 { 5605 aLine.append( "/TI " ); 5606 aLine.append( nTI ); 5607 aLine.append( "\n" ); 5608 if( rWidget.m_nFlags & 0x200000 ) // Multiselect 5609 { 5610 aLine.append( "/I [" ); 5611 aLine.append( nTI ); 5612 aLine.append( "]\n" ); 5613 } 5614 } 5615 } 5616 if( rWidget.m_eType == PDFWriter::Edit && rWidget.m_nMaxLen > 0 ) 5617 { 5618 aLine.append( "/MaxLen " ); 5619 aLine.append( rWidget.m_nMaxLen ); 5620 aLine.append( "\n" ); 5621 } 5622 if( rWidget.m_eType == PDFWriter::PushButton ) 5623 { 5624 if(!m_bIsPDF_A1) 5625 { 5626 OStringBuffer aDest; 5627 if( rWidget.m_nDest != -1 && appendDest( m_aDestinationIdTranslation[ rWidget.m_nDest ], aDest ) ) 5628 { 5629 aLine.append( "/AA<</D<</Type/Action/S/GoTo/D " ); 5630 aLine.append( aDest.makeStringAndClear() ); 5631 aLine.append( ">>>>\n" ); 5632 } 5633 else if( rWidget.m_aListEntries.empty() ) 5634 { 5635 // create a reset form action 5636 aLine.append( "/AA<</D<</Type/Action/S/ResetForm>>>>\n" ); 5637 } 5638 else if( rWidget.m_bSubmit ) 5639 { 5640 // create a submit form action 5641 aLine.append( "/AA<</D<</Type/Action/S/SubmitForm/F" ); 5642 appendLiteralStringEncrypt( rWidget.m_aListEntries.front(), rWidget.m_nObject, aLine, osl_getThreadTextEncoding() ); 5643 aLine.append( "/Flags " ); 5644 5645 sal_Int32 nFlags = 0; 5646 switch( m_aContext.SubmitFormat ) 5647 { 5648 case PDFWriter::HTML: 5649 nFlags |= 4; 5650 break; 5651 case PDFWriter::XML: 5652 if( m_aContext.Version > PDFWriter::PDF_1_3 ) 5653 nFlags |= 32; 5654 break; 5655 case PDFWriter::PDF: 5656 if( m_aContext.Version > PDFWriter::PDF_1_3 ) 5657 nFlags |= 256; 5658 break; 5659 case PDFWriter::FDF: 5660 default: 5661 break; 5662 } 5663 if( rWidget.m_bSubmitGet ) 5664 nFlags |= 8; 5665 aLine.append( nFlags ); 5666 aLine.append( ">>>>\n" ); 5667 } 5668 else 5669 { 5670 // create a URI action 5671 aLine.append( "/AA<</D<</Type/Action/S/URI/URI(" ); 5672 aLine.append( OUStringToOString( rWidget.m_aListEntries.front(), RTL_TEXTENCODING_ASCII_US ) ); 5673 aLine.append( ")>>>>\n" ); 5674 } 5675 } 5676 else 5677 m_aErrors.insert( PDFWriter::Warning_FormAction_Omitted_PDFA ); 5678 } 5679 if( rWidget.m_aDAString.getLength() ) 5680 { 5681 if( rWidget.m_aDRDict.getLength() ) 5682 { 5683 aLine.append( "/DR<<" ); 5684 aLine.append( rWidget.m_aDRDict ); 5685 aLine.append( ">>\n" ); 5686 } 5687 else 5688 { 5689 aLine.append( "/DR<</Font<<" ); 5690 appendBuiltinFontsToDict( aLine ); 5691 aLine.append( ">>>>\n" ); 5692 } 5693 aLine.append( "/DA" ); 5694 appendLiteralStringEncrypt( rWidget.m_aDAString, rWidget.m_nObject, aLine ); 5695 aLine.append( "\n" ); 5696 if( rWidget.m_nTextStyle & TEXT_DRAW_CENTER ) 5697 aLine.append( "/Q 1\n" ); 5698 else if( rWidget.m_nTextStyle & TEXT_DRAW_RIGHT ) 5699 aLine.append( "/Q 2\n" ); 5700 } 5701 // appearance charactristics for terminal fields 5702 // which are supposed to have an appearance constructed 5703 // by the viewer application 5704 if( rWidget.m_aMKDict.getLength() ) 5705 { 5706 aLine.append( "/MK<<" ); 5707 aLine.append( rWidget.m_aMKDict ); 5708 //add the CA string, encrypting it 5709 appendLiteralStringEncrypt(rWidget.m_aMKDictCAString, rWidget.m_nObject, aLine); 5710 aLine.append( ">>\n" ); 5711 } 5712 5713 CHECK_RETURN( emitAppearances( rWidget, aLine ) ); 5714 5715 aLine.append( ">>\n" 5716 "endobj\n\n" ); 5717 CHECK_RETURN( updateObject( rWidget.m_nObject ) ); 5718 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 5719 } 5720 return true; 5721 } 5722 5723 bool PDFWriterImpl::emitAnnotations() 5724 { 5725 if( m_aPages.size() < 1 ) 5726 return false; 5727 5728 CHECK_RETURN( emitLinkAnnotations() ); 5729 5730 CHECK_RETURN( emitNoteAnnotations() ); 5731 5732 CHECK_RETURN( emitWidgetAnnotations() ); 5733 5734 return true; 5735 } 5736 5737 #undef CHECK_RETURN 5738 #define CHECK_RETURN( x ) if( !x ) return false 5739 5740 bool PDFWriterImpl::emitCatalog() 5741 { 5742 // build page tree 5743 // currently there is only one node that contains all leaves 5744 5745 // first create a page tree node id 5746 sal_Int32 nTreeNode = createObject(); 5747 5748 // emit global resource dictionary (page emit needs it) 5749 CHECK_RETURN( emitResources() ); 5750 5751 // emit all pages 5752 for( std::vector<PDFPage>::iterator it = m_aPages.begin(); it != m_aPages.end(); ++it ) 5753 if( ! it->emit( nTreeNode ) ) 5754 return false; 5755 5756 sal_Int32 nNamedDestinationsDictionary = emitNamedDestinations(); 5757 5758 sal_Int32 nOutlineDict = emitOutline(); 5759 5760 //emit Output intent i59651 5761 sal_Int32 nOutputIntentObject = emitOutputIntent(); 5762 5763 //emit metadata 5764 sal_Int32 nMetadataObject = emitDocumentMetadata(); 5765 5766 sal_Int32 nStructureDict = 0; 5767 if(m_aStructure.size() > 1) 5768 { 5769 ///check if dummy structure containers are needed 5770 addInternalStructureContainer(m_aStructure[0]); 5771 nStructureDict = m_aStructure[0].m_nObject = createObject(); 5772 emitStructure( m_aStructure[ 0 ] ); 5773 } 5774 5775 // adjust tree node file offset 5776 if( ! updateObject( nTreeNode ) ) 5777 return false; 5778 5779 // emit tree node 5780 OStringBuffer aLine( 2048 ); 5781 aLine.append( nTreeNode ); 5782 aLine.append( " 0 obj\n" ); 5783 aLine.append( "<</Type/Pages\n" ); 5784 aLine.append( "/Resources " ); 5785 aLine.append( getResourceDictObj() ); 5786 aLine.append( " 0 R\n" ); 5787 5788 switch( m_eInheritedOrientation ) 5789 { 5790 case PDFWriter::Landscape: aLine.append( "/Rotate 90\n" );break; 5791 case PDFWriter::Seascape: aLine.append( "/Rotate -90\n" );break; 5792 5793 case PDFWriter::Inherit: // actually Inherit would be a bug, but insignificant 5794 case PDFWriter::Portrait: 5795 default: 5796 break; 5797 } 5798 sal_Int32 nMediaBoxWidth = 0; 5799 sal_Int32 nMediaBoxHeight = 0; 5800 if( m_aPages.empty() ) // sanity check, this should not happen 5801 { 5802 nMediaBoxWidth = m_nInheritedPageWidth; 5803 nMediaBoxHeight = m_nInheritedPageHeight; 5804 } 5805 else 5806 { 5807 for( std::vector<PDFPage>::const_iterator iter = m_aPages.begin(); iter != m_aPages.end(); ++iter ) 5808 { 5809 if( iter->m_nPageWidth > nMediaBoxWidth ) 5810 nMediaBoxWidth = iter->m_nPageWidth; 5811 if( iter->m_nPageHeight > nMediaBoxHeight ) 5812 nMediaBoxHeight = iter->m_nPageHeight; 5813 } 5814 } 5815 aLine.append( "/MediaBox[ 0 0 " ); 5816 aLine.append( nMediaBoxWidth ); 5817 aLine.append( ' ' ); 5818 aLine.append( nMediaBoxHeight ); 5819 aLine.append( " ]\n" 5820 "/Kids[ " ); 5821 unsigned int i = 0; 5822 for( std::vector<PDFPage>::const_iterator iter = m_aPages.begin(); iter != m_aPages.end(); ++iter, i++ ) 5823 { 5824 aLine.append( iter->m_nPageObject ); 5825 aLine.append( " 0 R" ); 5826 aLine.append( ( (i&15) == 15 ) ? "\n" : " " ); 5827 } 5828 aLine.append( "]\n" 5829 "/Count " ); 5830 aLine.append( (sal_Int32)m_aPages.size() ); 5831 aLine.append( ">>\n" 5832 "endobj\n\n" ); 5833 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 5834 5835 // emit annotation objects 5836 CHECK_RETURN( emitAnnotations() ); 5837 5838 // emit Catalog 5839 m_nCatalogObject = createObject(); 5840 if( ! updateObject( m_nCatalogObject ) ) 5841 return false; 5842 aLine.setLength( 0 ); 5843 aLine.append( m_nCatalogObject ); 5844 aLine.append( " 0 obj\n" 5845 "<</Type/Catalog/Pages " ); 5846 aLine.append( nTreeNode ); 5847 aLine.append( " 0 R\n" ); 5848 //--->i56629 5849 //check if there are named destinations to emit (root must be inside the catalog) 5850 if( nNamedDestinationsDictionary ) 5851 { 5852 aLine.append("/Dests "); 5853 aLine.append( nNamedDestinationsDictionary ); 5854 aLine.append( " 0 R\n" ); 5855 } 5856 //<---- 5857 if( m_aContext.PageLayout != PDFWriter::DefaultLayout ) 5858 switch( m_aContext.PageLayout ) 5859 { 5860 default : 5861 case PDFWriter::SinglePage : 5862 aLine.append( "/PageLayout/SinglePage\n" ); 5863 break; 5864 case PDFWriter::Continuous : 5865 aLine.append( "/PageLayout/OneColumn\n" ); 5866 break; 5867 case PDFWriter::ContinuousFacing : 5868 //the flag m_aContext.FirstPageLeft below is used to set the page on the left side 5869 aLine.append( "/PageLayout/TwoColumnRight\n" );//odd page on the right side 5870 break; 5871 } 5872 if( m_aContext.PDFDocumentMode != PDFWriter::ModeDefault && !m_aContext.OpenInFullScreenMode ) 5873 switch( m_aContext.PDFDocumentMode ) 5874 { 5875 default : 5876 aLine.append( "/PageMode/UseNone\n" ); 5877 break; 5878 case PDFWriter::UseOutlines : 5879 aLine.append( "/PageMode/UseOutlines\n" ); //document is opened with outline pane open 5880 break; 5881 case PDFWriter::UseThumbs : 5882 aLine.append( "/PageMode/UseThumbs\n" ); //document is opened with thumbnails pane open 5883 break; 5884 } 5885 else if( m_aContext.OpenInFullScreenMode ) 5886 aLine.append( "/PageMode/FullScreen\n" ); //document is opened full screen 5887 5888 OStringBuffer aInitPageRef; 5889 if( m_aContext.InitialPage >= 0 && m_aContext.InitialPage < (sal_Int32)m_aPages.size() ) 5890 { 5891 aInitPageRef.append( m_aPages[m_aContext.InitialPage].m_nPageObject ); 5892 aInitPageRef.append( " 0 R" ); 5893 } 5894 else 5895 aInitPageRef.append( "0" ); 5896 switch( m_aContext.PDFDocumentAction ) 5897 { 5898 case PDFWriter::ActionDefault : //do nothing, this is the Acrobat default 5899 default: 5900 if( aInitPageRef.getLength() > 1 ) 5901 { 5902 aLine.append( "/OpenAction[" ); 5903 aLine.append( aInitPageRef ); 5904 aLine.append( " /XYZ null null 0]\n" ); 5905 } 5906 break; 5907 case PDFWriter::FitInWindow : 5908 aLine.append( "/OpenAction[" ); 5909 aLine.append( aInitPageRef ); 5910 aLine.append( " /Fit]\n" ); //Open fit page 5911 break; 5912 case PDFWriter::FitWidth : 5913 aLine.append( "/OpenAction[" ); 5914 aLine.append( aInitPageRef ); 5915 aLine.append( " /FitH " ); 5916 aLine.append( m_nInheritedPageHeight );//Open fit width 5917 aLine.append( "]\n" ); 5918 break; 5919 case PDFWriter::FitVisible : 5920 aLine.append( "/OpenAction[" ); 5921 aLine.append( aInitPageRef ); 5922 aLine.append( " /FitBH " ); 5923 aLine.append( m_nInheritedPageHeight );//Open fit visible 5924 aLine.append( "]\n" ); 5925 break; 5926 case PDFWriter::ActionZoom : 5927 aLine.append( "/OpenAction[" ); 5928 aLine.append( aInitPageRef ); 5929 aLine.append( " /XYZ null null " ); 5930 if( m_aContext.Zoom >= 50 && m_aContext.Zoom <= 1600 ) 5931 aLine.append( (double)m_aContext.Zoom/100.0 ); 5932 else 5933 aLine.append( "0" ); 5934 aLine.append( "]\n" ); 5935 break; 5936 } 5937 // viewer preferences, if we had some, then emit 5938 if( m_aContext.HideViewerToolbar || 5939 ( m_aContext.Version > PDFWriter::PDF_1_3 && m_aContext.DocumentInfo.Title.Len() && m_aContext.DisplayPDFDocumentTitle ) || 5940 m_aContext.HideViewerMenubar || 5941 m_aContext.HideViewerWindowControls || m_aContext.FitWindow || 5942 m_aContext.CenterWindow || (m_aContext.FirstPageLeft && m_aContext.PageLayout == PDFWriter::ContinuousFacing ) || 5943 m_aContext.OpenInFullScreenMode ) 5944 { 5945 aLine.append( "/ViewerPreferences<<" ); 5946 if( m_aContext.HideViewerToolbar ) 5947 aLine.append( "/HideToolbar true\n" ); 5948 if( m_aContext.HideViewerMenubar ) 5949 aLine.append( "/HideMenubar true\n" ); 5950 if( m_aContext.HideViewerWindowControls ) 5951 aLine.append( "/HideWindowUI true\n" ); 5952 if( m_aContext.FitWindow ) 5953 aLine.append( "/FitWindow true\n" ); 5954 if( m_aContext.CenterWindow ) 5955 aLine.append( "/CenterWindow true\n" ); 5956 if( m_aContext.Version > PDFWriter::PDF_1_3 && m_aContext.DocumentInfo.Title.Len() && m_aContext.DisplayPDFDocumentTitle ) 5957 aLine.append( "/DisplayDocTitle true\n" ); 5958 if( m_aContext.FirstPageLeft && m_aContext.PageLayout == PDFWriter::ContinuousFacing ) 5959 aLine.append( "/Direction/R2L\n" ); 5960 if( m_aContext.OpenInFullScreenMode ) 5961 switch( m_aContext.PDFDocumentMode ) 5962 { 5963 default : 5964 case PDFWriter::ModeDefault : 5965 aLine.append( "/NonFullScreenPageMode/UseNone\n" ); 5966 break; 5967 case PDFWriter::UseOutlines : 5968 aLine.append( "/NonFullScreenPageMode/UseOutlines\n" ); 5969 break; 5970 case PDFWriter::UseThumbs : 5971 aLine.append( "/NonFullScreenPageMode/UseThumbs\n" ); 5972 break; 5973 } 5974 aLine.append( ">>\n" ); 5975 } 5976 5977 if( nOutlineDict ) 5978 { 5979 aLine.append( "/Outlines " ); 5980 aLine.append( nOutlineDict ); 5981 aLine.append( " 0 R\n" ); 5982 } 5983 if( nStructureDict ) 5984 { 5985 aLine.append( "/StructTreeRoot " ); 5986 aLine.append( nStructureDict ); 5987 aLine.append( " 0 R\n" ); 5988 } 5989 if( m_aContext.DocumentLocale.Language.getLength() > 0 ) 5990 { 5991 OUStringBuffer aLocBuf( 16 ); 5992 aLocBuf.append( m_aContext.DocumentLocale.Language.toAsciiLowerCase() ); 5993 if( m_aContext.DocumentLocale.Country.getLength() > 0 ) 5994 { 5995 aLocBuf.append( sal_Unicode('-') ); 5996 aLocBuf.append( m_aContext.DocumentLocale.Country ); 5997 } 5998 aLine.append( "/Lang" ); 5999 appendLiteralStringEncrypt( aLocBuf.makeStringAndClear(), m_nCatalogObject, aLine ); 6000 aLine.append( "\n" ); 6001 } 6002 if( m_aContext.Tagged && m_aContext.Version > PDFWriter::PDF_1_3 ) 6003 { 6004 aLine.append( "/MarkInfo<</Marked true>>\n" ); 6005 } 6006 if( m_aWidgets.size() > 0 ) 6007 { 6008 aLine.append( "/AcroForm<</Fields[\n" ); 6009 int nWidgets = m_aWidgets.size(); 6010 int nOut = 0; 6011 for( int j = 0; j < nWidgets; j++ ) 6012 { 6013 // output only root fields 6014 if( m_aWidgets[j].m_nParent < 1 ) 6015 { 6016 aLine.append( m_aWidgets[j].m_nObject ); 6017 aLine.append( (nOut++ % 5)==4 ? " 0 R\n" : " 0 R " ); 6018 } 6019 } 6020 aLine.append( "\n]/DR " ); 6021 aLine.append( getResourceDictObj() ); 6022 aLine.append( " 0 R" ); 6023 if( m_bIsPDF_A1 ) 6024 aLine.append( ">>\n" ); 6025 else 6026 aLine.append( "/NeedAppearances true>>\n" ); 6027 } 6028 //--->i59651 6029 //check if there is a Metadata object 6030 if( nOutputIntentObject ) 6031 { 6032 aLine.append("/OutputIntents["); 6033 aLine.append( nOutputIntentObject ); 6034 aLine.append( " 0 R]" ); 6035 } 6036 if( nMetadataObject ) 6037 { 6038 aLine.append("/Metadata "); 6039 aLine.append( nMetadataObject ); 6040 aLine.append( " 0 R" ); 6041 } 6042 //<---- 6043 aLine.append( ">>\n" 6044 "endobj\n\n" ); 6045 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6046 6047 return true; 6048 } 6049 6050 sal_Int32 PDFWriterImpl::emitInfoDict( ) 6051 { 6052 sal_Int32 nObject = createObject(); 6053 6054 if( updateObject( nObject ) ) 6055 { 6056 OStringBuffer aLine( 1024 ); 6057 aLine.append( nObject ); 6058 aLine.append( " 0 obj\n" 6059 "<<" ); 6060 if( m_aContext.DocumentInfo.Title.Len() ) 6061 { 6062 aLine.append( "/Title" ); 6063 appendUnicodeTextStringEncrypt( m_aContext.DocumentInfo.Title, nObject, aLine ); 6064 aLine.append( "\n" ); 6065 } 6066 if( m_aContext.DocumentInfo.Author.Len() ) 6067 { 6068 aLine.append( "/Author" ); 6069 appendUnicodeTextStringEncrypt( m_aContext.DocumentInfo.Author, nObject, aLine ); 6070 aLine.append( "\n" ); 6071 } 6072 if( m_aContext.DocumentInfo.Subject.Len() ) 6073 { 6074 aLine.append( "/Subject" ); 6075 appendUnicodeTextStringEncrypt( m_aContext.DocumentInfo.Subject, nObject, aLine ); 6076 aLine.append( "\n" ); 6077 } 6078 if( m_aContext.DocumentInfo.Keywords.Len() ) 6079 { 6080 aLine.append( "/Keywords" ); 6081 appendUnicodeTextStringEncrypt( m_aContext.DocumentInfo.Keywords, nObject, aLine ); 6082 aLine.append( "\n" ); 6083 } 6084 if( m_aContext.DocumentInfo.Creator.Len() ) 6085 { 6086 aLine.append( "/Creator" ); 6087 appendUnicodeTextStringEncrypt( m_aContext.DocumentInfo.Creator, nObject, aLine ); 6088 aLine.append( "\n" ); 6089 } 6090 if( m_aContext.DocumentInfo.Producer.Len() ) 6091 { 6092 aLine.append( "/Producer" ); 6093 appendUnicodeTextStringEncrypt( m_aContext.DocumentInfo.Producer, nObject, aLine ); 6094 aLine.append( "\n" ); 6095 } 6096 6097 aLine.append( "/CreationDate" ); 6098 appendLiteralStringEncrypt( m_aCreationDateString, nObject, aLine ); 6099 aLine.append( ">>\nendobj\n\n" ); 6100 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 6101 nObject = 0; 6102 } 6103 else 6104 nObject = 0; 6105 6106 return nObject; 6107 } 6108 6109 //--->i56629 6110 // Part of this function may be shared with method appendDest. 6111 // 6112 sal_Int32 PDFWriterImpl::emitNamedDestinations() 6113 { 6114 sal_Int32 nCount = m_aNamedDests.size(); 6115 if( nCount <= 0 ) 6116 return 0;//define internal error 6117 6118 //get the object number for all the destinations 6119 sal_Int32 nObject = createObject(); 6120 6121 if( updateObject( nObject ) ) 6122 { 6123 //emit the dictionary 6124 OStringBuffer aLine( 1024 ); 6125 aLine.append( nObject ); 6126 aLine.append( " 0 obj\n" 6127 "<<" ); 6128 6129 sal_Int32 nDestID; 6130 for( nDestID = 0; nDestID < nCount; nDestID++ ) 6131 { 6132 const PDFNamedDest& rDest = m_aNamedDests[ nDestID ]; 6133 // In order to correctly function both under an Internet browser and 6134 // directly with a reader (provided the reader has the feature) we 6135 // need to set the name of the destination the same way it will be encoded 6136 // in an Internet link 6137 INetURLObject aLocalURL( 6138 OUString( RTL_CONSTASCII_USTRINGPARAM( "http://ahost.ax" ) ) ); //dummy location, won't be used 6139 aLocalURL.SetMark( rDest.m_aDestName ); 6140 6141 const rtl::OUString aName = aLocalURL.GetMark( INetURLObject::NO_DECODE ); //same coding as 6142 // in link creation ( see PDFWriterImpl::emitLinkAnnotations ) 6143 const PDFPage& rDestPage = m_aPages[ rDest.m_nPage ]; 6144 6145 aLine.append( '/' ); 6146 appendDestinationName( aName, aLine ); // this conversion must be done when forming the link to target ( see in emitCatalog ) 6147 aLine.append( '[' ); // the '[' can be emitted immediately, because the appendDestinationName function 6148 //maps the preceeding character properly 6149 aLine.append( rDestPage.m_nPageObject ); 6150 aLine.append( " 0 R" ); 6151 6152 switch( rDest.m_eType ) 6153 { 6154 case PDFWriter::XYZ: 6155 default: 6156 aLine.append( "/XYZ " ); 6157 appendFixedInt( rDest.m_aRect.Left(), aLine ); 6158 aLine.append( ' ' ); 6159 appendFixedInt( rDest.m_aRect.Bottom(), aLine ); 6160 aLine.append( " 0" ); 6161 break; 6162 case PDFWriter::Fit: 6163 aLine.append( "/Fit" ); 6164 break; 6165 case PDFWriter::FitRectangle: 6166 aLine.append( "/FitR " ); 6167 appendFixedInt( rDest.m_aRect.Left(), aLine ); 6168 aLine.append( ' ' ); 6169 appendFixedInt( rDest.m_aRect.Top(), aLine ); 6170 aLine.append( ' ' ); 6171 appendFixedInt( rDest.m_aRect.Right(), aLine ); 6172 aLine.append( ' ' ); 6173 appendFixedInt( rDest.m_aRect.Bottom(), aLine ); 6174 break; 6175 case PDFWriter::FitHorizontal: 6176 aLine.append( "/FitH " ); 6177 appendFixedInt( rDest.m_aRect.Bottom(), aLine ); 6178 break; 6179 case PDFWriter::FitVertical: 6180 aLine.append( "/FitV " ); 6181 appendFixedInt( rDest.m_aRect.Left(), aLine ); 6182 break; 6183 case PDFWriter::FitPageBoundingBox: 6184 aLine.append( "/FitB" ); 6185 break; 6186 case PDFWriter::FitPageBoundingBoxHorizontal: 6187 aLine.append( "/FitBH " ); 6188 appendFixedInt( rDest.m_aRect.Bottom(), aLine ); 6189 break; 6190 case PDFWriter::FitPageBoundingBoxVertical: 6191 aLine.append( "/FitBV " ); 6192 appendFixedInt( rDest.m_aRect.Left(), aLine ); 6193 break; 6194 } 6195 aLine.append( "]\n" ); 6196 } 6197 //close 6198 6199 aLine.append( ">>\nendobj\n\n" ); 6200 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 6201 nObject = 0; 6202 } 6203 else 6204 nObject = 0; 6205 6206 return nObject; 6207 } 6208 //<--- i56629 6209 6210 //--->i59651 6211 // emits the output intent dictionary 6212 6213 sal_Int32 PDFWriterImpl::emitOutputIntent() 6214 { 6215 if( !m_bIsPDF_A1 ) 6216 return 0; 6217 6218 //emit the sRGB standard profile, in ICC format, in a stream, per IEC61966-2.1 6219 6220 OStringBuffer aLine( 1024 ); 6221 sal_Int32 nICCObject = createObject(); 6222 sal_Int32 nStreamLengthObject = createObject(); 6223 6224 aLine.append( nICCObject ); 6225 // sRGB has 3 colors, hence /N 3 below (PDF 1.4 table 4.16) 6226 aLine.append( " 0 obj\n<</N 3/Length " ); 6227 aLine.append( nStreamLengthObject ); 6228 aLine.append( " 0 R" ); 6229 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 6230 aLine.append( "/Filter/FlateDecode" ); 6231 #endif 6232 aLine.append( ">>\nstream\n" ); 6233 CHECK_RETURN( updateObject( nICCObject ) ); 6234 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6235 //get file position 6236 sal_uInt64 nBeginStreamPos = 0; 6237 osl_getFilePos( m_aFile, &nBeginStreamPos ); 6238 beginCompression(); 6239 checkAndEnableStreamEncryption( nICCObject ); 6240 sal_Int32 nStreamSize = writeBuffer( nsRGB_ICC_profile, (sal_Int32) sizeof( nsRGB_ICC_profile ) ); 6241 disableStreamEncryption(); 6242 endCompression(); 6243 sal_uInt64 nEndStreamPos = 0; 6244 osl_getFilePos( m_aFile, &nEndStreamPos ); 6245 6246 if( nStreamSize == 0 ) 6247 return 0; 6248 if( ! writeBuffer( "\nendstream\nendobj\n\n", 19 ) ) 6249 return 0 ; 6250 aLine.setLength( 0 ); 6251 6252 //emit the stream length object 6253 CHECK_RETURN( updateObject( nStreamLengthObject ) ); 6254 aLine.setLength( 0 ); 6255 aLine.append( nStreamLengthObject ); 6256 aLine.append( " 0 obj\n" ); 6257 aLine.append( (sal_Int64)(nEndStreamPos-nBeginStreamPos) ); 6258 aLine.append( "\nendobj\n\n" ); 6259 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6260 aLine.setLength( 0 ); 6261 6262 //emit the OutputIntent dictionary 6263 sal_Int32 nOIObject = createObject(); 6264 CHECK_RETURN( updateObject( nOIObject ) ); 6265 aLine.append( nOIObject ); 6266 aLine.append( " 0 obj\n" 6267 "<</Type/OutputIntent/S/GTS_PDFA1/OutputConditionIdentifier"); 6268 6269 rtl::OUString aComment( RTL_CONSTASCII_USTRINGPARAM( "sRGB IEC61966-2.1" ) ); 6270 appendLiteralStringEncrypt( aComment ,nOIObject, aLine ); 6271 aLine.append("/DestOutputProfile "); 6272 aLine.append( nICCObject ); 6273 aLine.append( " 0 R>>\nendobj\n\n" );; 6274 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6275 6276 return nOIObject; 6277 } 6278 6279 // formats the string for the XML stream 6280 static void escapeStringXML( const rtl::OUString& rStr, rtl::OUString &rValue) 6281 { 6282 const sal_Unicode* pUni = rStr.getStr(); 6283 int nLen = rStr.getLength(); 6284 for( ; nLen; nLen--, pUni++ ) 6285 { 6286 switch( *pUni ) 6287 { 6288 case sal_Unicode('&'): 6289 rValue += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "&" ) ); 6290 break; 6291 case sal_Unicode('<'): 6292 rValue += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "<" ) ); 6293 break; 6294 case sal_Unicode('>'): 6295 rValue += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( ">" ) ); 6296 break; 6297 case sal_Unicode('\''): 6298 rValue += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "'" ) ); 6299 break; 6300 case sal_Unicode('"'): 6301 rValue += rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( """ ) ); 6302 break; 6303 default: 6304 rValue += rtl::OUString( *pUni ); 6305 break; 6306 } 6307 } 6308 } 6309 6310 // emits the document metadata 6311 // 6312 sal_Int32 PDFWriterImpl::emitDocumentMetadata() 6313 { 6314 if( !m_bIsPDF_A1 ) 6315 return 0; 6316 6317 //get the object number for all the destinations 6318 sal_Int32 nObject = createObject(); 6319 6320 if( updateObject( nObject ) ) 6321 { 6322 // the following string are written in UTF-8 unicode 6323 OStringBuffer aMetadataStream( 8192 ); 6324 6325 aMetadataStream.append( "<?xpacket begin=\"" ); 6326 // this lines writes Unicode “zero width non-breaking space character” (U+FEFF) (aka byte-order mark ) used 6327 // as a byte-order marker. 6328 aMetadataStream.append( OUStringToOString( OUString( sal_Unicode( 0xFEFF ) ), RTL_TEXTENCODING_UTF8 ) ); 6329 aMetadataStream.append( "\" id=\"W5M0MpCehiHzreSzNTczkc9d\"?>\n" ); 6330 aMetadataStream.append( "<x:xmpmeta xmlns:x=\"adobe:ns:meta/\">\n" ); 6331 aMetadataStream.append( " <rdf:RDF xmlns:rdf=\"http://www.w3.org/1999/02/22-rdf-syntax-ns#\">\n" ); 6332 //PDF/A part ( ISO 19005-1:2005 - 6.7.11 ) 6333 aMetadataStream.append( " <rdf:Description rdf:about=\"\"\n" ); 6334 aMetadataStream.append( " xmlns:pdfaid=\"http://www.aiim.org/pdfa/ns/id/\">\n" ); 6335 aMetadataStream.append( " <pdfaid:part>1</pdfaid:part>\n" ); 6336 aMetadataStream.append( " <pdfaid:conformance>A</pdfaid:conformance>\n" ); 6337 aMetadataStream.append( " </rdf:Description>\n" ); 6338 //... Dublin Core properties go here 6339 if( m_aContext.DocumentInfo.Title.Len() || 6340 m_aContext.DocumentInfo.Author.Len() || 6341 m_aContext.DocumentInfo.Subject.Len() ) 6342 { 6343 aMetadataStream.append( " <rdf:Description rdf:about=\"\"\n" ); 6344 aMetadataStream.append( " xmlns:dc=\"http://purl.org/dc/elements/1.1/\">\n" ); 6345 if( m_aContext.DocumentInfo.Title.Len() ) 6346 { 6347 // this is according to PDF/A-1, technical corrigendum 1 (2007-04-01) 6348 aMetadataStream.append( " <dc:title>\n" ); 6349 aMetadataStream.append( " <rdf:Alt>\n" ); 6350 aMetadataStream.append( " <rdf:li xml:lang=\"x-default\">" ); 6351 rtl::OUString aTitle; 6352 escapeStringXML( m_aContext.DocumentInfo.Title, aTitle ); 6353 aMetadataStream.append( OUStringToOString( aTitle, RTL_TEXTENCODING_UTF8 ) ); 6354 aMetadataStream.append( "</rdf:li>\n" ); 6355 aMetadataStream.append( " </rdf:Alt>\n" ); 6356 aMetadataStream.append( " </dc:title>\n" ); 6357 } 6358 if( m_aContext.DocumentInfo.Author.Len() ) 6359 { 6360 aMetadataStream.append( " <dc:creator>\n" ); 6361 aMetadataStream.append( " <rdf:Seq>\n" ); 6362 aMetadataStream.append( " <rdf:li>" ); 6363 rtl::OUString aAuthor; 6364 escapeStringXML( m_aContext.DocumentInfo.Author, aAuthor ); 6365 aMetadataStream.append( OUStringToOString( aAuthor , RTL_TEXTENCODING_UTF8 ) ); 6366 aMetadataStream.append( "</rdf:li>\n" ); 6367 aMetadataStream.append( " </rdf:Seq>\n" ); 6368 aMetadataStream.append( " </dc:creator>\n" ); 6369 } 6370 if( m_aContext.DocumentInfo.Subject.Len() ) 6371 { 6372 // this is according to PDF/A-1, technical corrigendum 1 (2007-04-01) 6373 aMetadataStream.append( " <dc:description>\n" ); 6374 aMetadataStream.append( " <rdf:Alt>\n" ); 6375 aMetadataStream.append( " <rdf:li xml:lang=\"x-default\">" ); 6376 rtl::OUString aSubject; 6377 escapeStringXML( m_aContext.DocumentInfo.Subject, aSubject ); 6378 aMetadataStream.append( OUStringToOString( aSubject , RTL_TEXTENCODING_UTF8 ) ); 6379 aMetadataStream.append( "</rdf:li>\n" ); 6380 aMetadataStream.append( " </rdf:Alt>\n" ); 6381 aMetadataStream.append( " </dc:description>\n" ); 6382 } 6383 aMetadataStream.append( " </rdf:Description>\n" ); 6384 } 6385 6386 //... PDF properties go here 6387 if( m_aContext.DocumentInfo.Producer.Len() || 6388 m_aContext.DocumentInfo.Keywords.Len() ) 6389 { 6390 aMetadataStream.append( " <rdf:Description rdf:about=\"\"\n" ); 6391 aMetadataStream.append( " xmlns:pdf=\"http://ns.adobe.com/pdf/1.3/\">\n" ); 6392 if( m_aContext.DocumentInfo.Producer.Len() ) 6393 { 6394 aMetadataStream.append( " <pdf:Producer>" ); 6395 rtl::OUString aProducer; 6396 escapeStringXML( m_aContext.DocumentInfo.Producer, aProducer ); 6397 aMetadataStream.append( OUStringToOString( aProducer , RTL_TEXTENCODING_UTF8 ) ); 6398 aMetadataStream.append( "</pdf:Producer>\n" ); 6399 } 6400 if( m_aContext.DocumentInfo.Keywords.Len() ) 6401 { 6402 aMetadataStream.append( " <pdf:Keywords>" ); 6403 rtl::OUString aKeywords; 6404 escapeStringXML( m_aContext.DocumentInfo.Keywords, aKeywords ); 6405 aMetadataStream.append( OUStringToOString( aKeywords , RTL_TEXTENCODING_UTF8 ) ); 6406 aMetadataStream.append( "</pdf:Keywords>\n" ); 6407 } 6408 aMetadataStream.append( " </rdf:Description>\n" ); 6409 } 6410 6411 aMetadataStream.append( " <rdf:Description rdf:about=\"\"\n" ); 6412 aMetadataStream.append( " xmlns:xmp=\"http://ns.adobe.com/xap/1.0/\">\n" ); 6413 if( m_aContext.DocumentInfo.Creator.Len() ) 6414 { 6415 aMetadataStream.append( " <xmp:CreatorTool>" ); 6416 rtl::OUString aCreator; 6417 escapeStringXML( m_aContext.DocumentInfo.Creator, aCreator ); 6418 aMetadataStream.append( OUStringToOString( aCreator , RTL_TEXTENCODING_UTF8 ) ); 6419 aMetadataStream.append( "</xmp:CreatorTool>\n" ); 6420 } 6421 //creation date 6422 aMetadataStream.append( " <xmp:CreateDate>" ); 6423 aMetadataStream.append( m_aCreationMetaDateString ); 6424 aMetadataStream.append( "</xmp:CreateDate>\n" ); 6425 6426 aMetadataStream.append( " </rdf:Description>\n" ); 6427 aMetadataStream.append( " </rdf:RDF>\n" ); 6428 aMetadataStream.append( "</x:xmpmeta>\n" ); 6429 6430 //add the padding 6431 for( sal_Int32 nSpaces = 1; nSpaces <= 2100; nSpaces++ ) 6432 { 6433 aMetadataStream.append( " " ); 6434 if( nSpaces % 100 == 0 ) 6435 aMetadataStream.append( "\n" ); 6436 } 6437 6438 aMetadataStream.append( "<?xpacket end=\"w\"?>\n" ); 6439 6440 OStringBuffer aMetadataObj( 1024 ); 6441 6442 aMetadataObj.append( nObject ); 6443 aMetadataObj.append( " 0 obj\n" ); 6444 6445 aMetadataObj.append( "<</Type/Metadata/Subtype/XML/Length " ); 6446 6447 aMetadataObj.append( (sal_Int32) aMetadataStream.getLength() ); 6448 aMetadataObj.append( ">>\nstream\n" ); 6449 CHECK_RETURN( writeBuffer( aMetadataObj.getStr(), aMetadataObj.getLength() ) ); 6450 //emit the stream 6451 CHECK_RETURN( writeBuffer( aMetadataStream.getStr(), aMetadataStream.getLength() ) ); 6452 6453 aMetadataObj.setLength( 0 ); 6454 aMetadataObj.append( "\nendstream\nendobj\n\n" ); 6455 if( ! writeBuffer( aMetadataObj.getStr(), aMetadataObj.getLength() ) ) 6456 nObject = 0; 6457 } 6458 else 6459 nObject = 0; 6460 6461 return nObject; 6462 } 6463 //<---i59651 6464 6465 bool PDFWriterImpl::emitTrailer() 6466 { 6467 // emit doc info 6468 OString aInfoValuesOut; 6469 sal_Int32 nDocInfoObject = emitInfoDict( ); 6470 6471 sal_Int32 nSecObject = 0; 6472 6473 if( m_aContext.Encryption.Encrypt() ) 6474 { 6475 //emit the security information 6476 //must be emitted as indirect dictionary object, since 6477 //Acrobat Reader 5 works only with this kind of implementation 6478 nSecObject = createObject(); 6479 6480 if( updateObject( nSecObject ) ) 6481 { 6482 OStringBuffer aLineS( 1024 ); 6483 aLineS.append( nSecObject ); 6484 aLineS.append( " 0 obj\n" 6485 "<</Filter/Standard/V " ); 6486 // check the version 6487 if( m_aContext.Encryption.Security128bit ) 6488 aLineS.append( "2/Length 128/R 3" ); 6489 else 6490 aLineS.append( "1/R 2" ); 6491 6492 // emit the owner password, must not be encrypted 6493 aLineS.append( "/O(" ); 6494 appendLiteralString( (const sal_Char*)&m_aContext.Encryption.OValue[0], sal_Int32(m_aContext.Encryption.OValue.size()), aLineS ); 6495 aLineS.append( ")/U(" ); 6496 appendLiteralString( (const sal_Char*)&m_aContext.Encryption.UValue[0], sal_Int32(m_aContext.Encryption.UValue.size()), aLineS ); 6497 aLineS.append( ")/P " );// the permission set 6498 aLineS.append( m_nAccessPermissions ); 6499 aLineS.append( ">>\nendobj\n\n" ); 6500 if( !writeBuffer( aLineS.getStr(), aLineS.getLength() ) ) 6501 nSecObject = 0; 6502 } 6503 else 6504 nSecObject = 0; 6505 } 6506 // emit xref table 6507 // remember start 6508 sal_uInt64 nXRefOffset = 0; 6509 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nXRefOffset )) ); 6510 CHECK_RETURN( writeBuffer( "xref\n", 5 ) ); 6511 6512 sal_Int32 nObjects = m_aObjects.size(); 6513 OStringBuffer aLine; 6514 aLine.append( "0 " ); 6515 aLine.append( (sal_Int32)(nObjects+1) ); 6516 aLine.append( "\n" ); 6517 aLine.append( "0000000000 65535 f \n" ); 6518 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6519 6520 for( sal_Int32 i = 0; i < nObjects; i++ ) 6521 { 6522 aLine.setLength( 0 ); 6523 OString aOffset = OString::valueOf( (sal_Int64)m_aObjects[i] ); 6524 for( sal_Int32 j = 0; j < (10-aOffset.getLength()); j++ ) 6525 aLine.append( '0' ); 6526 aLine.append( aOffset ); 6527 aLine.append( " 00000 n \n" ); 6528 DBG_ASSERT( aLine.getLength() == 20, "invalid xref entry" ); 6529 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6530 } 6531 6532 // prepare document checksum 6533 OStringBuffer aDocChecksum( 2*RTL_DIGEST_LENGTH_MD5+1 ); 6534 if( m_aDocDigest ) 6535 { 6536 sal_uInt8 nMD5Sum[ RTL_DIGEST_LENGTH_MD5 ]; 6537 rtl_digest_getMD5( m_aDocDigest, nMD5Sum, sizeof(nMD5Sum) ); 6538 for( unsigned int i = 0; i < RTL_DIGEST_LENGTH_MD5; i++ ) 6539 appendHex( nMD5Sum[i], aDocChecksum ); 6540 } 6541 // document id set in setDocInfo method 6542 // emit trailer 6543 aLine.setLength( 0 ); 6544 aLine.append( "trailer\n" 6545 "<</Size " ); 6546 aLine.append( (sal_Int32)(nObjects+1) ); 6547 aLine.append( "/Root " ); 6548 aLine.append( m_nCatalogObject ); 6549 aLine.append( " 0 R\n" ); 6550 if( nSecObject |= 0 ) 6551 { 6552 aLine.append( "/Encrypt "); 6553 aLine.append( nSecObject ); 6554 aLine.append( " 0 R\n" ); 6555 } 6556 if( nDocInfoObject ) 6557 { 6558 aLine.append( "/Info " ); 6559 aLine.append( nDocInfoObject ); 6560 aLine.append( " 0 R\n" ); 6561 } 6562 if( ! m_aContext.Encryption.DocumentIdentifier.empty() ) 6563 { 6564 aLine.append( "/ID [ <" ); 6565 for( std::vector< sal_uInt8 >::const_iterator it = m_aContext.Encryption.DocumentIdentifier.begin(); 6566 it != m_aContext.Encryption.DocumentIdentifier.end(); ++it ) 6567 { 6568 appendHex( sal_Int8(*it), aLine ); 6569 } 6570 aLine.append( ">\n" 6571 "<" ); 6572 for( std::vector< sal_uInt8 >::const_iterator it = m_aContext.Encryption.DocumentIdentifier.begin(); 6573 it != m_aContext.Encryption.DocumentIdentifier.end(); ++it ) 6574 { 6575 appendHex( sal_Int8(*it), aLine ); 6576 } 6577 aLine.append( "> ]\n" ); 6578 } 6579 if( aDocChecksum.getLength() ) 6580 { 6581 aLine.append( "/DocChecksum /" ); 6582 aLine.append( aDocChecksum ); 6583 aLine.append( "\n" ); 6584 } 6585 if( m_aAdditionalStreams.size() > 0 ) 6586 { 6587 aLine.append( "/AdditionalStreams [" ); 6588 for( unsigned int i = 0; i < m_aAdditionalStreams.size(); i++ ) 6589 { 6590 aLine.append( "/" ); 6591 appendName( m_aAdditionalStreams[i].m_aMimeType, aLine ); 6592 aLine.append( " " ); 6593 aLine.append( m_aAdditionalStreams[i].m_nStreamObject ); 6594 aLine.append( " 0 R\n" ); 6595 } 6596 aLine.append( "]\n" ); 6597 } 6598 aLine.append( ">>\n" 6599 "startxref\n" ); 6600 aLine.append( (sal_Int64)nXRefOffset ); 6601 aLine.append( "\n" 6602 "%%EOF\n" ); 6603 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 6604 6605 return true; 6606 } 6607 6608 struct AnnotationSortEntry 6609 { 6610 sal_Int32 nTabOrder; 6611 sal_Int32 nObject; 6612 sal_Int32 nWidgetIndex; 6613 6614 AnnotationSortEntry( sal_Int32 nTab, sal_Int32 nObj, sal_Int32 nI ) : 6615 nTabOrder( nTab ), 6616 nObject( nObj ), 6617 nWidgetIndex( nI ) 6618 {} 6619 }; 6620 6621 struct AnnotSortContainer 6622 { 6623 std::set< sal_Int32 > aObjects; 6624 std::vector< AnnotationSortEntry > aSortedAnnots; 6625 }; 6626 6627 struct AnnotSorterLess 6628 { 6629 std::vector< PDFWriterImpl::PDFWidget >& m_rWidgets; 6630 6631 AnnotSorterLess( std::vector< PDFWriterImpl::PDFWidget >& rWidgets ) : m_rWidgets( rWidgets ) {} 6632 6633 bool operator()( const AnnotationSortEntry& rLeft, const AnnotationSortEntry& rRight ) 6634 { 6635 if( rLeft.nTabOrder < rRight.nTabOrder ) 6636 return true; 6637 if( rRight.nTabOrder < rLeft.nTabOrder ) 6638 return false; 6639 if( rLeft.nWidgetIndex < 0 && rRight.nWidgetIndex < 0 ) 6640 return false; 6641 if( rRight.nWidgetIndex < 0 ) 6642 return true; 6643 if( rLeft.nWidgetIndex < 0 ) 6644 return false; 6645 // remember: widget rects are in PDF coordinates, so they are ordered down up 6646 if( m_rWidgets[ rLeft.nWidgetIndex ].m_aRect.Top() > 6647 m_rWidgets[ rRight.nWidgetIndex ].m_aRect.Top() ) 6648 return true; 6649 if( m_rWidgets[ rRight.nWidgetIndex ].m_aRect.Top() > 6650 m_rWidgets[ rLeft.nWidgetIndex ].m_aRect.Top() ) 6651 return false; 6652 if( m_rWidgets[ rLeft.nWidgetIndex ].m_aRect.Left() < 6653 m_rWidgets[ rRight.nWidgetIndex ].m_aRect.Left() ) 6654 return true; 6655 return false; 6656 } 6657 }; 6658 6659 void PDFWriterImpl::sortWidgets() 6660 { 6661 // sort widget annotations on each page as per their 6662 // TabOrder attribute 6663 std::hash_map< sal_Int32, AnnotSortContainer > sorted; 6664 int nWidgets = m_aWidgets.size(); 6665 for( int nW = 0; nW < nWidgets; nW++ ) 6666 { 6667 const PDFWidget& rWidget = m_aWidgets[nW]; 6668 if( rWidget.m_nPage >= 0 ) 6669 { 6670 AnnotSortContainer& rCont = sorted[ rWidget.m_nPage ]; 6671 // optimize vector allocation 6672 if( rCont.aSortedAnnots.empty() ) 6673 rCont.aSortedAnnots.reserve( m_aPages[ rWidget.m_nPage ].m_aAnnotations.size() ); 6674 // insert widget to tab sorter 6675 // RadioButtons are not page annotations, only their individual check boxes are 6676 if( rWidget.m_eType != PDFWriter::RadioButton ) 6677 { 6678 rCont.aObjects.insert( rWidget.m_nObject ); 6679 rCont.aSortedAnnots.push_back( AnnotationSortEntry( rWidget.m_nTabOrder, rWidget.m_nObject, nW ) ); 6680 } 6681 } 6682 } 6683 for( std::hash_map< sal_Int32, AnnotSortContainer >::iterator it = sorted.begin(); it != sorted.end(); ++it ) 6684 { 6685 // append entries for non widget annotations 6686 PDFPage& rPage = m_aPages[ it->first ]; 6687 unsigned int nAnnots = rPage.m_aAnnotations.size(); 6688 for( unsigned int nA = 0; nA < nAnnots; nA++ ) 6689 if( it->second.aObjects.find( rPage.m_aAnnotations[nA] ) == it->second.aObjects.end()) 6690 it->second.aSortedAnnots.push_back( AnnotationSortEntry( 10000, rPage.m_aAnnotations[nA], -1 ) ); 6691 6692 AnnotSorterLess aLess( m_aWidgets ); 6693 std::stable_sort( it->second.aSortedAnnots.begin(), it->second.aSortedAnnots.end(), aLess ); 6694 // sanity check 6695 if( it->second.aSortedAnnots.size() == nAnnots) 6696 { 6697 for( unsigned int nA = 0; nA < nAnnots; nA++ ) 6698 rPage.m_aAnnotations[nA] = it->second.aSortedAnnots[nA].nObject; 6699 } 6700 else 6701 { 6702 DBG_ASSERT( 0, "wrong number of sorted annotations" ); 6703 #if OSL_DEBUG_LEVEL > 0 6704 fprintf( stderr, "PDFWriterImpl::sortWidgets(): wrong number of sorted assertions on page nr %ld\n" 6705 " %ld sorted and %ld unsorted\n", (long int)it->first, (long int)it->second.aSortedAnnots.size(), (long int)nAnnots ); 6706 #endif 6707 } 6708 } 6709 6710 // FIXME: implement tab order in structure tree for PDF 1.5 6711 } 6712 6713 namespace vcl { 6714 class PDFStreamIf : 6715 public cppu::WeakImplHelper1< com::sun::star::io::XOutputStream > 6716 { 6717 PDFWriterImpl* m_pWriter; 6718 bool m_bWrite; 6719 public: 6720 PDFStreamIf( PDFWriterImpl* pWriter ) : m_pWriter( pWriter ), m_bWrite( true ) {} 6721 virtual ~PDFStreamIf(); 6722 6723 virtual void SAL_CALL writeBytes( const com::sun::star::uno::Sequence< sal_Int8 >& aData ) throw(); 6724 virtual void SAL_CALL flush() throw(); 6725 virtual void SAL_CALL closeOutput() throw(); 6726 }; 6727 } 6728 6729 PDFStreamIf::~PDFStreamIf() 6730 { 6731 } 6732 6733 void SAL_CALL PDFStreamIf::writeBytes( const com::sun::star::uno::Sequence< sal_Int8 >& aData ) throw() 6734 { 6735 if( m_bWrite ) 6736 { 6737 sal_Int32 nBytes = aData.getLength(); 6738 if( nBytes > 0 ) 6739 m_pWriter->writeBuffer( aData.getConstArray(), nBytes ); 6740 } 6741 } 6742 6743 void SAL_CALL PDFStreamIf::flush() throw() 6744 { 6745 } 6746 6747 void SAL_CALL PDFStreamIf::closeOutput() throw() 6748 { 6749 m_bWrite = false; 6750 } 6751 6752 bool PDFWriterImpl::emitAdditionalStreams() 6753 { 6754 unsigned int nStreams = m_aAdditionalStreams.size(); 6755 for( unsigned int i = 0; i < nStreams; i++ ) 6756 { 6757 PDFAddStream& rStream = m_aAdditionalStreams[i]; 6758 rStream.m_nStreamObject = createObject(); 6759 sal_Int32 nSizeObject = createObject(); 6760 6761 if( ! updateObject( rStream.m_nStreamObject ) ) 6762 return false; 6763 6764 OStringBuffer aLine; 6765 aLine.append( rStream.m_nStreamObject ); 6766 aLine.append( " 0 obj\n<</Length " ); 6767 aLine.append( nSizeObject ); 6768 aLine.append( " 0 R" ); 6769 if( rStream.m_bCompress ) 6770 aLine.append( "/Filter/FlateDecode" ); 6771 aLine.append( ">>\nstream\n" ); 6772 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 6773 return false; 6774 sal_uInt64 nBeginStreamPos = 0, nEndStreamPos = 0; 6775 if( osl_File_E_None != osl_getFilePos( m_aFile, &nBeginStreamPos ) ) 6776 { 6777 osl_closeFile( m_aFile ); 6778 m_bOpen = false; 6779 } 6780 if( rStream.m_bCompress ) 6781 beginCompression(); 6782 6783 checkAndEnableStreamEncryption( rStream.m_nStreamObject ); 6784 com::sun::star::uno::Reference< com::sun::star::io::XOutputStream > xStream( new PDFStreamIf( this ) ); 6785 rStream.m_pStream->write( xStream ); 6786 xStream.clear(); 6787 delete rStream.m_pStream; 6788 rStream.m_pStream = NULL; 6789 disableStreamEncryption(); 6790 6791 if( rStream.m_bCompress ) 6792 endCompression(); 6793 6794 if( osl_File_E_None != osl_getFilePos( m_aFile, &nEndStreamPos ) ) 6795 { 6796 osl_closeFile( m_aFile ); 6797 m_bOpen = false; 6798 return false; 6799 } 6800 if( ! writeBuffer( "\nendstream\nendobj\n\n", 19 ) ) 6801 return false ; 6802 // emit stream length object 6803 if( ! updateObject( nSizeObject ) ) 6804 return false; 6805 aLine.setLength( 0 ); 6806 aLine.append( nSizeObject ); 6807 aLine.append( " 0 obj\n" ); 6808 aLine.append( (sal_Int64)(nEndStreamPos-nBeginStreamPos) ); 6809 aLine.append( "\nendobj\n\n" ); 6810 if( ! writeBuffer( aLine.getStr(), aLine.getLength() ) ) 6811 return false; 6812 } 6813 return true; 6814 } 6815 6816 bool PDFWriterImpl::emit() 6817 { 6818 endPage(); 6819 6820 // resort structure tree and annotations if necessary 6821 // needed for widget tab order 6822 sortWidgets(); 6823 6824 // emit additional streams 6825 CHECK_RETURN( emitAdditionalStreams() ); 6826 6827 // emit catalog 6828 CHECK_RETURN( emitCatalog() ); 6829 6830 // emit trailer 6831 CHECK_RETURN( emitTrailer() ); 6832 6833 osl_closeFile( m_aFile ); 6834 m_bOpen = false; 6835 6836 return true; 6837 } 6838 6839 std::set< PDFWriter::ErrorCode > PDFWriterImpl::getErrors() 6840 { 6841 return m_aErrors; 6842 } 6843 6844 sal_Int32 PDFWriterImpl::getSystemFont( const Font& i_rFont ) 6845 { 6846 getReferenceDevice()->Push(); 6847 getReferenceDevice()->SetFont( i_rFont ); 6848 getReferenceDevice()->ImplNewFont(); 6849 6850 const ImplFontData* pDevFont = m_pReferenceDevice->mpFontEntry->maFontSelData.mpFontData; 6851 sal_Int32 nFontID = 0; 6852 FontEmbedData::iterator it = m_aSystemFonts.find( pDevFont ); 6853 if( it != m_aSystemFonts.end() ) 6854 nFontID = it->second.m_nNormalFontID; 6855 else 6856 { 6857 nFontID = m_nNextFID++; 6858 m_aSystemFonts[ pDevFont ] = EmbedFont(); 6859 m_aSystemFonts[ pDevFont ].m_nNormalFontID = nFontID; 6860 } 6861 6862 getReferenceDevice()->Pop(); 6863 getReferenceDevice()->ImplNewFont(); 6864 6865 return nFontID; 6866 } 6867 6868 void PDFWriterImpl::registerGlyphs( int nGlyphs, 6869 sal_GlyphId* pGlyphs, 6870 sal_Int32* pGlyphWidths, 6871 sal_Ucs* pUnicodes, 6872 sal_Int32* pUnicodesPerGlyph, 6873 sal_uInt8* pMappedGlyphs, 6874 sal_Int32* pMappedFontObjects, 6875 const ImplFontData* pFallbackFonts[] ) 6876 { 6877 const ImplFontData* pDevFont = m_pReferenceDevice->mpFontEntry->maFontSelData.mpFontData; 6878 sal_Ucs* pCurUnicode = pUnicodes; 6879 for( int i = 0; i < nGlyphs; pCurUnicode += pUnicodesPerGlyph[i] , i++ ) 6880 { 6881 const int nFontGlyphId = pGlyphs[i] & (GF_IDXMASK | GF_ISCHAR | GF_GSUB); 6882 const ImplFontData* pCurrentFont = pFallbackFonts[i] ? pFallbackFonts[i] : pDevFont; 6883 6884 if( isBuiltinFont( pCurrentFont ) ) 6885 { 6886 sal_Int32 nFontID = 0; 6887 FontEmbedData::iterator it = m_aEmbeddedFonts.find( pCurrentFont ); 6888 if( it != m_aEmbeddedFonts.end() ) 6889 nFontID = it->second.m_nNormalFontID; 6890 else 6891 { 6892 nFontID = m_nNextFID++; 6893 m_aEmbeddedFonts[ pCurrentFont ] = EmbedFont(); 6894 m_aEmbeddedFonts[ pCurrentFont ].m_nNormalFontID = nFontID; 6895 } 6896 6897 pGlyphWidths[ i ] = 0; 6898 pMappedGlyphs[ i ] = sal::static_int_cast<sal_Int8>( nFontGlyphId ); 6899 pMappedFontObjects[ i ] = nFontID; 6900 const ImplPdfBuiltinFontData* pFD = GetPdfFontData( pCurrentFont ); 6901 if( pFD ) 6902 { 6903 const BuiltinFont* pBuiltinFont = pFD->GetBuiltinFont(); 6904 pGlyphWidths[i] = pBuiltinFont->m_aWidths[ nFontGlyphId & 0x00ff ]; 6905 } 6906 } 6907 else if( pCurrentFont->mbSubsettable ) 6908 { 6909 FontSubset& rSubset = m_aSubsets[ pCurrentFont ]; 6910 // search for font specific glyphID 6911 FontMapping::iterator it = rSubset.m_aMapping.find( nFontGlyphId ); 6912 if( it != rSubset.m_aMapping.end() ) 6913 { 6914 pMappedFontObjects[i] = it->second.m_nFontID; 6915 pMappedGlyphs[i] = it->second.m_nSubsetGlyphID; 6916 } 6917 else 6918 { 6919 // create new subset if necessary 6920 if( rSubset.m_aSubsets.empty() 6921 || (rSubset.m_aSubsets.back().m_aMapping.size() > 254) ) 6922 { 6923 rSubset.m_aSubsets.push_back( FontEmit( m_nNextFID++ ) ); 6924 } 6925 6926 // copy font id 6927 pMappedFontObjects[i] = rSubset.m_aSubsets.back().m_nFontID; 6928 // create new glyph in subset 6929 sal_uInt8 nNewId = sal::static_int_cast<sal_uInt8>(rSubset.m_aSubsets.back().m_aMapping.size()+1); 6930 pMappedGlyphs[i] = nNewId; 6931 6932 // add new glyph to emitted font subset 6933 GlyphEmit& rNewGlyphEmit = rSubset.m_aSubsets.back().m_aMapping[ nFontGlyphId ]; 6934 rNewGlyphEmit.setGlyphId( nNewId ); 6935 for( sal_Int32 n = 0; n < pUnicodesPerGlyph[i]; n++ ) 6936 rNewGlyphEmit.addCode( pCurUnicode[n] ); 6937 6938 // add new glyph to font mapping 6939 Glyph& rNewGlyph = rSubset.m_aMapping[ nFontGlyphId ]; 6940 rNewGlyph.m_nFontID = pMappedFontObjects[i]; 6941 rNewGlyph.m_nSubsetGlyphID = nNewId; 6942 } 6943 getReferenceDevice()->ImplGetGraphics(); 6944 const bool bVertical = ((pGlyphs[i] & GF_ROTMASK) != 0); 6945 pGlyphWidths[i] = m_aFontCache.getGlyphWidth( pCurrentFont, 6946 nFontGlyphId, 6947 bVertical, 6948 m_pReferenceDevice->mpGraphics ); 6949 } 6950 else if( pCurrentFont->IsEmbeddable() ) 6951 { 6952 sal_Int32 nFontID = 0; 6953 FontEmbedData::iterator it = m_aEmbeddedFonts.find( pCurrentFont ); 6954 if( it != m_aEmbeddedFonts.end() ) 6955 nFontID = it->second.m_nNormalFontID; 6956 else 6957 { 6958 nFontID = m_nNextFID++; 6959 m_aEmbeddedFonts[ pCurrentFont ] = EmbedFont(); 6960 m_aEmbeddedFonts[ pCurrentFont ].m_nNormalFontID = nFontID; 6961 } 6962 EmbedFont& rEmbedFont = m_aEmbeddedFonts[pCurrentFont]; 6963 6964 const Ucs2SIntMap* pEncoding = NULL; 6965 const Ucs2OStrMap* pNonEncoded = NULL; 6966 getReferenceDevice()->ImplGetGraphics(); 6967 pEncoding = m_pReferenceDevice->mpGraphics->GetFontEncodingVector( pCurrentFont, &pNonEncoded ); 6968 6969 Ucs2SIntMap::const_iterator enc_it; 6970 Ucs2OStrMap::const_iterator nonenc_it; 6971 6972 sal_Int32 nCurFontID = nFontID; 6973 sal_Ucs cChar = *pCurUnicode; 6974 if( pEncoding ) 6975 { 6976 enc_it = pEncoding->find( cChar ); 6977 if( enc_it != pEncoding->end() && enc_it->second > 0 ) 6978 { 6979 DBG_ASSERT( (enc_it->second & 0xffffff00) == 0, "Invalid character code" ); 6980 cChar = (sal_Ucs)enc_it->second; 6981 } 6982 else if( (enc_it == pEncoding->end() || enc_it->second == -1) && 6983 pNonEncoded && 6984 (nonenc_it = pNonEncoded->find( cChar )) != pNonEncoded->end() ) 6985 { 6986 nCurFontID = 0; 6987 // find non encoded glyph 6988 for( std::list< EmbedEncoding >::iterator nec_it = rEmbedFont.m_aExtendedEncodings.begin(); nec_it != rEmbedFont.m_aExtendedEncodings.end(); ++nec_it ) 6989 { 6990 if( nec_it->m_aCMap.find( cChar ) != nec_it->m_aCMap.end() ) 6991 { 6992 nCurFontID = nec_it->m_nFontID; 6993 cChar = (sal_Ucs)nec_it->m_aCMap[ cChar ]; 6994 break; 6995 } 6996 } 6997 if( nCurFontID == 0 ) // new nonencoded glyph 6998 { 6999 if( rEmbedFont.m_aExtendedEncodings.empty() || rEmbedFont.m_aExtendedEncodings.back().m_aEncVector.size() == 255 ) 7000 { 7001 rEmbedFont.m_aExtendedEncodings.push_back( EmbedEncoding() ); 7002 rEmbedFont.m_aExtendedEncodings.back().m_nFontID = m_nNextFID++; 7003 } 7004 EmbedEncoding& rEncoding = rEmbedFont.m_aExtendedEncodings.back(); 7005 rEncoding.m_aEncVector.push_back( EmbedCode() ); 7006 rEncoding.m_aEncVector.back().m_aUnicode = cChar; 7007 rEncoding.m_aEncVector.back().m_aName = nonenc_it->second; 7008 rEncoding.m_aCMap[ cChar ] = (sal_Int8)(rEncoding.m_aEncVector.size()-1); 7009 nCurFontID = rEncoding.m_nFontID; 7010 cChar = (sal_Ucs)rEncoding.m_aCMap[ cChar ]; 7011 } 7012 } 7013 else 7014 pEncoding = NULL; 7015 } 7016 if( ! pEncoding ) 7017 { 7018 if( cChar & 0xff00 ) 7019 { 7020 // some characters can be used by conversion 7021 if( cChar >= 0xf000 && cChar <= 0xf0ff ) // symbol encoding in private use area 7022 cChar -= 0xf000; 7023 else 7024 { 7025 String aString(cChar); 7026 ByteString aChar( aString, RTL_TEXTENCODING_MS_1252 ); 7027 cChar = ((sal_Ucs)aChar.GetChar( 0 )) & 0x00ff; 7028 } 7029 } 7030 } 7031 7032 pMappedGlyphs[ i ] = (sal_Int8)cChar; 7033 pMappedFontObjects[ i ] = nCurFontID; 7034 pGlyphWidths[ i ] = m_aFontCache.getGlyphWidth( pCurrentFont, 7035 (pEncoding ? *pCurUnicode : cChar) | GF_ISCHAR, 7036 false, 7037 m_pReferenceDevice->mpGraphics ); 7038 } 7039 } 7040 } 7041 7042 void PDFWriterImpl::drawRelief( SalLayout& rLayout, const String& rText, bool bTextLines ) 7043 { 7044 push( PUSH_ALL ); 7045 7046 FontRelief eRelief = m_aCurrentPDFState.m_aFont.GetRelief(); 7047 7048 Color aTextColor = m_aCurrentPDFState.m_aFont.GetColor(); 7049 Color aTextLineColor = m_aCurrentPDFState.m_aTextLineColor; 7050 Color aOverlineColor = m_aCurrentPDFState.m_aOverlineColor; 7051 Color aReliefColor( COL_LIGHTGRAY ); 7052 if( aTextColor == COL_BLACK ) 7053 aTextColor = Color( COL_WHITE ); 7054 if( aTextLineColor == COL_BLACK ) 7055 aTextLineColor = Color( COL_WHITE ); 7056 if( aOverlineColor == COL_BLACK ) 7057 aOverlineColor = Color( COL_WHITE ); 7058 if( aTextColor == COL_WHITE ) 7059 aReliefColor = Color( COL_BLACK ); 7060 7061 Font aSetFont = m_aCurrentPDFState.m_aFont; 7062 aSetFont.SetRelief( RELIEF_NONE ); 7063 aSetFont.SetShadow( sal_False ); 7064 7065 aSetFont.SetColor( aReliefColor ); 7066 setTextLineColor( aReliefColor ); 7067 setOverlineColor( aReliefColor ); 7068 setFont( aSetFont ); 7069 long nOff = 1 + getReferenceDevice()->mnDPIX/300; 7070 if( eRelief == RELIEF_ENGRAVED ) 7071 nOff = -nOff; 7072 7073 rLayout.DrawOffset() += Point( nOff, nOff ); 7074 updateGraphicsState(); 7075 drawLayout( rLayout, rText, bTextLines ); 7076 7077 rLayout.DrawOffset() -= Point( nOff, nOff ); 7078 setTextLineColor( aTextLineColor ); 7079 setOverlineColor( aOverlineColor ); 7080 aSetFont.SetColor( aTextColor ); 7081 setFont( aSetFont ); 7082 updateGraphicsState(); 7083 drawLayout( rLayout, rText, bTextLines ); 7084 7085 // clean up the mess 7086 pop(); 7087 } 7088 7089 void PDFWriterImpl::drawShadow( SalLayout& rLayout, const String& rText, bool bTextLines ) 7090 { 7091 Font aSaveFont = m_aCurrentPDFState.m_aFont; 7092 Color aSaveTextLineColor = m_aCurrentPDFState.m_aTextLineColor; 7093 Color aSaveOverlineColor = m_aCurrentPDFState.m_aOverlineColor; 7094 7095 Font& rFont = m_aCurrentPDFState.m_aFont; 7096 if( rFont.GetColor() == Color( COL_BLACK ) || rFont.GetColor().GetLuminance() < 8 ) 7097 rFont.SetColor( Color( COL_LIGHTGRAY ) ); 7098 else 7099 rFont.SetColor( Color( COL_BLACK ) ); 7100 rFont.SetShadow( sal_False ); 7101 rFont.SetOutline( sal_False ); 7102 setFont( rFont ); 7103 setTextLineColor( rFont.GetColor() ); 7104 setOverlineColor( rFont.GetColor() ); 7105 updateGraphicsState(); 7106 7107 long nOff = 1 + ((m_pReferenceDevice->mpFontEntry->mnLineHeight-24)/24); 7108 if( rFont.IsOutline() ) 7109 nOff++; 7110 rLayout.DrawBase() += Point( nOff, nOff ); 7111 drawLayout( rLayout, rText, bTextLines ); 7112 rLayout.DrawBase() -= Point( nOff, nOff ); 7113 7114 setFont( aSaveFont ); 7115 setTextLineColor( aSaveTextLineColor ); 7116 setOverlineColor( aSaveOverlineColor ); 7117 updateGraphicsState(); 7118 } 7119 7120 void PDFWriterImpl::drawVerticalGlyphs( 7121 const std::vector<PDFWriterImpl::PDFGlyph>& rGlyphs, 7122 OStringBuffer& rLine, 7123 const Point& rAlignOffset, 7124 const Matrix3& rRotScale, 7125 double fAngle, 7126 double fXScale, 7127 double fSkew, 7128 sal_Int32 nFontHeight ) 7129 { 7130 long nXOffset = 0; 7131 Point aCurPos( rGlyphs[0].m_aPos ); 7132 aCurPos = m_pReferenceDevice->PixelToLogic( aCurPos ); 7133 aCurPos += rAlignOffset; 7134 for( size_t i = 0; i < rGlyphs.size(); i++ ) 7135 { 7136 // have to emit each glyph on its own 7137 double fDeltaAngle = 0.0; 7138 double fYScale = 1.0; 7139 double fTempXScale = fXScale; 7140 double fSkewB = fSkew; 7141 double fSkewA = 0.0; 7142 7143 Point aDeltaPos; 7144 if( ( rGlyphs[i].m_nGlyphId & GF_ROTMASK ) == GF_ROTL ) 7145 { 7146 fDeltaAngle = M_PI/2.0; 7147 aDeltaPos.X() = m_pReferenceDevice->GetFontMetric().GetAscent(); 7148 aDeltaPos.Y() = (int)((double)m_pReferenceDevice->GetFontMetric().GetDescent() * fXScale); 7149 fYScale = fXScale; 7150 fTempXScale = 1.0; 7151 fSkewA = -fSkewB; 7152 fSkewB = 0.0; 7153 } 7154 else if( ( rGlyphs[i].m_nGlyphId & GF_ROTMASK ) == GF_ROTR ) 7155 { 7156 fDeltaAngle = -M_PI/2.0; 7157 aDeltaPos.X() = (int)((double)m_pReferenceDevice->GetFontMetric().GetDescent()*fXScale); 7158 aDeltaPos.Y() = -m_pReferenceDevice->GetFontMetric().GetAscent(); 7159 fYScale = fXScale; 7160 fTempXScale = 1.0; 7161 fSkewA = fSkewB; 7162 fSkewB = 0.0; 7163 } 7164 aDeltaPos += (m_pReferenceDevice->PixelToLogic( Point( (int)((double)nXOffset/fXScale), 0 ) ) - m_pReferenceDevice->PixelToLogic( Point() ) ); 7165 if( i < rGlyphs.size()-1 ) 7166 nXOffset += rGlyphs[i+1].m_aPos.Y() - rGlyphs[i].m_aPos.Y(); 7167 if( ! rGlyphs[i].m_nGlyphId ) 7168 continue; 7169 7170 aDeltaPos = rRotScale.transform( aDeltaPos ); 7171 7172 Matrix3 aMat; 7173 if( fSkewB != 0.0 || fSkewA != 0.0 ) 7174 aMat.skew( fSkewA, fSkewB ); 7175 aMat.scale( fTempXScale, fYScale ); 7176 aMat.rotate( fAngle+fDeltaAngle ); 7177 aMat.translate( aCurPos.X()+aDeltaPos.X(), aCurPos.Y()+aDeltaPos.Y() ); 7178 aMat.append( m_aPages.back(), rLine ); 7179 rLine.append( " Tm" ); 7180 if( i == 0 || rGlyphs[i-1].m_nMappedFontId != rGlyphs[i].m_nMappedFontId ) 7181 { 7182 rLine.append( " /F" ); 7183 rLine.append( rGlyphs[i].m_nMappedFontId ); 7184 rLine.append( ' ' ); 7185 m_aPages.back().appendMappedLength( nFontHeight, rLine, true ); 7186 rLine.append( " Tf" ); 7187 } 7188 rLine.append( "<" ); 7189 appendHex( rGlyphs[i].m_nMappedGlyphId, rLine ); 7190 rLine.append( ">Tj\n" ); 7191 } 7192 } 7193 7194 void PDFWriterImpl::drawHorizontalGlyphs( 7195 const std::vector<PDFWriterImpl::PDFGlyph>& rGlyphs, 7196 OStringBuffer& rLine, 7197 const Point& rAlignOffset, 7198 double fAngle, 7199 double fXScale, 7200 double fSkew, 7201 sal_Int32 nFontHeight, 7202 sal_Int32 nPixelFontHeight 7203 ) 7204 { 7205 // horizontal (= normal) case 7206 7207 // fill in run end indices 7208 // end is marked by index of the first glyph of the next run 7209 // a run is marked by same mapped font id and same Y position 7210 std::vector< sal_uInt32 > aRunEnds; 7211 aRunEnds.reserve( rGlyphs.size() ); 7212 for( size_t i = 1; i < rGlyphs.size(); i++ ) 7213 { 7214 if( rGlyphs[i].m_nMappedFontId != rGlyphs[i-1].m_nMappedFontId || 7215 rGlyphs[i].m_aPos.Y() != rGlyphs[i-1].m_aPos.Y() ) 7216 { 7217 aRunEnds.push_back(i); 7218 } 7219 } 7220 // last run ends at last glyph 7221 aRunEnds.push_back( rGlyphs.size() ); 7222 7223 // loop over runs of the same font 7224 sal_uInt32 nBeginRun = 0; 7225 for( size_t nRun = 0; nRun < aRunEnds.size(); nRun++ ) 7226 { 7227 // setup text matrix 7228 Point aCurPos = rGlyphs[nBeginRun].m_aPos; 7229 // back transformation to current coordinate system 7230 aCurPos = m_pReferenceDevice->PixelToLogic( aCurPos ); 7231 aCurPos += rAlignOffset; 7232 // the first run can be set with "Td" operator 7233 // subsequent use of that operator would move 7234 // the texline matrix relative to what was set before 7235 // making use of that would drive us into rounding issues 7236 Matrix3 aMat; 7237 if( nRun == 0 && fAngle == 0.0 && fXScale == 1.0 && fSkew == 0.0 ) 7238 { 7239 m_aPages.back().appendPoint( aCurPos, rLine, false ); 7240 rLine.append( " Td " ); 7241 } 7242 else 7243 { 7244 if( fSkew != 0.0 ) 7245 aMat.skew( 0.0, fSkew ); 7246 aMat.scale( fXScale, 1.0 ); 7247 aMat.rotate( fAngle ); 7248 aMat.translate( aCurPos.X(), aCurPos.Y() ); 7249 aMat.append( m_aPages.back(), rLine ); 7250 rLine.append( " Tm\n" ); 7251 } 7252 // set up correct font 7253 rLine.append( "/F" ); 7254 rLine.append( rGlyphs[nBeginRun].m_nMappedFontId ); 7255 rLine.append( ' ' ); 7256 m_aPages.back().appendMappedLength( nFontHeight, rLine, true ); 7257 rLine.append( " Tf" ); 7258 7259 // output glyphs using Tj or TJ 7260 OStringBuffer aKernedLine( 256 ), aUnkernedLine( 256 ); 7261 aKernedLine.append( "[<" ); 7262 aUnkernedLine.append( '<' ); 7263 appendHex( rGlyphs[nBeginRun].m_nMappedGlyphId, aKernedLine ); 7264 appendHex( rGlyphs[nBeginRun].m_nMappedGlyphId, aUnkernedLine ); 7265 7266 aMat.invert(); 7267 bool bNeedKern = false; 7268 for( sal_uInt32 nPos = nBeginRun+1; nPos < aRunEnds[nRun]; nPos++ ) 7269 { 7270 appendHex( rGlyphs[nPos].m_nMappedGlyphId, aUnkernedLine ); 7271 // check if default glyph positioning is sufficient 7272 const Point aThisPos = aMat.transform( rGlyphs[nPos].m_aPos ); 7273 const Point aPrevPos = aMat.transform( rGlyphs[nPos-1].m_aPos ); 7274 double fAdvance = aThisPos.X() - aPrevPos.X(); 7275 fAdvance *= 1000.0 / nPixelFontHeight; 7276 const sal_Int32 nAdjustment = (sal_Int32)(rGlyphs[nPos-1].m_nNativeWidth - fAdvance + 0.5); 7277 if( nAdjustment != 0 ) 7278 { 7279 // apply individual glyph positioning 7280 bNeedKern = true; 7281 aKernedLine.append( ">" ); 7282 aKernedLine.append( nAdjustment ); 7283 aKernedLine.append( "<" ); 7284 } 7285 appendHex( rGlyphs[nPos].m_nMappedGlyphId, aKernedLine ); 7286 } 7287 aKernedLine.append( ">]TJ\n" ); 7288 aUnkernedLine.append( ">Tj\n" ); 7289 rLine.append( bNeedKern ? aKernedLine : aUnkernedLine ); 7290 7291 // set beginning of next run 7292 nBeginRun = aRunEnds[nRun]; 7293 } 7294 } 7295 7296 void PDFWriterImpl::drawLayout( SalLayout& rLayout, const String& rText, bool bTextLines ) 7297 { 7298 // relief takes precedence over shadow (see outdev3.cxx) 7299 if( m_aCurrentPDFState.m_aFont.GetRelief() != RELIEF_NONE ) 7300 { 7301 drawRelief( rLayout, rText, bTextLines ); 7302 return; 7303 } 7304 else if( m_aCurrentPDFState.m_aFont.IsShadow() ) 7305 drawShadow( rLayout, rText, bTextLines ); 7306 7307 OStringBuffer aLine( 512 ); 7308 7309 const int nMaxGlyphs = 256; 7310 7311 sal_GlyphId pGlyphs[nMaxGlyphs]; 7312 sal_Int32 pGlyphWidths[nMaxGlyphs]; 7313 sal_uInt8 pMappedGlyphs[nMaxGlyphs]; 7314 sal_Int32 pMappedFontObjects[nMaxGlyphs]; 7315 std::vector<sal_Ucs> aUnicodes; 7316 aUnicodes.reserve( nMaxGlyphs ); 7317 sal_Int32 pUnicodesPerGlyph[nMaxGlyphs]; 7318 int pCharPosAry[nMaxGlyphs]; 7319 sal_Int32 nAdvanceWidths[nMaxGlyphs]; 7320 const ImplFontData* pFallbackFonts[nMaxGlyphs]; 7321 bool bVertical = m_aCurrentPDFState.m_aFont.IsVertical(); 7322 int nGlyphs; 7323 int nIndex = 0; 7324 int nMinCharPos = 0, nMaxCharPos = rText.Len()-1; 7325 double fXScale = 1.0; 7326 double fSkew = 0.0; 7327 sal_Int32 nPixelFontHeight = m_pReferenceDevice->mpFontEntry->maFontSelData.mnHeight; 7328 TextAlign eAlign = m_aCurrentPDFState.m_aFont.GetAlign(); 7329 7330 // transform font height back to current units 7331 // note: the layout calculates in outdevs device pixel !! 7332 sal_Int32 nFontHeight = m_pReferenceDevice->ImplDevicePixelToLogicHeight( nPixelFontHeight ); 7333 if( m_aCurrentPDFState.m_aFont.GetWidth() ) 7334 { 7335 Font aFont( m_aCurrentPDFState.m_aFont ); 7336 aFont.SetWidth( 0 ); 7337 FontMetric aMetric = m_pReferenceDevice->GetFontMetric( aFont ); 7338 if( aMetric.GetWidth() != m_aCurrentPDFState.m_aFont.GetWidth() ) 7339 { 7340 fXScale = 7341 (double)m_aCurrentPDFState.m_aFont.GetWidth() / 7342 (double)aMetric.GetWidth(); 7343 } 7344 // force state before GetFontMetric 7345 m_pReferenceDevice->ImplNewFont(); 7346 } 7347 7348 // perform artificial italics if necessary 7349 if( ( m_aCurrentPDFState.m_aFont.GetItalic() == ITALIC_NORMAL || 7350 m_aCurrentPDFState.m_aFont.GetItalic() == ITALIC_OBLIQUE ) && 7351 !( m_pReferenceDevice->mpFontEntry->maFontSelData.mpFontData->GetSlant() == ITALIC_NORMAL || 7352 m_pReferenceDevice->mpFontEntry->maFontSelData.mpFontData->GetSlant() == ITALIC_OBLIQUE ) 7353 ) 7354 { 7355 fSkew = M_PI/12.0; 7356 } 7357 7358 // if the mapmode is distorted we need to adjust for that also 7359 if( m_aCurrentPDFState.m_aMapMode.GetScaleX() != m_aCurrentPDFState.m_aMapMode.GetScaleY() ) 7360 { 7361 fXScale *= double(m_aCurrentPDFState.m_aMapMode.GetScaleX()) / double(m_aCurrentPDFState.m_aMapMode.GetScaleY()); 7362 } 7363 7364 int nAngle = m_aCurrentPDFState.m_aFont.GetOrientation(); 7365 // normalize angles 7366 while( nAngle < 0 ) 7367 nAngle += 3600; 7368 nAngle = nAngle % 3600; 7369 double fAngle = (double)nAngle * M_PI / 1800.0; 7370 7371 Matrix3 aRotScale; 7372 aRotScale.scale( fXScale, 1.0 ); 7373 if( fAngle != 0.0 ) 7374 aRotScale.rotate( -fAngle ); 7375 7376 bool bPop = false; 7377 bool bABold = false; 7378 // artificial bold necessary ? 7379 if( m_pReferenceDevice->mpFontEntry->maFontSelData.mpFontData->GetWeight() <= WEIGHT_MEDIUM && 7380 m_pReferenceDevice->mpFontEntry->maFontSelData.GetWeight() > WEIGHT_MEDIUM ) 7381 { 7382 if( ! bPop ) 7383 aLine.append( "q " ); 7384 bPop = true; 7385 bABold = true; 7386 } 7387 // setup text colors (if necessary) 7388 Color aStrokeColor( COL_TRANSPARENT ); 7389 Color aNonStrokeColor( COL_TRANSPARENT ); 7390 7391 if( m_aCurrentPDFState.m_aFont.IsOutline() ) 7392 { 7393 aStrokeColor = m_aCurrentPDFState.m_aFont.GetColor(); 7394 aNonStrokeColor = Color( COL_WHITE ); 7395 } 7396 else 7397 aNonStrokeColor = m_aCurrentPDFState.m_aFont.GetColor(); 7398 if( bABold ) 7399 aStrokeColor = m_aCurrentPDFState.m_aFont.GetColor(); 7400 7401 if( aStrokeColor != Color( COL_TRANSPARENT ) && aStrokeColor != m_aCurrentPDFState.m_aLineColor ) 7402 { 7403 if( ! bPop ) 7404 aLine.append( "q " ); 7405 bPop = true; 7406 appendStrokingColor( aStrokeColor, aLine ); 7407 aLine.append( "\n" ); 7408 } 7409 if( aNonStrokeColor != Color( COL_TRANSPARENT ) && aNonStrokeColor != m_aCurrentPDFState.m_aFillColor ) 7410 { 7411 if( ! bPop ) 7412 aLine.append( "q " ); 7413 bPop = true; 7414 appendNonStrokingColor( aNonStrokeColor, aLine ); 7415 aLine.append( "\n" ); 7416 } 7417 7418 // begin text object 7419 aLine.append( "BT\n" ); 7420 // outline attribute ? 7421 if( m_aCurrentPDFState.m_aFont.IsOutline() || bABold ) 7422 { 7423 // set correct text mode, set stroke width 7424 aLine.append( "2 Tr " ); // fill, then stroke 7425 7426 if( m_aCurrentPDFState.m_aFont.IsOutline() ) 7427 { 7428 // unclear what to do in case of outline and artificial bold 7429 // for the time being outline wins 7430 aLine.append( "0.25 w \n" ); 7431 } 7432 else 7433 { 7434 double fW = (double)m_aCurrentPDFState.m_aFont.GetHeight() / 30.0; 7435 m_aPages.back().appendMappedLength( fW, aLine ); 7436 aLine.append ( " w\n" ); 7437 } 7438 } 7439 7440 FontMetric aRefDevFontMetric = m_pReferenceDevice->GetFontMetric(); 7441 7442 // collect the glyphs into a single array 7443 const int nTmpMaxGlyphs = rLayout.GetOrientation() ? 1 : nMaxGlyphs; // #i97991# temporary workaround for #i87686# 7444 std::vector< PDFGlyph > aGlyphs; 7445 aGlyphs.reserve( nTmpMaxGlyphs ); 7446 // first get all the glyphs and register them; coordinates still in Pixel 7447 Point aGNGlyphPos; 7448 while( (nGlyphs = rLayout.GetNextGlyphs( nTmpMaxGlyphs, pGlyphs, aGNGlyphPos, nIndex, nAdvanceWidths, pCharPosAry )) != 0 ) 7449 { 7450 aUnicodes.clear(); 7451 for( int i = 0; i < nGlyphs; i++ ) 7452 { 7453 pFallbackFonts[i] = rLayout.GetFallbackFontData( pGlyphs[i] ); 7454 7455 // default case: 1 glyph is one unicode 7456 pUnicodesPerGlyph[i] = 1; 7457 if( (pGlyphs[i] & GF_ISCHAR) ) 7458 { 7459 aUnicodes.push_back( static_cast<sal_Ucs>(pGlyphs[i] & GF_IDXMASK) ); 7460 } 7461 else if( pCharPosAry[i] >= nMinCharPos && pCharPosAry[i] <= nMaxCharPos ) 7462 { 7463 int nChars = 1; 7464 aUnicodes.push_back( rText.GetChar( sal::static_int_cast<xub_StrLen>(pCharPosAry[i]) ) ); 7465 pUnicodesPerGlyph[i] = 1; 7466 // try to handle ligatures and such 7467 if( i < nGlyphs-1 ) 7468 { 7469 nChars = pCharPosAry[i+1] - pCharPosAry[i]; 7470 // #i115618# fix for simple RTL+CTL cases 7471 // TODO: sanitize for RTL ligatures, more complex CTL, etc. 7472 if( nChars < 0 ) 7473 nChars = -nChars; 7474 else if( nChars == 0 ) 7475 nChars = 1; 7476 pUnicodesPerGlyph[i] = nChars; 7477 for( int n = 1; n < nChars; n++ ) 7478 aUnicodes.push_back( rText.GetChar( sal::static_int_cast<xub_StrLen>(pCharPosAry[i]+n) ) ); 7479 } 7480 // #i36691# hack that is needed because currently the pGlyphs[] 7481 // argument is ignored for embeddable fonts and so the layout 7482 // engine's glyph work is ignored (i.e. char mirroring) 7483 // TODO: a real solution would be to map the layout engine's 7484 // glyphid (i.e. FreeType's synthetic glyphid for a Type1 font) 7485 // back to unicode and then to embeddable font's encoding 7486 if( getReferenceDevice()->GetLayoutMode() & TEXT_LAYOUT_BIDI_RTL ) 7487 { 7488 size_t nI = aUnicodes.size()-1; 7489 for( int n = 0; n < nChars; n++, nI-- ) 7490 aUnicodes[nI] = static_cast<sal_Ucs>(GetMirroredChar(aUnicodes[nI])); 7491 } 7492 } 7493 else 7494 aUnicodes.push_back( 0 ); 7495 // note: in case of ctl one character may result 7496 // in multiple glyphs. The current SalLayout 7497 // implementations set -1 then to indicate that no direct 7498 // mapping is possible 7499 } 7500 7501 registerGlyphs( nGlyphs, pGlyphs, pGlyphWidths, &aUnicodes[0], pUnicodesPerGlyph, pMappedGlyphs, pMappedFontObjects, pFallbackFonts ); 7502 7503 for( int i = 0; i < nGlyphs; i++ ) 7504 { 7505 aGlyphs.push_back( PDFGlyph( aGNGlyphPos, 7506 pGlyphWidths[i], 7507 pGlyphs[i], 7508 pMappedFontObjects[i], 7509 pMappedGlyphs[i] ) ); 7510 if( bVertical ) 7511 aGNGlyphPos.Y() += nAdvanceWidths[i]/rLayout.GetUnitsPerPixel(); 7512 else 7513 aGNGlyphPos.X() += nAdvanceWidths[i]/rLayout.GetUnitsPerPixel(); 7514 } 7515 } 7516 7517 Point aAlignOffset; 7518 if ( eAlign == ALIGN_BOTTOM ) 7519 aAlignOffset.Y() -= aRefDevFontMetric.GetDescent(); 7520 else if ( eAlign == ALIGN_TOP ) 7521 aAlignOffset.Y() += aRefDevFontMetric.GetAscent(); 7522 if( aAlignOffset.X() || aAlignOffset.Y() ) 7523 aAlignOffset = aRotScale.transform( aAlignOffset ); 7524 7525 /* #159153# do not emit an empty glyph vector; this can happen if e.g. the original 7526 string contained only on of the UTF16 BOMs 7527 */ 7528 if( ! aGlyphs.empty() ) 7529 { 7530 if( bVertical ) 7531 drawVerticalGlyphs( aGlyphs, aLine, aAlignOffset, aRotScale, fAngle, fXScale, fSkew, nFontHeight ); 7532 else 7533 drawHorizontalGlyphs( aGlyphs, aLine, aAlignOffset, fAngle, fXScale, fSkew, nFontHeight, nPixelFontHeight ); 7534 } 7535 7536 // end textobject 7537 aLine.append( "ET\n" ); 7538 if( bPop ) 7539 aLine.append( "Q\n" ); 7540 7541 writeBuffer( aLine.getStr(), aLine.getLength() ); 7542 7543 // draw eventual textlines 7544 FontStrikeout eStrikeout = m_aCurrentPDFState.m_aFont.GetStrikeout(); 7545 FontUnderline eUnderline = m_aCurrentPDFState.m_aFont.GetUnderline(); 7546 FontUnderline eOverline = m_aCurrentPDFState.m_aFont.GetOverline(); 7547 if( bTextLines && 7548 ( 7549 ( eUnderline != UNDERLINE_NONE && eUnderline != UNDERLINE_DONTKNOW ) || 7550 ( eOverline != UNDERLINE_NONE && eOverline != UNDERLINE_DONTKNOW ) || 7551 ( eStrikeout != STRIKEOUT_NONE && eStrikeout != STRIKEOUT_DONTKNOW ) 7552 ) 7553 ) 7554 { 7555 sal_Bool bUnderlineAbove = OutputDevice::ImplIsUnderlineAbove( m_aCurrentPDFState.m_aFont ); 7556 if( m_aCurrentPDFState.m_aFont.IsWordLineMode() ) 7557 { 7558 Point aPos, aStartPt; 7559 sal_Int32 nWidth = 0, nAdvance=0; 7560 for( int nStart = 0;;) 7561 { 7562 sal_GlyphId nGlyphIndex; 7563 if( !rLayout.GetNextGlyphs( 1, &nGlyphIndex, aPos, nStart, &nAdvance ) ) 7564 break; 7565 7566 if( !rLayout.IsSpacingGlyph( nGlyphIndex ) ) 7567 { 7568 if( !nWidth ) 7569 aStartPt = aPos; 7570 7571 nWidth += nAdvance; 7572 } 7573 else if( nWidth > 0 ) 7574 { 7575 drawTextLine( m_pReferenceDevice->PixelToLogic( aStartPt ), 7576 m_pReferenceDevice->ImplDevicePixelToLogicWidth( nWidth ), 7577 eStrikeout, eUnderline, eOverline, bUnderlineAbove ); 7578 nWidth = 0; 7579 } 7580 } 7581 7582 if( nWidth > 0 ) 7583 { 7584 drawTextLine( m_pReferenceDevice->PixelToLogic( aStartPt ), 7585 m_pReferenceDevice->ImplDevicePixelToLogicWidth( nWidth ), 7586 eStrikeout, eUnderline, eOverline, bUnderlineAbove ); 7587 } 7588 } 7589 else 7590 { 7591 Point aStartPt = rLayout.GetDrawPosition(); 7592 int nWidth = rLayout.GetTextWidth() / rLayout.GetUnitsPerPixel(); 7593 drawTextLine( m_pReferenceDevice->PixelToLogic( aStartPt ), 7594 m_pReferenceDevice->ImplDevicePixelToLogicWidth( nWidth ), 7595 eStrikeout, eUnderline, eOverline, bUnderlineAbove ); 7596 } 7597 } 7598 7599 // write eventual emphasis marks 7600 if( m_aCurrentPDFState.m_aFont.GetEmphasisMark() & EMPHASISMARK_STYLE ) 7601 { 7602 PolyPolygon aEmphPoly; 7603 Rectangle aEmphRect1; 7604 Rectangle aEmphRect2; 7605 long nEmphYOff; 7606 long nEmphWidth; 7607 long nEmphHeight; 7608 sal_Bool bEmphPolyLine; 7609 FontEmphasisMark nEmphMark; 7610 7611 push( PUSH_ALL ); 7612 7613 aLine.setLength( 0 ); 7614 aLine.append( "q\n" ); 7615 7616 nEmphMark = m_pReferenceDevice->ImplGetEmphasisMarkStyle( m_aCurrentPDFState.m_aFont ); 7617 if ( nEmphMark & EMPHASISMARK_POS_BELOW ) 7618 nEmphHeight = m_pReferenceDevice->mnEmphasisDescent; 7619 else 7620 nEmphHeight = m_pReferenceDevice->mnEmphasisAscent; 7621 m_pReferenceDevice->ImplGetEmphasisMark( aEmphPoly, 7622 bEmphPolyLine, 7623 aEmphRect1, 7624 aEmphRect2, 7625 nEmphYOff, 7626 nEmphWidth, 7627 nEmphMark, 7628 m_pReferenceDevice->ImplDevicePixelToLogicWidth(nEmphHeight), 7629 m_pReferenceDevice->mpFontEntry->mnOrientation ); 7630 if ( bEmphPolyLine ) 7631 { 7632 setLineColor( m_aCurrentPDFState.m_aFont.GetColor() ); 7633 setFillColor( Color( COL_TRANSPARENT ) ); 7634 } 7635 else 7636 { 7637 setFillColor( m_aCurrentPDFState.m_aFont.GetColor() ); 7638 setLineColor( Color( COL_TRANSPARENT ) ); 7639 } 7640 writeBuffer( aLine.getStr(), aLine.getLength() ); 7641 7642 Point aOffset = Point(0,0); 7643 7644 if ( nEmphMark & EMPHASISMARK_POS_BELOW ) 7645 aOffset.Y() += m_pReferenceDevice->mpFontEntry->maMetric.mnDescent + nEmphYOff; 7646 else 7647 aOffset.Y() -= m_pReferenceDevice->mpFontEntry->maMetric.mnAscent + nEmphYOff; 7648 7649 long nEmphWidth2 = nEmphWidth / 2; 7650 long nEmphHeight2 = nEmphHeight / 2; 7651 aOffset += Point( nEmphWidth2, nEmphHeight2 ); 7652 7653 if ( eAlign == ALIGN_BOTTOM ) 7654 aOffset.Y() -= m_pReferenceDevice->mpFontEntry->maMetric.mnDescent; 7655 else if ( eAlign == ALIGN_TOP ) 7656 aOffset.Y() += m_pReferenceDevice->mpFontEntry->maMetric.mnAscent; 7657 7658 for( int nStart = 0;;) 7659 { 7660 Point aPos; 7661 sal_GlyphId nGlyphIndex; 7662 sal_Int32 nAdvance; 7663 if( !rLayout.GetNextGlyphs( 1, &nGlyphIndex, aPos, nStart, &nAdvance ) ) 7664 break; 7665 7666 if( !rLayout.IsSpacingGlyph( nGlyphIndex ) ) 7667 { 7668 Point aAdjOffset = aOffset; 7669 aAdjOffset.X() += (nAdvance - nEmphWidth) / 2; 7670 aAdjOffset = aRotScale.transform( aAdjOffset ); 7671 7672 aAdjOffset -= Point( nEmphWidth2, nEmphHeight2 ); 7673 7674 aPos += aAdjOffset; 7675 aPos = m_pReferenceDevice->PixelToLogic( aPos ); 7676 drawEmphasisMark( aPos.X(), aPos.Y(), 7677 aEmphPoly, bEmphPolyLine, 7678 aEmphRect1, aEmphRect2 ); 7679 } 7680 } 7681 7682 writeBuffer( "Q\n", 2 ); 7683 pop(); 7684 } 7685 7686 } 7687 7688 void PDFWriterImpl::drawEmphasisMark( long nX, long nY, 7689 const PolyPolygon& rPolyPoly, sal_Bool bPolyLine, 7690 const Rectangle& rRect1, const Rectangle& rRect2 ) 7691 { 7692 // TODO: pass nWidth as width of this mark 7693 // long nWidth = 0; 7694 7695 if ( rPolyPoly.Count() ) 7696 { 7697 if ( bPolyLine ) 7698 { 7699 Polygon aPoly = rPolyPoly.GetObject( 0 ); 7700 aPoly.Move( nX, nY ); 7701 drawPolyLine( aPoly ); 7702 } 7703 else 7704 { 7705 PolyPolygon aPolyPoly = rPolyPoly; 7706 aPolyPoly.Move( nX, nY ); 7707 drawPolyPolygon( aPolyPoly ); 7708 } 7709 } 7710 7711 if ( !rRect1.IsEmpty() ) 7712 { 7713 Rectangle aRect( Point( nX+rRect1.Left(), 7714 nY+rRect1.Top() ), rRect1.GetSize() ); 7715 drawRectangle( aRect ); 7716 } 7717 7718 if ( !rRect2.IsEmpty() ) 7719 { 7720 Rectangle aRect( Point( nX+rRect2.Left(), 7721 nY+rRect2.Top() ), rRect2.GetSize() ); 7722 7723 drawRectangle( aRect ); 7724 } 7725 } 7726 7727 void PDFWriterImpl::drawText( const Point& rPos, const String& rText, xub_StrLen nIndex, xub_StrLen nLen, bool bTextLines ) 7728 { 7729 MARK( "drawText" ); 7730 7731 updateGraphicsState(); 7732 7733 // get a layout from the OuputDevice's SalGraphics 7734 // this also enforces font substitution and sets the font on SalGraphics 7735 SalLayout* pLayout = m_pReferenceDevice->ImplLayout( rText, nIndex, nLen, rPos ); 7736 if( pLayout ) 7737 { 7738 drawLayout( *pLayout, rText, bTextLines ); 7739 pLayout->Release(); 7740 } 7741 } 7742 7743 void PDFWriterImpl::drawTextArray( const Point& rPos, const String& rText, const sal_Int32* pDXArray, xub_StrLen nIndex, xub_StrLen nLen, bool bTextLines ) 7744 { 7745 MARK( "drawText with array" ); 7746 7747 updateGraphicsState(); 7748 7749 // get a layout from the OuputDevice's SalGraphics 7750 // this also enforces font substitution and sets the font on SalGraphics 7751 SalLayout* pLayout = m_pReferenceDevice->ImplLayout( rText, nIndex, nLen, rPos, 0, pDXArray ); 7752 if( pLayout ) 7753 { 7754 drawLayout( *pLayout, rText, bTextLines ); 7755 pLayout->Release(); 7756 } 7757 } 7758 7759 void PDFWriterImpl::drawStretchText( const Point& rPos, sal_uLong nWidth, const String& rText, xub_StrLen nIndex, xub_StrLen nLen, bool bTextLines ) 7760 { 7761 MARK( "drawStretchText" ); 7762 7763 updateGraphicsState(); 7764 7765 // get a layout from the OuputDevice's SalGraphics 7766 // this also enforces font substitution and sets the font on SalGraphics 7767 SalLayout* pLayout = m_pReferenceDevice->ImplLayout( rText, nIndex, nLen, rPos, nWidth ); 7768 if( pLayout ) 7769 { 7770 drawLayout( *pLayout, rText, bTextLines ); 7771 pLayout->Release(); 7772 } 7773 } 7774 7775 void PDFWriterImpl::drawText( const Rectangle& rRect, const String& rOrigStr, sal_uInt16 nStyle, bool bTextLines ) 7776 { 7777 long nWidth = rRect.GetWidth(); 7778 long nHeight = rRect.GetHeight(); 7779 7780 if ( nWidth <= 0 || nHeight <= 0 ) 7781 return; 7782 7783 MARK( "drawText with rectangle" ); 7784 7785 updateGraphicsState(); 7786 7787 // clip with rectangle 7788 OStringBuffer aLine; 7789 aLine.append( "q " ); 7790 m_aPages.back().appendRect( rRect, aLine ); 7791 aLine.append( " W* n\n" ); 7792 writeBuffer( aLine.getStr(), aLine.getLength() ); 7793 7794 // if disabled text is needed, put in here 7795 7796 Point aPos = rRect.TopLeft(); 7797 7798 long nTextHeight = m_pReferenceDevice->GetTextHeight(); 7799 xub_StrLen nMnemonicPos = STRING_NOTFOUND; 7800 7801 String aStr = rOrigStr; 7802 if ( nStyle & TEXT_DRAW_MNEMONIC ) 7803 aStr = m_pReferenceDevice->GetNonMnemonicString( aStr, nMnemonicPos ); 7804 7805 // multiline text 7806 if ( nStyle & TEXT_DRAW_MULTILINE ) 7807 { 7808 XubString aLastLine; 7809 ImplMultiTextLineInfo aMultiLineInfo; 7810 ImplTextLineInfo* pLineInfo; 7811 long nMaxTextWidth; 7812 xub_StrLen i; 7813 xub_StrLen nLines; 7814 xub_StrLen nFormatLines; 7815 7816 if ( nTextHeight ) 7817 { 7818 ::vcl::DefaultTextLayout aLayout( *m_pReferenceDevice ); 7819 nMaxTextWidth = OutputDevice::ImplGetTextLines( aMultiLineInfo, nWidth, aStr, nStyle, aLayout ); 7820 nLines = (xub_StrLen)(nHeight/nTextHeight); 7821 nFormatLines = aMultiLineInfo.Count(); 7822 if ( !nLines ) 7823 nLines = 1; 7824 if ( nFormatLines > nLines ) 7825 { 7826 if ( nStyle & TEXT_DRAW_ENDELLIPSIS ) 7827 { 7828 // handle last line 7829 nFormatLines = nLines-1; 7830 7831 pLineInfo = aMultiLineInfo.GetLine( nFormatLines ); 7832 aLastLine = aStr.Copy( pLineInfo->GetIndex() ); 7833 aLastLine.ConvertLineEnd( LINEEND_LF ); 7834 // replace line feed by space 7835 xub_StrLen nLastLineLen = aLastLine.Len(); 7836 for ( i = 0; i < nLastLineLen; i++ ) 7837 { 7838 if ( aLastLine.GetChar( i ) == _LF ) 7839 aLastLine.SetChar( i, ' ' ); 7840 } 7841 aLastLine = m_pReferenceDevice->GetEllipsisString( aLastLine, nWidth, nStyle ); 7842 nStyle &= ~(TEXT_DRAW_VCENTER | TEXT_DRAW_BOTTOM); 7843 nStyle |= TEXT_DRAW_TOP; 7844 } 7845 } 7846 7847 // vertical alignment 7848 if ( nStyle & TEXT_DRAW_BOTTOM ) 7849 aPos.Y() += nHeight-(nFormatLines*nTextHeight); 7850 else if ( nStyle & TEXT_DRAW_VCENTER ) 7851 aPos.Y() += (nHeight-(nFormatLines*nTextHeight))/2; 7852 7853 // draw all lines excluding the last 7854 for ( i = 0; i < nFormatLines; i++ ) 7855 { 7856 pLineInfo = aMultiLineInfo.GetLine( i ); 7857 if ( nStyle & TEXT_DRAW_RIGHT ) 7858 aPos.X() += nWidth-pLineInfo->GetWidth(); 7859 else if ( nStyle & TEXT_DRAW_CENTER ) 7860 aPos.X() += (nWidth-pLineInfo->GetWidth())/2; 7861 xub_StrLen nIndex = pLineInfo->GetIndex(); 7862 xub_StrLen nLineLen = pLineInfo->GetLen(); 7863 drawText( aPos, aStr, nIndex, nLineLen, bTextLines ); 7864 // mnemonics should not appear in documents, 7865 // if the need arises, put them in here 7866 aPos.Y() += nTextHeight; 7867 aPos.X() = rRect.Left(); 7868 } 7869 7870 7871 // output last line left adjusted since it was shortened 7872 if ( aLastLine.Len() ) 7873 drawText( aPos, aLastLine, 0, STRING_LEN, bTextLines ); 7874 } 7875 } 7876 else 7877 { 7878 long nTextWidth = m_pReferenceDevice->GetTextWidth( aStr ); 7879 7880 // Evt. Text kuerzen 7881 if ( nTextWidth > nWidth ) 7882 { 7883 if ( nStyle & (TEXT_DRAW_ENDELLIPSIS | TEXT_DRAW_PATHELLIPSIS | TEXT_DRAW_NEWSELLIPSIS) ) 7884 { 7885 aStr = m_pReferenceDevice->GetEllipsisString( aStr, nWidth, nStyle ); 7886 nStyle &= ~(TEXT_DRAW_CENTER | TEXT_DRAW_RIGHT); 7887 nStyle |= TEXT_DRAW_LEFT; 7888 nTextWidth = m_pReferenceDevice->GetTextWidth( aStr ); 7889 } 7890 } 7891 7892 // vertical alignment 7893 if ( nStyle & TEXT_DRAW_RIGHT ) 7894 aPos.X() += nWidth-nTextWidth; 7895 else if ( nStyle & TEXT_DRAW_CENTER ) 7896 aPos.X() += (nWidth-nTextWidth)/2; 7897 7898 if ( nStyle & TEXT_DRAW_BOTTOM ) 7899 aPos.Y() += nHeight-nTextHeight; 7900 else if ( nStyle & TEXT_DRAW_VCENTER ) 7901 aPos.Y() += (nHeight-nTextHeight)/2; 7902 7903 // mnemonics should be inserted here if the need arises 7904 7905 // draw the actual text 7906 drawText( aPos, aStr, 0, STRING_LEN, bTextLines ); 7907 } 7908 7909 // reset clip region to original value 7910 aLine.setLength( 0 ); 7911 aLine.append( "Q\n" ); 7912 writeBuffer( aLine.getStr(), aLine.getLength() ); 7913 } 7914 7915 void PDFWriterImpl::drawLine( const Point& rStart, const Point& rStop ) 7916 { 7917 MARK( "drawLine" ); 7918 7919 updateGraphicsState(); 7920 7921 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) ) 7922 return; 7923 7924 OStringBuffer aLine; 7925 m_aPages.back().appendPoint( rStart, aLine ); 7926 aLine.append( " m " ); 7927 m_aPages.back().appendPoint( rStop, aLine ); 7928 aLine.append( " l S\n" ); 7929 7930 writeBuffer( aLine.getStr(), aLine.getLength() ); 7931 } 7932 7933 void PDFWriterImpl::drawLine( const Point& rStart, const Point& rStop, const LineInfo& rInfo ) 7934 { 7935 MARK( "drawLine with LineInfo" ); 7936 updateGraphicsState(); 7937 7938 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) ) 7939 return; 7940 7941 if( rInfo.GetStyle() == LINE_SOLID && rInfo.GetWidth() < 2 ) 7942 { 7943 drawLine( rStart, rStop ); 7944 return; 7945 } 7946 7947 OStringBuffer aLine; 7948 7949 aLine.append( "q " ); 7950 if( m_aPages.back().appendLineInfo( rInfo, aLine ) ) 7951 { 7952 m_aPages.back().appendPoint( rStart, aLine ); 7953 aLine.append( " m " ); 7954 m_aPages.back().appendPoint( rStop, aLine ); 7955 aLine.append( " l S Q\n" ); 7956 7957 writeBuffer( aLine.getStr(), aLine.getLength() ); 7958 } 7959 else 7960 { 7961 PDFWriter::ExtLineInfo aInfo; 7962 convertLineInfoToExtLineInfo( rInfo, aInfo ); 7963 Point aPolyPoints[2] = { rStart, rStop }; 7964 Polygon aPoly( 2, aPolyPoints ); 7965 drawPolyLine( aPoly, aInfo ); 7966 } 7967 } 7968 7969 void PDFWriterImpl::drawWaveLine( const Point& rStart, const Point& rStop, sal_Int32 nDelta, sal_Int32 nLineWidth ) 7970 { 7971 Point aDiff( rStop-rStart ); 7972 double fLen = sqrt( (double)(aDiff.X()*aDiff.X() + aDiff.Y()*aDiff.Y()) ); 7973 if( fLen < 1.0 ) 7974 return; 7975 7976 MARK( "drawWaveLine" ); 7977 updateGraphicsState(); 7978 7979 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) ) 7980 return; 7981 7982 OStringBuffer aLine( 512 ); 7983 aLine.append( "q " ); 7984 m_aPages.back().appendMappedLength( nLineWidth, aLine, true ); 7985 aLine.append( " w " ); 7986 7987 appendDouble( (double)aDiff.X()/fLen, aLine ); 7988 aLine.append( ' ' ); 7989 appendDouble( -(double)aDiff.Y()/fLen, aLine ); 7990 aLine.append( ' ' ); 7991 appendDouble( (double)aDiff.Y()/fLen, aLine ); 7992 aLine.append( ' ' ); 7993 appendDouble( (double)aDiff.X()/fLen, aLine ); 7994 aLine.append( ' ' ); 7995 m_aPages.back().appendPoint( rStart, aLine ); 7996 aLine.append( " cm " ); 7997 m_aPages.back().appendWaveLine( (sal_Int32)fLen, 0, nDelta, aLine ); 7998 aLine.append( "Q\n" ); 7999 writeBuffer( aLine.getStr(), aLine.getLength() ); 8000 } 8001 8002 #define WCONV( x ) m_pReferenceDevice->ImplDevicePixelToLogicWidth( x ) 8003 #define HCONV( x ) m_pReferenceDevice->ImplDevicePixelToLogicHeight( x ) 8004 8005 void PDFWriterImpl::drawWaveTextLine( OStringBuffer& aLine, long nWidth, FontUnderline eTextLine, Color aColor, bool bIsAbove ) 8006 { 8007 // note: units in pFontEntry are ref device pixel 8008 ImplFontEntry* pFontEntry = m_pReferenceDevice->mpFontEntry; 8009 long nLineHeight = 0; 8010 long nLinePos = 0; 8011 8012 appendStrokingColor( aColor, aLine ); 8013 aLine.append( "\n" ); 8014 8015 if ( bIsAbove ) 8016 { 8017 if ( !pFontEntry->maMetric.mnAboveWUnderlineSize ) 8018 m_pReferenceDevice->ImplInitAboveTextLineSize(); 8019 nLineHeight = HCONV( pFontEntry->maMetric.mnAboveWUnderlineSize ); 8020 nLinePos = HCONV( pFontEntry->maMetric.mnAboveWUnderlineOffset ); 8021 } 8022 else 8023 { 8024 if ( !pFontEntry->maMetric.mnWUnderlineSize ) 8025 m_pReferenceDevice->ImplInitTextLineSize(); 8026 nLineHeight = HCONV( pFontEntry->maMetric.mnWUnderlineSize ); 8027 nLinePos = HCONV( pFontEntry->maMetric.mnWUnderlineOffset ); 8028 } 8029 if ( (eTextLine == UNDERLINE_SMALLWAVE) && (nLineHeight > 3) ) 8030 nLineHeight = 3; 8031 8032 long nLineWidth = getReferenceDevice()->mnDPIX/450; 8033 if ( ! nLineWidth ) 8034 nLineWidth = 1; 8035 8036 if ( eTextLine == UNDERLINE_BOLDWAVE ) 8037 nLineWidth = 3*nLineWidth; 8038 8039 m_aPages.back().appendMappedLength( (sal_Int32)nLineWidth, aLine ); 8040 aLine.append( " w " ); 8041 8042 if ( eTextLine == UNDERLINE_DOUBLEWAVE ) 8043 { 8044 long nOrgLineHeight = nLineHeight; 8045 nLineHeight /= 3; 8046 if ( nLineHeight < 2 ) 8047 { 8048 if ( nOrgLineHeight > 1 ) 8049 nLineHeight = 2; 8050 else 8051 nLineHeight = 1; 8052 } 8053 long nLineDY = nOrgLineHeight-(nLineHeight*2); 8054 if ( nLineDY < nLineWidth ) 8055 nLineDY = nLineWidth; 8056 long nLineDY2 = nLineDY/2; 8057 if ( !nLineDY2 ) 8058 nLineDY2 = 1; 8059 8060 nLinePos -= nLineWidth-nLineDY2; 8061 8062 m_aPages.back().appendWaveLine( nWidth, -nLinePos, 2*nLineHeight, aLine ); 8063 8064 nLinePos += nLineWidth+nLineDY; 8065 m_aPages.back().appendWaveLine( nWidth, -nLinePos, 2*nLineHeight, aLine ); 8066 } 8067 else 8068 { 8069 if ( eTextLine != UNDERLINE_BOLDWAVE ) 8070 nLinePos -= nLineWidth/2; 8071 m_aPages.back().appendWaveLine( nWidth, -nLinePos, nLineHeight, aLine ); 8072 } 8073 } 8074 8075 void PDFWriterImpl::drawStraightTextLine( OStringBuffer& aLine, long nWidth, FontUnderline eTextLine, Color aColor, bool bIsAbove ) 8076 { 8077 // note: units in pFontEntry are ref device pixel 8078 ImplFontEntry* pFontEntry = m_pReferenceDevice->mpFontEntry; 8079 long nLineHeight = 0; 8080 long nLinePos = 0; 8081 long nLinePos2 = 0; 8082 8083 if ( eTextLine > UNDERLINE_BOLDWAVE ) 8084 eTextLine = UNDERLINE_SINGLE; 8085 8086 switch ( eTextLine ) 8087 { 8088 case UNDERLINE_SINGLE: 8089 case UNDERLINE_DOTTED: 8090 case UNDERLINE_DASH: 8091 case UNDERLINE_LONGDASH: 8092 case UNDERLINE_DASHDOT: 8093 case UNDERLINE_DASHDOTDOT: 8094 if ( bIsAbove ) 8095 { 8096 if ( !pFontEntry->maMetric.mnAboveUnderlineSize ) 8097 m_pReferenceDevice->ImplInitAboveTextLineSize(); 8098 nLineHeight = HCONV( pFontEntry->maMetric.mnAboveUnderlineSize ); 8099 nLinePos = HCONV( pFontEntry->maMetric.mnAboveUnderlineOffset ); 8100 } 8101 else 8102 { 8103 if ( !pFontEntry->maMetric.mnUnderlineSize ) 8104 m_pReferenceDevice->ImplInitTextLineSize(); 8105 nLineHeight = HCONV( pFontEntry->maMetric.mnUnderlineSize ); 8106 nLinePos = HCONV( pFontEntry->maMetric.mnUnderlineOffset ); 8107 } 8108 break; 8109 case UNDERLINE_BOLD: 8110 case UNDERLINE_BOLDDOTTED: 8111 case UNDERLINE_BOLDDASH: 8112 case UNDERLINE_BOLDLONGDASH: 8113 case UNDERLINE_BOLDDASHDOT: 8114 case UNDERLINE_BOLDDASHDOTDOT: 8115 if ( bIsAbove ) 8116 { 8117 if ( !pFontEntry->maMetric.mnAboveBUnderlineSize ) 8118 m_pReferenceDevice->ImplInitAboveTextLineSize(); 8119 nLineHeight = HCONV( pFontEntry->maMetric.mnAboveBUnderlineSize ); 8120 nLinePos = HCONV( pFontEntry->maMetric.mnAboveBUnderlineOffset ); 8121 } 8122 else 8123 { 8124 if ( !pFontEntry->maMetric.mnBUnderlineSize ) 8125 m_pReferenceDevice->ImplInitTextLineSize(); 8126 nLineHeight = HCONV( pFontEntry->maMetric.mnBUnderlineSize ); 8127 nLinePos = HCONV( pFontEntry->maMetric.mnBUnderlineOffset ); 8128 nLinePos += nLineHeight/2; 8129 } 8130 break; 8131 case UNDERLINE_DOUBLE: 8132 if ( bIsAbove ) 8133 { 8134 if ( !pFontEntry->maMetric.mnAboveDUnderlineSize ) 8135 m_pReferenceDevice->ImplInitAboveTextLineSize(); 8136 nLineHeight = HCONV( pFontEntry->maMetric.mnAboveDUnderlineSize ); 8137 nLinePos = HCONV( pFontEntry->maMetric.mnAboveDUnderlineOffset1 ); 8138 nLinePos2 = HCONV( pFontEntry->maMetric.mnAboveDUnderlineOffset2 ); 8139 } 8140 else 8141 { 8142 if ( !pFontEntry->maMetric.mnDUnderlineSize ) 8143 m_pReferenceDevice->ImplInitTextLineSize(); 8144 nLineHeight = HCONV( pFontEntry->maMetric.mnDUnderlineSize ); 8145 nLinePos = HCONV( pFontEntry->maMetric.mnDUnderlineOffset1 ); 8146 nLinePos2 = HCONV( pFontEntry->maMetric.mnDUnderlineOffset2 ); 8147 } 8148 default: 8149 break; 8150 } 8151 8152 if ( nLineHeight ) 8153 { 8154 m_aPages.back().appendMappedLength( (sal_Int32)nLineHeight, aLine, true ); 8155 aLine.append( " w " ); 8156 appendStrokingColor( aColor, aLine ); 8157 aLine.append( "\n" ); 8158 8159 switch ( eTextLine ) 8160 { 8161 case UNDERLINE_DOTTED: 8162 case UNDERLINE_BOLDDOTTED: 8163 aLine.append( "[ " ); 8164 m_aPages.back().appendMappedLength( (sal_Int32)nLineHeight, aLine, false ); 8165 aLine.append( " ] 0 d\n" ); 8166 break; 8167 case UNDERLINE_DASH: 8168 case UNDERLINE_LONGDASH: 8169 case UNDERLINE_BOLDDASH: 8170 case UNDERLINE_BOLDLONGDASH: 8171 { 8172 sal_Int32 nDashLength = 4*nLineHeight; 8173 sal_Int32 nVoidLength = 2*nLineHeight; 8174 if ( ( eTextLine == UNDERLINE_LONGDASH ) || ( eTextLine == UNDERLINE_BOLDLONGDASH ) ) 8175 nDashLength = 8*nLineHeight; 8176 8177 aLine.append( "[ " ); 8178 m_aPages.back().appendMappedLength( nDashLength, aLine, false ); 8179 aLine.append( ' ' ); 8180 m_aPages.back().appendMappedLength( nVoidLength, aLine, false ); 8181 aLine.append( " ] 0 d\n" ); 8182 } 8183 break; 8184 case UNDERLINE_DASHDOT: 8185 case UNDERLINE_BOLDDASHDOT: 8186 { 8187 sal_Int32 nDashLength = 4*nLineHeight; 8188 sal_Int32 nVoidLength = 2*nLineHeight; 8189 aLine.append( "[ " ); 8190 m_aPages.back().appendMappedLength( nDashLength, aLine, false ); 8191 aLine.append( ' ' ); 8192 m_aPages.back().appendMappedLength( nVoidLength, aLine, false ); 8193 aLine.append( ' ' ); 8194 m_aPages.back().appendMappedLength( (sal_Int32)nLineHeight, aLine, false ); 8195 aLine.append( ' ' ); 8196 m_aPages.back().appendMappedLength( nVoidLength, aLine, false ); 8197 aLine.append( " ] 0 d\n" ); 8198 } 8199 break; 8200 case UNDERLINE_DASHDOTDOT: 8201 case UNDERLINE_BOLDDASHDOTDOT: 8202 { 8203 sal_Int32 nDashLength = 4*nLineHeight; 8204 sal_Int32 nVoidLength = 2*nLineHeight; 8205 aLine.append( "[ " ); 8206 m_aPages.back().appendMappedLength( nDashLength, aLine, false ); 8207 aLine.append( ' ' ); 8208 m_aPages.back().appendMappedLength( nVoidLength, aLine, false ); 8209 aLine.append( ' ' ); 8210 m_aPages.back().appendMappedLength( (sal_Int32)nLineHeight, aLine, false ); 8211 aLine.append( ' ' ); 8212 m_aPages.back().appendMappedLength( nVoidLength, aLine, false ); 8213 aLine.append( ' ' ); 8214 m_aPages.back().appendMappedLength( (sal_Int32)nLineHeight, aLine, false ); 8215 aLine.append( ' ' ); 8216 m_aPages.back().appendMappedLength( nVoidLength, aLine, false ); 8217 aLine.append( " ] 0 d\n" ); 8218 } 8219 break; 8220 default: 8221 break; 8222 } 8223 8224 aLine.append( "0 " ); 8225 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos), aLine, true ); 8226 aLine.append( " m " ); 8227 m_aPages.back().appendMappedLength( (sal_Int32)nWidth, aLine, false ); 8228 aLine.append( ' ' ); 8229 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos), aLine, true ); 8230 aLine.append( " l S\n" ); 8231 if ( eTextLine == UNDERLINE_DOUBLE ) 8232 { 8233 aLine.append( "0 " ); 8234 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos2-nLineHeight), aLine, true ); 8235 aLine.append( " m " ); 8236 m_aPages.back().appendMappedLength( (sal_Int32)nWidth, aLine, false ); 8237 aLine.append( ' ' ); 8238 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos2-nLineHeight), aLine, true ); 8239 aLine.append( " l S\n" ); 8240 } 8241 } 8242 } 8243 8244 void PDFWriterImpl::drawStrikeoutLine( OStringBuffer& aLine, long nWidth, FontStrikeout eStrikeout, Color aColor ) 8245 { 8246 // note: units in pFontEntry are ref device pixel 8247 ImplFontEntry* pFontEntry = m_pReferenceDevice->mpFontEntry; 8248 long nLineHeight = 0; 8249 long nLinePos = 0; 8250 long nLinePos2 = 0; 8251 8252 if ( eStrikeout > STRIKEOUT_X ) 8253 eStrikeout = STRIKEOUT_SINGLE; 8254 8255 switch ( eStrikeout ) 8256 { 8257 case STRIKEOUT_SINGLE: 8258 if ( !pFontEntry->maMetric.mnStrikeoutSize ) 8259 m_pReferenceDevice->ImplInitTextLineSize(); 8260 nLineHeight = HCONV( pFontEntry->maMetric.mnStrikeoutSize ); 8261 nLinePos = HCONV( pFontEntry->maMetric.mnStrikeoutOffset ); 8262 break; 8263 case STRIKEOUT_BOLD: 8264 if ( !pFontEntry->maMetric.mnBStrikeoutSize ) 8265 m_pReferenceDevice->ImplInitTextLineSize(); 8266 nLineHeight = HCONV( pFontEntry->maMetric.mnBStrikeoutSize ); 8267 nLinePos = HCONV( pFontEntry->maMetric.mnBStrikeoutOffset ); 8268 break; 8269 case STRIKEOUT_DOUBLE: 8270 if ( !pFontEntry->maMetric.mnDStrikeoutSize ) 8271 m_pReferenceDevice->ImplInitTextLineSize(); 8272 nLineHeight = HCONV( pFontEntry->maMetric.mnDStrikeoutSize ); 8273 nLinePos = HCONV( pFontEntry->maMetric.mnDStrikeoutOffset1 ); 8274 nLinePos2 = HCONV( pFontEntry->maMetric.mnDStrikeoutOffset2 ); 8275 break; 8276 default: 8277 break; 8278 } 8279 8280 if ( nLineHeight ) 8281 { 8282 m_aPages.back().appendMappedLength( (sal_Int32)nLineHeight, aLine, true ); 8283 aLine.append( " w " ); 8284 appendStrokingColor( aColor, aLine ); 8285 aLine.append( "\n" ); 8286 8287 aLine.append( "0 " ); 8288 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos), aLine, true ); 8289 aLine.append( " m " ); 8290 m_aPages.back().appendMappedLength( (sal_Int32)nWidth, aLine, true ); 8291 aLine.append( ' ' ); 8292 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos), aLine, true ); 8293 aLine.append( " l S\n" ); 8294 8295 if ( eStrikeout == STRIKEOUT_DOUBLE ) 8296 { 8297 aLine.append( "0 " ); 8298 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos2-nLineHeight), aLine, true ); 8299 aLine.append( " m " ); 8300 m_aPages.back().appendMappedLength( (sal_Int32)nWidth, aLine, true ); 8301 aLine.append( ' ' ); 8302 m_aPages.back().appendMappedLength( (sal_Int32)(-nLinePos2-nLineHeight), aLine, true ); 8303 aLine.append( " l S\n" ); 8304 } 8305 } 8306 } 8307 8308 void PDFWriterImpl::drawStrikeoutChar( const Point& rPos, long nWidth, FontStrikeout eStrikeout ) 8309 { 8310 String aStrikeoutChar = String::CreateFromAscii( eStrikeout == STRIKEOUT_SLASH ? "/" : "X" ); 8311 String aStrikeout = aStrikeoutChar; 8312 while( m_pReferenceDevice->GetTextWidth( aStrikeout ) < nWidth ) 8313 aStrikeout.Append( aStrikeout ); 8314 8315 // do not get broader than nWidth modulo 1 character 8316 while( m_pReferenceDevice->GetTextWidth( aStrikeout ) >= nWidth ) 8317 aStrikeout.Erase( 0, 1 ); 8318 aStrikeout.Append( aStrikeoutChar ); 8319 sal_Bool bShadow = m_aCurrentPDFState.m_aFont.IsShadow(); 8320 if ( bShadow ) 8321 { 8322 Font aFont = m_aCurrentPDFState.m_aFont; 8323 aFont.SetShadow( sal_False ); 8324 setFont( aFont ); 8325 updateGraphicsState(); 8326 } 8327 8328 // strikeout string is left aligned non-CTL text 8329 sal_uLong nOrigTLM = m_pReferenceDevice->GetLayoutMode(); 8330 m_pReferenceDevice->SetLayoutMode( TEXT_LAYOUT_BIDI_STRONG|TEXT_LAYOUT_COMPLEX_DISABLED ); 8331 drawText( rPos, aStrikeout, 0, aStrikeout.Len(), false ); 8332 m_pReferenceDevice->SetLayoutMode( nOrigTLM ); 8333 8334 if ( bShadow ) 8335 { 8336 Font aFont = m_aCurrentPDFState.m_aFont; 8337 aFont.SetShadow( sal_True ); 8338 setFont( aFont ); 8339 updateGraphicsState(); 8340 } 8341 } 8342 8343 void PDFWriterImpl::drawTextLine( const Point& rPos, long nWidth, FontStrikeout eStrikeout, FontUnderline eUnderline, FontUnderline eOverline, bool bUnderlineAbove ) 8344 { 8345 if ( !nWidth || 8346 ( ((eStrikeout == STRIKEOUT_NONE)||(eStrikeout == STRIKEOUT_DONTKNOW)) && 8347 ((eUnderline == UNDERLINE_NONE)||(eUnderline == UNDERLINE_DONTKNOW)) && 8348 ((eOverline == UNDERLINE_NONE)||(eOverline == UNDERLINE_DONTKNOW)) ) ) 8349 return; 8350 8351 MARK( "drawTextLine" ); 8352 updateGraphicsState(); 8353 8354 // note: units in pFontEntry are ref device pixel 8355 ImplFontEntry* pFontEntry = m_pReferenceDevice->mpFontEntry; 8356 Color aUnderlineColor = m_aCurrentPDFState.m_aTextLineColor; 8357 Color aOverlineColor = m_aCurrentPDFState.m_aOverlineColor; 8358 Color aStrikeoutColor = m_aCurrentPDFState.m_aFont.GetColor(); 8359 bool bStrikeoutDone = false; 8360 bool bUnderlineDone = false; 8361 bool bOverlineDone = false; 8362 8363 if ( (eStrikeout == STRIKEOUT_SLASH) || (eStrikeout == STRIKEOUT_X) ) 8364 { 8365 drawStrikeoutChar( rPos, nWidth, eStrikeout ); 8366 bStrikeoutDone = true; 8367 } 8368 8369 Point aPos( rPos ); 8370 TextAlign eAlign = m_aCurrentPDFState.m_aFont.GetAlign(); 8371 if( eAlign == ALIGN_TOP ) 8372 aPos.Y() += HCONV( pFontEntry->maMetric.mnAscent ); 8373 else if( eAlign == ALIGN_BOTTOM ) 8374 aPos.Y() -= HCONV( pFontEntry->maMetric.mnDescent ); 8375 8376 OStringBuffer aLine( 512 ); 8377 // save GS 8378 aLine.append( "q " ); 8379 8380 // rotate and translate matrix 8381 double fAngle = (double)m_aCurrentPDFState.m_aFont.GetOrientation() * M_PI / 1800.0; 8382 Matrix3 aMat; 8383 aMat.rotate( fAngle ); 8384 aMat.translate( aPos.X(), aPos.Y() ); 8385 aMat.append( m_aPages.back(), aLine ); 8386 aLine.append( " cm\n" ); 8387 8388 if ( aUnderlineColor.GetTransparency() != 0 ) 8389 aUnderlineColor = aStrikeoutColor; 8390 8391 if ( (eUnderline == UNDERLINE_SMALLWAVE) || 8392 (eUnderline == UNDERLINE_WAVE) || 8393 (eUnderline == UNDERLINE_DOUBLEWAVE) || 8394 (eUnderline == UNDERLINE_BOLDWAVE) ) 8395 { 8396 drawWaveTextLine( aLine, nWidth, eUnderline, aUnderlineColor, bUnderlineAbove ); 8397 bUnderlineDone = true; 8398 } 8399 8400 if ( (eOverline == UNDERLINE_SMALLWAVE) || 8401 (eOverline == UNDERLINE_WAVE) || 8402 (eOverline == UNDERLINE_DOUBLEWAVE) || 8403 (eOverline == UNDERLINE_BOLDWAVE) ) 8404 { 8405 drawWaveTextLine( aLine, nWidth, eOverline, aOverlineColor, true ); 8406 bOverlineDone = true; 8407 } 8408 8409 if ( !bUnderlineDone ) 8410 { 8411 drawStraightTextLine( aLine, nWidth, eUnderline, aUnderlineColor, bUnderlineAbove ); 8412 } 8413 8414 if ( !bOverlineDone ) 8415 { 8416 drawStraightTextLine( aLine, nWidth, eOverline, aOverlineColor, true ); 8417 } 8418 8419 if ( !bStrikeoutDone ) 8420 { 8421 drawStrikeoutLine( aLine, nWidth, eStrikeout, aStrikeoutColor ); 8422 } 8423 8424 aLine.append( "Q\n" ); 8425 writeBuffer( aLine.getStr(), aLine.getLength() ); 8426 } 8427 8428 void PDFWriterImpl::drawPolygon( const Polygon& rPoly ) 8429 { 8430 MARK( "drawPolygon" ); 8431 8432 updateGraphicsState(); 8433 8434 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8435 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8436 return; 8437 8438 int nPoints = rPoly.GetSize(); 8439 OStringBuffer aLine( 20 * nPoints ); 8440 m_aPages.back().appendPolygon( rPoly, aLine ); 8441 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) && 8442 m_aGraphicsStack.front().m_aFillColor != Color( COL_TRANSPARENT ) ) 8443 aLine.append( "B*\n" ); 8444 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 8445 aLine.append( "S\n" ); 8446 else 8447 aLine.append( "f*\n" ); 8448 8449 writeBuffer( aLine.getStr(), aLine.getLength() ); 8450 } 8451 8452 void PDFWriterImpl::drawPolyPolygon( const PolyPolygon& rPolyPoly ) 8453 { 8454 MARK( "drawPolyPolygon" ); 8455 8456 updateGraphicsState(); 8457 8458 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8459 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8460 return; 8461 8462 int nPolygons = rPolyPoly.Count(); 8463 8464 OStringBuffer aLine( 40 * nPolygons ); 8465 m_aPages.back().appendPolyPolygon( rPolyPoly, aLine ); 8466 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) && 8467 m_aGraphicsStack.front().m_aFillColor != Color( COL_TRANSPARENT ) ) 8468 aLine.append( "B*\n" ); 8469 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 8470 aLine.append( "S\n" ); 8471 else 8472 aLine.append( "f*\n" ); 8473 8474 writeBuffer( aLine.getStr(), aLine.getLength() ); 8475 } 8476 8477 void PDFWriterImpl::drawTransparent( const PolyPolygon& rPolyPoly, sal_uInt32 nTransparentPercent ) 8478 { 8479 DBG_ASSERT( nTransparentPercent <= 100, "invalid alpha value" ); 8480 nTransparentPercent = nTransparentPercent % 100; 8481 8482 MARK( "drawTransparent" ); 8483 8484 updateGraphicsState(); 8485 8486 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8487 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8488 return; 8489 8490 if( m_bIsPDF_A1 || m_aContext.Version < PDFWriter::PDF_1_4 ) 8491 { 8492 m_aErrors.insert( m_bIsPDF_A1 ? 8493 PDFWriter::Warning_Transparency_Omitted_PDFA : 8494 PDFWriter::Warning_Transparency_Omitted_PDF13 ); 8495 8496 drawPolyPolygon( rPolyPoly ); 8497 return; 8498 } 8499 8500 // create XObject 8501 m_aTransparentObjects.push_back( TransparencyEmit() ); 8502 // FIXME: polygons with beziers may yield incorrect bound rect 8503 m_aTransparentObjects.back().m_aBoundRect = rPolyPoly.GetBoundRect(); 8504 // convert rectangle to default user space 8505 m_aPages.back().convertRect( m_aTransparentObjects.back().m_aBoundRect ); 8506 m_aTransparentObjects.back().m_nObject = createObject(); 8507 m_aTransparentObjects.back().m_nExtGStateObject = createObject(); 8508 m_aTransparentObjects.back().m_fAlpha = (double)(100-nTransparentPercent) / 100.0; 8509 m_aTransparentObjects.back().m_pContentStream = new SvMemoryStream( 256, 256 ); 8510 // create XObject's content stream 8511 OStringBuffer aContent( 256 ); 8512 m_aPages.back().appendPolyPolygon( rPolyPoly, aContent ); 8513 if( m_aCurrentPDFState.m_aLineColor != Color( COL_TRANSPARENT ) && 8514 m_aCurrentPDFState.m_aFillColor != Color( COL_TRANSPARENT ) ) 8515 aContent.append( " B*\n" ); 8516 else if( m_aCurrentPDFState.m_aLineColor != Color( COL_TRANSPARENT ) ) 8517 aContent.append( " S\n" ); 8518 else 8519 aContent.append( " f*\n" ); 8520 m_aTransparentObjects.back().m_pContentStream->Write( aContent.getStr(), aContent.getLength() ); 8521 8522 OStringBuffer aObjName( 16 ); 8523 aObjName.append( "Tr" ); 8524 aObjName.append( m_aTransparentObjects.back().m_nObject ); 8525 OString aTrName( aObjName.makeStringAndClear() ); 8526 aObjName.append( "EGS" ); 8527 aObjName.append( m_aTransparentObjects.back().m_nExtGStateObject ); 8528 OString aExtName( aObjName.makeStringAndClear() ); 8529 8530 OStringBuffer aLine( 80 ); 8531 // insert XObject 8532 aLine.append( "q /" ); 8533 aLine.append( aExtName ); 8534 aLine.append( " gs /" ); 8535 aLine.append( aTrName ); 8536 aLine.append( " Do Q\n" ); 8537 writeBuffer( aLine.getStr(), aLine.getLength() ); 8538 8539 pushResource( ResXObject, aTrName, m_aTransparentObjects.back().m_nObject ); 8540 pushResource( ResExtGState, aExtName, m_aTransparentObjects.back().m_nExtGStateObject ); 8541 } 8542 8543 void PDFWriterImpl::pushResource( ResourceKind eKind, const OString& rResource, sal_Int32 nObject ) 8544 { 8545 if( nObject >= 0 ) 8546 { 8547 switch( eKind ) 8548 { 8549 case ResXObject: 8550 m_aGlobalResourceDict.m_aXObjects[ rResource ] = nObject; 8551 if( ! m_aOutputStreams.empty() ) 8552 m_aOutputStreams.front().m_aResourceDict.m_aXObjects[ rResource ] = nObject; 8553 break; 8554 case ResExtGState: 8555 m_aGlobalResourceDict.m_aExtGStates[ rResource ] = nObject; 8556 if( ! m_aOutputStreams.empty() ) 8557 m_aOutputStreams.front().m_aResourceDict.m_aExtGStates[ rResource ] = nObject; 8558 break; 8559 case ResShading: 8560 m_aGlobalResourceDict.m_aShadings[ rResource ] = nObject; 8561 if( ! m_aOutputStreams.empty() ) 8562 m_aOutputStreams.front().m_aResourceDict.m_aShadings[ rResource ] = nObject; 8563 break; 8564 case ResPattern: 8565 m_aGlobalResourceDict.m_aPatterns[ rResource ] = nObject; 8566 if( ! m_aOutputStreams.empty() ) 8567 m_aOutputStreams.front().m_aResourceDict.m_aPatterns[ rResource ] = nObject; 8568 break; 8569 } 8570 } 8571 } 8572 8573 void PDFWriterImpl::beginRedirect( SvStream* pStream, const Rectangle& rTargetRect ) 8574 { 8575 push( PUSH_ALL ); 8576 8577 // force reemitting clip region 8578 clearClipRegion(); 8579 updateGraphicsState(); 8580 8581 m_aOutputStreams.push_front( StreamRedirect() ); 8582 m_aOutputStreams.front().m_pStream = pStream; 8583 m_aOutputStreams.front().m_aMapMode = m_aMapMode; 8584 8585 if( !rTargetRect.IsEmpty() ) 8586 { 8587 m_aOutputStreams.front().m_aTargetRect = 8588 lcl_convert( m_aGraphicsStack.front().m_aMapMode, 8589 m_aMapMode, 8590 getReferenceDevice(), 8591 rTargetRect ); 8592 Point aDelta = m_aOutputStreams.front().m_aTargetRect.BottomLeft(); 8593 long nPageHeight = pointToPixel(m_aPages[m_nCurrentPage].getHeight()); 8594 aDelta.Y() = -(nPageHeight - m_aOutputStreams.front().m_aTargetRect.Bottom()); 8595 m_aMapMode.SetOrigin( m_aMapMode.GetOrigin() + aDelta ); 8596 } 8597 8598 // setup graphics state for independent object stream 8599 8600 // force reemitting colors 8601 m_aCurrentPDFState.m_aLineColor = Color( COL_TRANSPARENT ); 8602 m_aCurrentPDFState.m_aFillColor = Color( COL_TRANSPARENT ); 8603 } 8604 8605 Rectangle PDFWriterImpl::getRedirectTargetRect() const 8606 { 8607 return m_aOutputStreams.empty() ? Rectangle() : m_aOutputStreams.front().m_aTargetRect; 8608 } 8609 8610 SvStream* PDFWriterImpl::endRedirect() 8611 { 8612 SvStream* pStream = NULL; 8613 if( ! m_aOutputStreams.empty() ) 8614 { 8615 pStream = m_aOutputStreams.front().m_pStream; 8616 m_aMapMode = m_aOutputStreams.front().m_aMapMode; 8617 m_aOutputStreams.pop_front(); 8618 } 8619 8620 pop(); 8621 // force reemitting colors and clip region 8622 clearClipRegion(); 8623 m_aCurrentPDFState.m_bClipRegion = m_aGraphicsStack.front().m_bClipRegion; 8624 m_aCurrentPDFState.m_aClipRegion = m_aGraphicsStack.front().m_aClipRegion; 8625 m_aCurrentPDFState.m_aLineColor = Color( COL_TRANSPARENT ); 8626 m_aCurrentPDFState.m_aFillColor = Color( COL_TRANSPARENT ); 8627 8628 updateGraphicsState(); 8629 8630 return pStream; 8631 } 8632 8633 void PDFWriterImpl::beginTransparencyGroup() 8634 { 8635 updateGraphicsState(); 8636 if( m_aContext.Version >= PDFWriter::PDF_1_4 ) 8637 beginRedirect( new SvMemoryStream( 1024, 1024 ), Rectangle() ); 8638 } 8639 8640 void PDFWriterImpl::endTransparencyGroup( const Rectangle& rBoundingBox, sal_uInt32 nTransparentPercent ) 8641 { 8642 DBG_ASSERT( nTransparentPercent <= 100, "invalid alpha value" ); 8643 nTransparentPercent = nTransparentPercent % 100; 8644 8645 if( m_aContext.Version >= PDFWriter::PDF_1_4 ) 8646 { 8647 // create XObject 8648 m_aTransparentObjects.push_back( TransparencyEmit() ); 8649 m_aTransparentObjects.back().m_aBoundRect = rBoundingBox; 8650 // convert rectangle to default user space 8651 m_aPages.back().convertRect( m_aTransparentObjects.back().m_aBoundRect ); 8652 m_aTransparentObjects.back().m_nObject = createObject(); 8653 m_aTransparentObjects.back().m_fAlpha = (double)(100-nTransparentPercent) / 100.0; 8654 // get XObject's content stream 8655 m_aTransparentObjects.back().m_pContentStream = static_cast<SvMemoryStream*>(endRedirect()); 8656 m_aTransparentObjects.back().m_nExtGStateObject = createObject(); 8657 8658 OStringBuffer aObjName( 16 ); 8659 aObjName.append( "Tr" ); 8660 aObjName.append( m_aTransparentObjects.back().m_nObject ); 8661 OString aTrName( aObjName.makeStringAndClear() ); 8662 aObjName.append( "EGS" ); 8663 aObjName.append( m_aTransparentObjects.back().m_nExtGStateObject ); 8664 OString aExtName( aObjName.makeStringAndClear() ); 8665 8666 OStringBuffer aLine( 80 ); 8667 // insert XObject 8668 aLine.append( "q /" ); 8669 aLine.append( aExtName ); 8670 aLine.append( " gs /" ); 8671 aLine.append( aTrName ); 8672 aLine.append( " Do Q\n" ); 8673 writeBuffer( aLine.getStr(), aLine.getLength() ); 8674 8675 pushResource( ResXObject, aTrName, m_aTransparentObjects.back().m_nObject ); 8676 pushResource( ResExtGState, aExtName, m_aTransparentObjects.back().m_nExtGStateObject ); 8677 } 8678 } 8679 8680 void PDFWriterImpl::endTransparencyGroup( const Rectangle& rBoundingBox, const Bitmap& rAlphaMask ) 8681 { 8682 if( m_aContext.Version >= PDFWriter::PDF_1_4 ) 8683 { 8684 // create XObject 8685 m_aTransparentObjects.push_back( TransparencyEmit() ); 8686 m_aTransparentObjects.back().m_aBoundRect = rBoundingBox; 8687 // convert rectangle to default user space 8688 m_aPages.back().convertRect( m_aTransparentObjects.back().m_aBoundRect ); 8689 m_aTransparentObjects.back().m_nObject = createObject(); 8690 m_aTransparentObjects.back().m_fAlpha = 0.0; 8691 // get XObject's content stream 8692 m_aTransparentObjects.back().m_pContentStream = static_cast<SvMemoryStream*>(endRedirect()); 8693 m_aTransparentObjects.back().m_nExtGStateObject = createObject(); 8694 8695 // draw soft mask 8696 beginRedirect( new SvMemoryStream( 1024, 1024 ), Rectangle() ); 8697 drawBitmap( rBoundingBox.TopLeft(), rBoundingBox.GetSize(), rAlphaMask ); 8698 m_aTransparentObjects.back().m_pSoftMaskStream = static_cast<SvMemoryStream*>(endRedirect()); 8699 8700 OStringBuffer aObjName( 16 ); 8701 aObjName.append( "Tr" ); 8702 aObjName.append( m_aTransparentObjects.back().m_nObject ); 8703 OString aTrName( aObjName.makeStringAndClear() ); 8704 aObjName.append( "EGS" ); 8705 aObjName.append( m_aTransparentObjects.back().m_nExtGStateObject ); 8706 OString aExtName( aObjName.makeStringAndClear() ); 8707 8708 OStringBuffer aLine( 80 ); 8709 // insert XObject 8710 aLine.append( "q /" ); 8711 aLine.append( aExtName ); 8712 aLine.append( " gs /" ); 8713 aLine.append( aTrName ); 8714 aLine.append( " Do Q\n" ); 8715 writeBuffer( aLine.getStr(), aLine.getLength() ); 8716 8717 pushResource( ResXObject, aTrName, m_aTransparentObjects.back().m_nObject ); 8718 pushResource( ResExtGState, aExtName, m_aTransparentObjects.back().m_nExtGStateObject ); 8719 } 8720 } 8721 8722 void PDFWriterImpl::drawRectangle( const Rectangle& rRect ) 8723 { 8724 MARK( "drawRectangle" ); 8725 8726 updateGraphicsState(); 8727 8728 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8729 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8730 return; 8731 8732 OStringBuffer aLine( 40 ); 8733 m_aPages.back().appendRect( rRect, aLine ); 8734 8735 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) && 8736 m_aGraphicsStack.front().m_aFillColor != Color( COL_TRANSPARENT ) ) 8737 aLine.append( " B*\n" ); 8738 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 8739 aLine.append( " S\n" ); 8740 else 8741 aLine.append( " f*\n" ); 8742 8743 writeBuffer( aLine.getStr(), aLine.getLength() ); 8744 } 8745 8746 void PDFWriterImpl::drawRectangle( const Rectangle& rRect, sal_uInt32 nHorzRound, sal_uInt32 nVertRound ) 8747 { 8748 MARK( "drawRectangle with rounded edges" ); 8749 8750 if( !nHorzRound && !nVertRound ) 8751 drawRectangle( rRect ); 8752 8753 updateGraphicsState(); 8754 8755 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8756 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8757 return; 8758 8759 if( nHorzRound > (sal_uInt32)rRect.GetWidth()/2 ) 8760 nHorzRound = rRect.GetWidth()/2; 8761 if( nVertRound > (sal_uInt32)rRect.GetHeight()/2 ) 8762 nVertRound = rRect.GetHeight()/2; 8763 8764 Point aPoints[16]; 8765 const double kappa = 0.5522847498; 8766 const sal_uInt32 kx = (sal_uInt32)((kappa*(double)nHorzRound)+0.5); 8767 const sal_uInt32 ky = (sal_uInt32)((kappa*(double)nVertRound)+0.5); 8768 8769 aPoints[1] = Point( rRect.TopLeft().X() + nHorzRound, rRect.TopLeft().Y() ); 8770 aPoints[0] = Point( aPoints[1].X() - kx, aPoints[1].Y() ); 8771 aPoints[2] = Point( rRect.TopRight().X()+1 - nHorzRound, aPoints[1].Y() ); 8772 aPoints[3] = Point( aPoints[2].X()+kx, aPoints[2].Y() ); 8773 8774 aPoints[5] = Point( rRect.TopRight().X()+1, rRect.TopRight().Y()+nVertRound ); 8775 aPoints[4] = Point( aPoints[5].X(), aPoints[5].Y()-ky ); 8776 aPoints[6] = Point( aPoints[5].X(), rRect.BottomRight().Y()+1 - nVertRound ); 8777 aPoints[7] = Point( aPoints[6].X(), aPoints[6].Y()+ky ); 8778 8779 aPoints[9] = Point( rRect.BottomRight().X()+1-nHorzRound, rRect.BottomRight().Y()+1 ); 8780 aPoints[8] = Point( aPoints[9].X()+kx, aPoints[9].Y() ); 8781 aPoints[10] = Point( rRect.BottomLeft().X() + nHorzRound, aPoints[9].Y() ); 8782 aPoints[11] = Point( aPoints[10].X()-kx, aPoints[10].Y() ); 8783 8784 aPoints[13] = Point( rRect.BottomLeft().X(), rRect.BottomLeft().Y()+1-nVertRound ); 8785 aPoints[12] = Point( aPoints[13].X(), aPoints[13].Y()+ky ); 8786 aPoints[14] = Point( rRect.TopLeft().X(), rRect.TopLeft().Y()+nVertRound ); 8787 aPoints[15] = Point( aPoints[14].X(), aPoints[14].Y()-ky ); 8788 8789 8790 OStringBuffer aLine( 80 ); 8791 m_aPages.back().appendPoint( aPoints[1], aLine ); 8792 aLine.append( " m " ); 8793 m_aPages.back().appendPoint( aPoints[2], aLine ); 8794 aLine.append( " l " ); 8795 m_aPages.back().appendPoint( aPoints[3], aLine ); 8796 aLine.append( ' ' ); 8797 m_aPages.back().appendPoint( aPoints[4], aLine ); 8798 aLine.append( ' ' ); 8799 m_aPages.back().appendPoint( aPoints[5], aLine ); 8800 aLine.append( " c\n" ); 8801 m_aPages.back().appendPoint( aPoints[6], aLine ); 8802 aLine.append( " l " ); 8803 m_aPages.back().appendPoint( aPoints[7], aLine ); 8804 aLine.append( ' ' ); 8805 m_aPages.back().appendPoint( aPoints[8], aLine ); 8806 aLine.append( ' ' ); 8807 m_aPages.back().appendPoint( aPoints[9], aLine ); 8808 aLine.append( " c\n" ); 8809 m_aPages.back().appendPoint( aPoints[10], aLine ); 8810 aLine.append( " l " ); 8811 m_aPages.back().appendPoint( aPoints[11], aLine ); 8812 aLine.append( ' ' ); 8813 m_aPages.back().appendPoint( aPoints[12], aLine ); 8814 aLine.append( ' ' ); 8815 m_aPages.back().appendPoint( aPoints[13], aLine ); 8816 aLine.append( " c\n" ); 8817 m_aPages.back().appendPoint( aPoints[14], aLine ); 8818 aLine.append( " l " ); 8819 m_aPages.back().appendPoint( aPoints[15], aLine ); 8820 aLine.append( ' ' ); 8821 m_aPages.back().appendPoint( aPoints[0], aLine ); 8822 aLine.append( ' ' ); 8823 m_aPages.back().appendPoint( aPoints[1], aLine ); 8824 aLine.append( " c " ); 8825 8826 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) && 8827 m_aGraphicsStack.front().m_aFillColor != Color( COL_TRANSPARENT ) ) 8828 aLine.append( "b*\n" ); 8829 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 8830 aLine.append( "s\n" ); 8831 else 8832 aLine.append( "f*\n" ); 8833 8834 writeBuffer( aLine.getStr(), aLine.getLength() ); 8835 } 8836 8837 void PDFWriterImpl::drawEllipse( const Rectangle& rRect ) 8838 { 8839 MARK( "drawEllipse" ); 8840 8841 updateGraphicsState(); 8842 8843 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8844 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8845 return; 8846 8847 Point aPoints[12]; 8848 const double kappa = 0.5522847498; 8849 const sal_uInt32 kx = (sal_uInt32)((kappa*(double)rRect.GetWidth()/2.0)+0.5); 8850 const sal_uInt32 ky = (sal_uInt32)((kappa*(double)rRect.GetHeight()/2.0)+0.5); 8851 8852 aPoints[1] = Point( rRect.TopLeft().X() + rRect.GetWidth()/2, rRect.TopLeft().Y() ); 8853 aPoints[0] = Point( aPoints[1].X() - kx, aPoints[1].Y() ); 8854 aPoints[2] = Point( aPoints[1].X() + kx, aPoints[1].Y() ); 8855 8856 aPoints[4] = Point( rRect.TopRight().X()+1, rRect.TopRight().Y() + rRect.GetHeight()/2 ); 8857 aPoints[3] = Point( aPoints[4].X(), aPoints[4].Y() - ky ); 8858 aPoints[5] = Point( aPoints[4].X(), aPoints[4].Y() + ky ); 8859 8860 aPoints[7] = Point( rRect.BottomLeft().X() + rRect.GetWidth()/2, rRect.BottomLeft().Y()+1 ); 8861 aPoints[6] = Point( aPoints[7].X() + kx, aPoints[7].Y() ); 8862 aPoints[8] = Point( aPoints[7].X() - kx, aPoints[7].Y() ); 8863 8864 aPoints[10] = Point( rRect.TopLeft().X(), rRect.TopLeft().Y() + rRect.GetHeight()/2 ); 8865 aPoints[9] = Point( aPoints[10].X(), aPoints[10].Y() + ky ); 8866 aPoints[11] = Point( aPoints[10].X(), aPoints[10].Y() - ky ); 8867 8868 OStringBuffer aLine( 80 ); 8869 m_aPages.back().appendPoint( aPoints[1], aLine ); 8870 aLine.append( " m " ); 8871 m_aPages.back().appendPoint( aPoints[2], aLine ); 8872 aLine.append( ' ' ); 8873 m_aPages.back().appendPoint( aPoints[3], aLine ); 8874 aLine.append( ' ' ); 8875 m_aPages.back().appendPoint( aPoints[4], aLine ); 8876 aLine.append( " c\n" ); 8877 m_aPages.back().appendPoint( aPoints[5], aLine ); 8878 aLine.append( ' ' ); 8879 m_aPages.back().appendPoint( aPoints[6], aLine ); 8880 aLine.append( ' ' ); 8881 m_aPages.back().appendPoint( aPoints[7], aLine ); 8882 aLine.append( " c\n" ); 8883 m_aPages.back().appendPoint( aPoints[8], aLine ); 8884 aLine.append( ' ' ); 8885 m_aPages.back().appendPoint( aPoints[9], aLine ); 8886 aLine.append( ' ' ); 8887 m_aPages.back().appendPoint( aPoints[10], aLine ); 8888 aLine.append( " c\n" ); 8889 m_aPages.back().appendPoint( aPoints[11], aLine ); 8890 aLine.append( ' ' ); 8891 m_aPages.back().appendPoint( aPoints[0], aLine ); 8892 aLine.append( ' ' ); 8893 m_aPages.back().appendPoint( aPoints[1], aLine ); 8894 aLine.append( " c " ); 8895 8896 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) && 8897 m_aGraphicsStack.front().m_aFillColor != Color( COL_TRANSPARENT ) ) 8898 aLine.append( "b*\n" ); 8899 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 8900 aLine.append( "s\n" ); 8901 else 8902 aLine.append( "f*\n" ); 8903 8904 writeBuffer( aLine.getStr(), aLine.getLength() ); 8905 } 8906 8907 static double calcAngle( const Rectangle& rRect, const Point& rPoint ) 8908 { 8909 Point aOrigin((rRect.Left()+rRect.Right()+1)/2, 8910 (rRect.Top()+rRect.Bottom()+1)/2); 8911 Point aPoint = rPoint - aOrigin; 8912 8913 double fX = (double)aPoint.X(); 8914 double fY = (double)-aPoint.Y(); 8915 8916 if( rRect.GetWidth() > rRect.GetHeight() ) 8917 fY = fY*((double)rRect.GetWidth()/(double)rRect.GetHeight()); 8918 else if( rRect.GetHeight() > rRect.GetWidth() ) 8919 fX = fX*((double)rRect.GetHeight()/(double)rRect.GetWidth()); 8920 return atan2( fY, fX ); 8921 } 8922 8923 void PDFWriterImpl::drawArc( const Rectangle& rRect, const Point& rStart, const Point& rStop, bool bWithPie, bool bWithChord ) 8924 { 8925 MARK( "drawArc" ); 8926 8927 updateGraphicsState(); 8928 8929 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 8930 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 8931 return; 8932 8933 // calculate start and stop angles 8934 const double fStartAngle = calcAngle( rRect, rStart ); 8935 double fStopAngle = calcAngle( rRect, rStop ); 8936 while( fStopAngle < fStartAngle ) 8937 fStopAngle += 2.0*M_PI; 8938 const int nFragments = (int)((fStopAngle-fStartAngle)/(M_PI/2.0))+1; 8939 const double fFragmentDelta = (fStopAngle-fStartAngle)/(double)nFragments; 8940 const double kappa = fabs( 4.0 * (1.0-cos(fFragmentDelta/2.0))/sin(fFragmentDelta/2.0) / 3.0); 8941 const double halfWidth = (double)rRect.GetWidth()/2.0; 8942 const double halfHeight = (double)rRect.GetHeight()/2.0; 8943 8944 const Point aCenter( (rRect.Left()+rRect.Right()+1)/2, 8945 (rRect.Top()+rRect.Bottom()+1)/2 ); 8946 8947 OStringBuffer aLine( 30*nFragments ); 8948 Point aPoint( (int)(halfWidth * cos(fStartAngle) ), 8949 -(int)(halfHeight * sin(fStartAngle) ) ); 8950 aPoint += aCenter; 8951 m_aPages.back().appendPoint( aPoint, aLine ); 8952 aLine.append( " m " ); 8953 if( !basegfx::fTools::equal(fStartAngle, fStopAngle) ) 8954 { 8955 for( int i = 0; i < nFragments; i++ ) 8956 { 8957 const double fStartFragment = fStartAngle + (double)i*fFragmentDelta; 8958 const double fStopFragment = fStartFragment + fFragmentDelta; 8959 aPoint = Point( (int)(halfWidth * (cos(fStartFragment) - kappa*sin(fStartFragment) ) ), 8960 -(int)(halfHeight * (sin(fStartFragment) + kappa*cos(fStartFragment) ) ) ); 8961 aPoint += aCenter; 8962 m_aPages.back().appendPoint( aPoint, aLine ); 8963 aLine.append( ' ' ); 8964 8965 aPoint = Point( (int)(halfWidth * (cos(fStopFragment) + kappa*sin(fStopFragment) ) ), 8966 -(int)(halfHeight * (sin(fStopFragment) - kappa*cos(fStopFragment) ) ) ); 8967 aPoint += aCenter; 8968 m_aPages.back().appendPoint( aPoint, aLine ); 8969 aLine.append( ' ' ); 8970 8971 aPoint = Point( (int)(halfWidth * cos(fStopFragment) ), 8972 -(int)(halfHeight * sin(fStopFragment) ) ); 8973 aPoint += aCenter; 8974 m_aPages.back().appendPoint( aPoint, aLine ); 8975 aLine.append( " c\n" ); 8976 } 8977 } 8978 if( bWithChord || bWithPie ) 8979 { 8980 if( bWithPie ) 8981 { 8982 m_aPages.back().appendPoint( aCenter, aLine ); 8983 aLine.append( " l " ); 8984 } 8985 aLine.append( "h " ); 8986 } 8987 if( ! bWithChord && ! bWithPie ) 8988 aLine.append( "S\n" ); 8989 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) && 8990 m_aGraphicsStack.front().m_aFillColor != Color( COL_TRANSPARENT ) ) 8991 aLine.append( "B*\n" ); 8992 else if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 8993 aLine.append( "S\n" ); 8994 else 8995 aLine.append( "f*\n" ); 8996 8997 writeBuffer( aLine.getStr(), aLine.getLength() ); 8998 } 8999 9000 void PDFWriterImpl::drawPolyLine( const Polygon& rPoly ) 9001 { 9002 MARK( "drawPolyLine" ); 9003 9004 sal_uInt16 nPoints = rPoly.GetSize(); 9005 if( nPoints < 2 ) 9006 return; 9007 9008 updateGraphicsState(); 9009 9010 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) ) 9011 return; 9012 9013 OStringBuffer aLine( 20 * nPoints ); 9014 m_aPages.back().appendPolygon( rPoly, aLine, rPoly[0] == rPoly[nPoints-1] ); 9015 aLine.append( "S\n" ); 9016 9017 writeBuffer( aLine.getStr(), aLine.getLength() ); 9018 } 9019 9020 void PDFWriterImpl::drawPolyLine( const Polygon& rPoly, const LineInfo& rInfo ) 9021 { 9022 MARK( "drawPolyLine with LineInfo" ); 9023 9024 updateGraphicsState(); 9025 9026 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) ) 9027 return; 9028 9029 OStringBuffer aLine; 9030 aLine.append( "q " ); 9031 if( m_aPages.back().appendLineInfo( rInfo, aLine ) ) 9032 { 9033 writeBuffer( aLine.getStr(), aLine.getLength() ); 9034 drawPolyLine( rPoly ); 9035 writeBuffer( "Q\n", 2 ); 9036 } 9037 else 9038 { 9039 PDFWriter::ExtLineInfo aInfo; 9040 convertLineInfoToExtLineInfo( rInfo, aInfo ); 9041 drawPolyLine( rPoly, aInfo ); 9042 } 9043 } 9044 9045 void PDFWriterImpl::convertLineInfoToExtLineInfo( const LineInfo& rIn, PDFWriter::ExtLineInfo& rOut ) 9046 { 9047 DBG_ASSERT( rIn.GetStyle() == LINE_DASH, "invalid conversion" ); 9048 rOut.m_fLineWidth = rIn.GetWidth(); 9049 rOut.m_fTransparency = 0.0; 9050 rOut.m_eCap = PDFWriter::capButt; 9051 rOut.m_eJoin = PDFWriter::joinMiter; 9052 rOut.m_fMiterLimit = 10; 9053 rOut.m_aDashArray.clear(); 9054 9055 int nDashes = rIn.GetDashCount(); 9056 int nDashLen = rIn.GetDashLen(); 9057 int nDistance = rIn.GetDistance(); 9058 for( int n = 0; n < nDashes; n++ ) 9059 { 9060 rOut.m_aDashArray.push_back( nDashLen ); 9061 rOut.m_aDashArray.push_back( nDistance ); 9062 } 9063 int nDots = rIn.GetDotCount(); 9064 int nDotLen = rIn.GetDotLen(); 9065 for( int n = 0; n < nDots; n++ ) 9066 { 9067 rOut.m_aDashArray.push_back( nDotLen ); 9068 rOut.m_aDashArray.push_back( nDistance ); 9069 } 9070 } 9071 9072 void PDFWriterImpl::drawPolyLine( const Polygon& rPoly, const PDFWriter::ExtLineInfo& rInfo ) 9073 { 9074 MARK( "drawPolyLine with ExtLineInfo" ); 9075 9076 updateGraphicsState(); 9077 9078 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) ) 9079 return; 9080 9081 if( rInfo.m_fTransparency >= 1.0 ) 9082 return; 9083 9084 if( rInfo.m_fTransparency != 0.0 ) 9085 beginTransparencyGroup(); 9086 9087 OStringBuffer aLine; 9088 aLine.append( "q " ); 9089 m_aPages.back().appendMappedLength( rInfo.m_fLineWidth, aLine ); 9090 aLine.append( " w" ); 9091 if( rInfo.m_aDashArray.size() < 10 ) // implmentation limit of acrobat reader 9092 { 9093 switch( rInfo.m_eCap ) 9094 { 9095 default: 9096 case PDFWriter::capButt: aLine.append( " 0 J" );break; 9097 case PDFWriter::capRound: aLine.append( " 1 J" );break; 9098 case PDFWriter::capSquare: aLine.append( " 2 J" );break; 9099 } 9100 switch( rInfo.m_eJoin ) 9101 { 9102 default: 9103 case PDFWriter::joinMiter: 9104 { 9105 double fLimit = rInfo.m_fMiterLimit; 9106 if( rInfo.m_fLineWidth < rInfo.m_fMiterLimit ) 9107 fLimit = fLimit / rInfo.m_fLineWidth; 9108 if( fLimit < 1.0 ) 9109 fLimit = 1.0; 9110 aLine.append( " 0 j " ); 9111 appendDouble( fLimit, aLine ); 9112 aLine.append( " M" ); 9113 } 9114 break; 9115 case PDFWriter::joinRound: aLine.append( " 1 j" );break; 9116 case PDFWriter::joinBevel: aLine.append( " 2 j" );break; 9117 } 9118 if( rInfo.m_aDashArray.size() > 0 ) 9119 { 9120 aLine.append( " [ " ); 9121 for( std::vector<double>::const_iterator it = rInfo.m_aDashArray.begin(); 9122 it != rInfo.m_aDashArray.end(); ++it ) 9123 { 9124 m_aPages.back().appendMappedLength( *it, aLine ); 9125 aLine.append( ' ' ); 9126 } 9127 aLine.append( "] 0 d" ); 9128 } 9129 aLine.append( "\n" ); 9130 writeBuffer( aLine.getStr(), aLine.getLength() ); 9131 drawPolyLine( rPoly ); 9132 } 9133 else 9134 { 9135 basegfx::B2DPolygon aPoly(rPoly.getB2DPolygon()); 9136 basegfx::B2DPolyPolygon aPolyPoly; 9137 9138 basegfx::tools::applyLineDashing(aPoly, rInfo.m_aDashArray, &aPolyPoly); 9139 9140 // Old applyLineDashing subdivided the polygon. New one will create bezier curve segments. 9141 // To mimic old behaviour, apply subdivide here. If beziers shall be written (better quality) 9142 // this line needs to be removed and the loop below adapted accordingly 9143 aPolyPoly = basegfx::tools::adaptiveSubdivideByAngle(aPolyPoly); 9144 9145 const sal_uInt32 nPolygonCount(aPolyPoly.count()); 9146 9147 for( sal_uInt32 nPoly = 0; nPoly < nPolygonCount; nPoly++ ) 9148 { 9149 aLine.append( (nPoly != 0 && (nPoly & 7) == 0) ? "\n" : " " ); 9150 aPoly = aPolyPoly.getB2DPolygon( nPoly ); 9151 const sal_uInt32 nPointCount(aPoly.count()); 9152 9153 if(nPointCount) 9154 { 9155 const sal_uInt32 nEdgeCount(aPoly.isClosed() ? nPointCount : nPointCount - 1); 9156 basegfx::B2DPoint aCurrent(aPoly.getB2DPoint(0)); 9157 9158 for(sal_uInt32 a(0); a < nEdgeCount; a++) 9159 { 9160 if( a > 0 ) 9161 aLine.append( " " ); 9162 const sal_uInt32 nNextIndex((a + 1) % nPointCount); 9163 const basegfx::B2DPoint aNext(aPoly.getB2DPoint(nNextIndex)); 9164 9165 m_aPages.back().appendPoint( Point( FRound(aCurrent.getX()), 9166 FRound(aCurrent.getY()) ), 9167 aLine ); 9168 aLine.append( " m " ); 9169 m_aPages.back().appendPoint( Point( FRound(aNext.getX()), 9170 FRound(aNext.getY()) ), 9171 aLine ); 9172 aLine.append( " l" ); 9173 9174 // prepare next edge 9175 aCurrent = aNext; 9176 } 9177 } 9178 } 9179 aLine.append( " S " ); 9180 writeBuffer( aLine.getStr(), aLine.getLength() ); 9181 } 9182 writeBuffer( "Q\n", 2 ); 9183 9184 if( rInfo.m_fTransparency != 0.0 ) 9185 { 9186 // FIXME: actually this may be incorrect with bezier polygons 9187 Rectangle aBoundRect( rPoly.GetBoundRect() ); 9188 // avoid clipping with thick lines 9189 if( rInfo.m_fLineWidth > 0.0 ) 9190 { 9191 sal_Int32 nLW = sal_Int32(rInfo.m_fLineWidth); 9192 aBoundRect.Top() -= nLW; 9193 aBoundRect.Left() -= nLW; 9194 aBoundRect.Right() += nLW; 9195 aBoundRect.Bottom() += nLW; 9196 } 9197 endTransparencyGroup( aBoundRect, (sal_uInt16)(100.0*rInfo.m_fTransparency) ); 9198 } 9199 } 9200 9201 void PDFWriterImpl::drawPixel( const Point& rPoint, const Color& rColor ) 9202 { 9203 MARK( "drawPixel" ); 9204 9205 Color aColor = ( rColor == Color( COL_TRANSPARENT ) ? m_aGraphicsStack.front().m_aLineColor : rColor ); 9206 9207 if( aColor == Color( COL_TRANSPARENT ) ) 9208 return; 9209 9210 // pixels are drawn in line color, so have to set 9211 // the nonstroking color to line color 9212 Color aOldFillColor = m_aGraphicsStack.front().m_aFillColor; 9213 setFillColor( aColor ); 9214 9215 updateGraphicsState(); 9216 9217 OStringBuffer aLine( 20 ); 9218 m_aPages.back().appendPoint( rPoint, aLine ); 9219 aLine.append( ' ' ); 9220 appendDouble( 1.0/double(getReferenceDevice()->ImplGetDPIX()), aLine ); 9221 aLine.append( ' ' ); 9222 appendDouble( 1.0/double(getReferenceDevice()->ImplGetDPIY()), aLine ); 9223 aLine.append( " re f\n" ); 9224 writeBuffer( aLine.getStr(), aLine.getLength() ); 9225 9226 setFillColor( aOldFillColor ); 9227 } 9228 9229 void PDFWriterImpl::drawPixel( const Polygon& rPoints, const Color* pColors ) 9230 { 9231 MARK( "drawPixel with Polygon" ); 9232 9233 updateGraphicsState(); 9234 9235 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && ! pColors ) 9236 return; 9237 9238 sal_uInt16 nPoints = rPoints.GetSize(); 9239 OStringBuffer aLine( nPoints*40 ); 9240 aLine.append( "q " ); 9241 if( ! pColors ) 9242 { 9243 appendNonStrokingColor( m_aGraphicsStack.front().m_aLineColor, aLine ); 9244 aLine.append( ' ' ); 9245 } 9246 9247 OStringBuffer aPixel(32); 9248 aPixel.append( ' ' ); 9249 appendDouble( 1.0/double(getReferenceDevice()->ImplGetDPIX()), aPixel ); 9250 aPixel.append( ' ' ); 9251 appendDouble( 1.0/double(getReferenceDevice()->ImplGetDPIY()), aPixel ); 9252 OString aPixelStr = aPixel.makeStringAndClear(); 9253 for( sal_uInt16 i = 0; i < nPoints; i++ ) 9254 { 9255 if( pColors ) 9256 { 9257 if( pColors[i] == Color( COL_TRANSPARENT ) ) 9258 continue; 9259 9260 appendNonStrokingColor( pColors[i], aLine ); 9261 aLine.append( ' ' ); 9262 } 9263 m_aPages.back().appendPoint( rPoints[i], aLine ); 9264 aLine.append( aPixelStr ); 9265 aLine.append( " re f\n" ); 9266 } 9267 aLine.append( "Q\n" ); 9268 writeBuffer( aLine.getStr(), aLine.getLength() ); 9269 } 9270 9271 class AccessReleaser 9272 { 9273 BitmapReadAccess* m_pAccess; 9274 public: 9275 AccessReleaser( BitmapReadAccess* pAccess ) : m_pAccess( pAccess ){} 9276 ~AccessReleaser() { delete m_pAccess; } 9277 }; 9278 9279 bool PDFWriterImpl::writeTransparentObject( TransparencyEmit& rObject ) 9280 { 9281 CHECK_RETURN( updateObject( rObject.m_nObject ) ); 9282 9283 bool bFlateFilter = compressStream( rObject.m_pContentStream ); 9284 rObject.m_pContentStream->Seek( STREAM_SEEK_TO_END ); 9285 sal_uLong nSize = rObject.m_pContentStream->Tell(); 9286 rObject.m_pContentStream->Seek( STREAM_SEEK_TO_BEGIN ); 9287 #if OSL_DEBUG_LEVEL > 1 9288 emitComment( "PDFWriterImpl::writeTransparentObject" ); 9289 #endif 9290 OStringBuffer aLine( 512 ); 9291 CHECK_RETURN( updateObject( rObject.m_nObject ) ); 9292 aLine.append( rObject.m_nObject ); 9293 aLine.append( " 0 obj\n" 9294 "<</Type/XObject\n" 9295 "/Subtype/Form\n" 9296 "/BBox[ " ); 9297 appendFixedInt( rObject.m_aBoundRect.Left(), aLine ); 9298 aLine.append( ' ' ); 9299 appendFixedInt( rObject.m_aBoundRect.Top(), aLine ); 9300 aLine.append( ' ' ); 9301 appendFixedInt( rObject.m_aBoundRect.Right(), aLine ); 9302 aLine.append( ' ' ); 9303 appendFixedInt( rObject.m_aBoundRect.Bottom()+1, aLine ); 9304 aLine.append( " ]\n" ); 9305 if( ! rObject.m_pSoftMaskStream ) 9306 { 9307 if( ! m_bIsPDF_A1 ) 9308 { 9309 aLine.append( "/Group<</S/Transparency/CS/DeviceRGB/K true>>\n" ); 9310 } 9311 } 9312 /* #i42884# the PDF reference recommends that each Form XObject 9313 * should have a resource dict; alas if that is the same object 9314 * as the one of the page it triggers an endless recursion in 9315 * acroread 5 (6 and up have that fixed). Since we have only one 9316 * resource dict anyway, let's use the one from the page by NOT 9317 * emitting a Resources entry. 9318 */ 9319 #if 0 9320 aLine.append( " /Resources " ); 9321 aLine.append( getResourceDictObj() ); 9322 aLine.append( " 0 R\n" ); 9323 #endif 9324 9325 aLine.append( "/Length " ); 9326 aLine.append( (sal_Int32)(nSize) ); 9327 aLine.append( "\n" ); 9328 if( bFlateFilter ) 9329 aLine.append( "/Filter/FlateDecode\n" ); 9330 aLine.append( ">>\n" 9331 "stream\n" ); 9332 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9333 checkAndEnableStreamEncryption( rObject.m_nObject ); 9334 CHECK_RETURN( writeBuffer( rObject.m_pContentStream->GetData(), nSize ) ); 9335 disableStreamEncryption(); 9336 aLine.setLength( 0 ); 9337 aLine.append( "\n" 9338 "endstream\n" 9339 "endobj\n\n" ); 9340 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9341 9342 // write ExtGState dict for this XObject 9343 aLine.setLength( 0 ); 9344 aLine.append( rObject.m_nExtGStateObject ); 9345 aLine.append( " 0 obj\n" 9346 "<<" ); 9347 if( ! rObject.m_pSoftMaskStream ) 9348 { 9349 //i59651 9350 if( m_bIsPDF_A1 ) 9351 { 9352 aLine.append( "/CA 1.0/ca 1.0" ); 9353 m_aErrors.insert( PDFWriter::Warning_Transparency_Omitted_PDFA ); 9354 } 9355 else 9356 { 9357 aLine.append( "/CA " ); 9358 appendDouble( rObject.m_fAlpha, aLine ); 9359 aLine.append( "\n" 9360 " /ca " ); 9361 appendDouble( rObject.m_fAlpha, aLine ); 9362 } 9363 aLine.append( "\n" ); 9364 } 9365 else 9366 { 9367 if( m_bIsPDF_A1 ) 9368 { 9369 aLine.append( "/SMask/None" ); 9370 m_aErrors.insert( PDFWriter::Warning_Transparency_Omitted_PDFA ); 9371 } 9372 else 9373 { 9374 rObject.m_pSoftMaskStream->Seek( STREAM_SEEK_TO_END ); 9375 sal_Int32 nMaskSize = (sal_Int32)rObject.m_pSoftMaskStream->Tell(); 9376 rObject.m_pSoftMaskStream->Seek( STREAM_SEEK_TO_BEGIN ); 9377 sal_Int32 nMaskObject = createObject(); 9378 aLine.append( "/SMask<</Type/Mask/S/Luminosity/G " ); 9379 aLine.append( nMaskObject ); 9380 aLine.append( " 0 R>>\n" ); 9381 9382 OStringBuffer aMask; 9383 aMask.append( nMaskObject ); 9384 aMask.append( " 0 obj\n" 9385 "<</Type/XObject\n" 9386 "/Subtype/Form\n" 9387 "/BBox[" ); 9388 appendFixedInt( rObject.m_aBoundRect.Left(), aMask ); 9389 aMask.append( ' ' ); 9390 appendFixedInt( rObject.m_aBoundRect.Top(), aMask ); 9391 aMask.append( ' ' ); 9392 appendFixedInt( rObject.m_aBoundRect.Right(), aMask ); 9393 aMask.append( ' ' ); 9394 appendFixedInt( rObject.m_aBoundRect.Bottom()+1, aMask ); 9395 aMask.append( "]\n" ); 9396 9397 /* #i42884# see above */ 9398 #if 0 9399 aLine.append( "/Resources " ); 9400 aMask.append( getResourceDictObj() ); 9401 aMask.append( " 0 R\n" ); 9402 #endif 9403 9404 aMask.append( "/Group<</S/Transparency/CS/DeviceRGB>>\n" ); 9405 aMask.append( "/Length " ); 9406 aMask.append( nMaskSize ); 9407 aMask.append( ">>\n" 9408 "stream\n" ); 9409 CHECK_RETURN( updateObject( nMaskObject ) ); 9410 checkAndEnableStreamEncryption( nMaskObject ); 9411 CHECK_RETURN( writeBuffer( aMask.getStr(), aMask.getLength() ) ); 9412 CHECK_RETURN( writeBuffer( rObject.m_pSoftMaskStream->GetData(), nMaskSize ) ); 9413 disableStreamEncryption(); 9414 aMask.setLength( 0 ); 9415 aMask.append( "\nendstream\n" 9416 "endobj\n\n" ); 9417 CHECK_RETURN( writeBuffer( aMask.getStr(), aMask.getLength() ) ); 9418 } 9419 } 9420 aLine.append( ">>\n" 9421 "endobj\n\n" ); 9422 CHECK_RETURN( updateObject( rObject.m_nExtGStateObject ) ); 9423 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9424 9425 return true; 9426 } 9427 9428 bool PDFWriterImpl::writeGradientFunction( GradientEmit& rObject ) 9429 { 9430 sal_Int32 nFunctionObject = createObject(); 9431 CHECK_RETURN( updateObject( nFunctionObject ) ); 9432 9433 VirtualDevice aDev; 9434 aDev.SetOutputSizePixel( rObject.m_aSize ); 9435 aDev.SetMapMode( MapMode( MAP_PIXEL ) ); 9436 if( m_aContext.ColorMode == PDFWriter::DrawGreyscale ) 9437 aDev.SetDrawMode( aDev.GetDrawMode() | 9438 ( DRAWMODE_GRAYLINE | DRAWMODE_GRAYFILL | DRAWMODE_GRAYTEXT | 9439 DRAWMODE_GRAYBITMAP | DRAWMODE_GRAYGRADIENT ) ); 9440 aDev.DrawGradient( Rectangle( Point( 0, 0 ), rObject.m_aSize ), rObject.m_aGradient ); 9441 9442 Bitmap aSample = aDev.GetBitmap( Point( 0, 0 ), rObject.m_aSize ); 9443 BitmapReadAccess* pAccess = aSample.AcquireReadAccess(); 9444 AccessReleaser aReleaser( pAccess ); 9445 9446 Size aSize = aSample.GetSizePixel(); 9447 9448 sal_Int32 nStreamLengthObject = createObject(); 9449 #if OSL_DEBUG_LEVEL > 1 9450 emitComment( "PDFWriterImpl::writeGradientFunction" ); 9451 #endif 9452 OStringBuffer aLine( 120 ); 9453 aLine.append( nFunctionObject ); 9454 aLine.append( " 0 obj\n" 9455 "<</FunctionType 0\n" 9456 "/Domain[ 0 1 0 1 ]\n" 9457 "/Size[ " ); 9458 aLine.append( (sal_Int32)aSize.Width() ); 9459 aLine.append( ' ' ); 9460 aLine.append( (sal_Int32)aSize.Height() ); 9461 aLine.append( " ]\n" 9462 "/BitsPerSample 8\n" 9463 "/Range[ 0 1 0 1 0 1 ]\n" 9464 "/Order 3\n" 9465 "/Length " ); 9466 aLine.append( nStreamLengthObject ); 9467 aLine.append( " 0 R\n" 9468 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 9469 "/Filter/FlateDecode" 9470 #endif 9471 ">>\n" 9472 "stream\n" ); 9473 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9474 9475 sal_uInt64 nStartStreamPos = 0; 9476 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nStartStreamPos )) ); 9477 9478 checkAndEnableStreamEncryption( nFunctionObject ); 9479 beginCompression(); 9480 for( int y = aSize.Height()-1; y >= 0; y-- ) 9481 { 9482 for( int x = 0; x < aSize.Width(); x++ ) 9483 { 9484 sal_uInt8 aCol[3]; 9485 BitmapColor aColor = pAccess->GetColor( y, x ); 9486 aCol[0] = aColor.GetRed(); 9487 aCol[1] = aColor.GetGreen(); 9488 aCol[2] = aColor.GetBlue(); 9489 CHECK_RETURN( writeBuffer( aCol, 3 ) ); 9490 } 9491 } 9492 endCompression(); 9493 disableStreamEncryption(); 9494 9495 sal_uInt64 nEndStreamPos = 0; 9496 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nEndStreamPos )) ); 9497 9498 aLine.setLength( 0 ); 9499 aLine.append( "\nendstream\nendobj\n\n" ); 9500 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9501 9502 // write stream length 9503 CHECK_RETURN( updateObject( nStreamLengthObject ) ); 9504 aLine.setLength( 0 ); 9505 aLine.append( nStreamLengthObject ); 9506 aLine.append( " 0 obj\n" ); 9507 aLine.append( (sal_Int64)(nEndStreamPos-nStartStreamPos) ); 9508 aLine.append( "\nendobj\n\n" ); 9509 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9510 9511 CHECK_RETURN( updateObject( rObject.m_nObject ) ); 9512 aLine.setLength( 0 ); 9513 aLine.append( rObject.m_nObject ); 9514 aLine.append( " 0 obj\n" 9515 "<</ShadingType 1\n" 9516 "/ColorSpace/DeviceRGB\n" 9517 "/AntiAlias true\n" 9518 "/Domain[ 0 1 0 1 ]\n" 9519 "/Matrix[ " ); 9520 aLine.append( (sal_Int32)aSize.Width() ); 9521 aLine.append( " 0 0 " ); 9522 aLine.append( (sal_Int32)aSize.Height() ); 9523 aLine.append( " 0 0 ]\n" 9524 "/Function " ); 9525 aLine.append( nFunctionObject ); 9526 aLine.append( " 0 R\n" 9527 ">>\n" 9528 "endobj\n\n" ); 9529 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9530 9531 return true; 9532 } 9533 9534 bool PDFWriterImpl::writeJPG( JPGEmit& rObject ) 9535 { 9536 CHECK_RETURN( rObject.m_pStream ); 9537 CHECK_RETURN( updateObject( rObject.m_nObject ) ); 9538 9539 sal_Int32 nLength = 0; 9540 rObject.m_pStream->Seek( STREAM_SEEK_TO_END ); 9541 nLength = rObject.m_pStream->Tell(); 9542 rObject.m_pStream->Seek( STREAM_SEEK_TO_BEGIN ); 9543 9544 sal_Int32 nMaskObject = 0; 9545 if( !!rObject.m_aMask ) 9546 { 9547 if( rObject.m_aMask.GetBitCount() == 1 || 9548 ( rObject.m_aMask.GetBitCount() == 8 && m_aContext.Version >= PDFWriter::PDF_1_4 && !m_bIsPDF_A1 )//i59651 9549 ) 9550 { 9551 nMaskObject = createObject(); 9552 } 9553 else if( m_bIsPDF_A1 ) 9554 m_aErrors.insert( PDFWriter::Warning_Transparency_Omitted_PDFA ); 9555 else if( m_aContext.Version < PDFWriter::PDF_1_4 ) 9556 m_aErrors.insert( PDFWriter::Warning_Transparency_Omitted_PDF13 ); 9557 9558 } 9559 #if OSL_DEBUG_LEVEL > 1 9560 emitComment( "PDFWriterImpl::writeJPG" ); 9561 #endif 9562 9563 OStringBuffer aLine(200); 9564 aLine.append( rObject.m_nObject ); 9565 aLine.append( " 0 obj\n" 9566 "<</Type/XObject/Subtype/Image/Width " ); 9567 aLine.append( (sal_Int32)rObject.m_aID.m_aPixelSize.Width() ); 9568 aLine.append( " /Height " ); 9569 aLine.append( (sal_Int32)rObject.m_aID.m_aPixelSize.Height() ); 9570 aLine.append( " /BitsPerComponent 8 " ); 9571 if( rObject.m_bTrueColor ) 9572 aLine.append( "/ColorSpace/DeviceRGB" ); 9573 else 9574 aLine.append( "/ColorSpace/DeviceGray" ); 9575 aLine.append( "/Filter/DCTDecode/Length " ); 9576 aLine.append( nLength ); 9577 if( nMaskObject ) 9578 { 9579 aLine.append( rObject.m_aMask.GetBitCount() == 1 ? " /Mask " : " /SMask " ); 9580 aLine.append( nMaskObject ); 9581 aLine.append( " 0 R " ); 9582 } 9583 aLine.append( ">>\nstream\n" ); 9584 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9585 9586 checkAndEnableStreamEncryption( rObject.m_nObject ); 9587 CHECK_RETURN( writeBuffer( rObject.m_pStream->GetData(), nLength ) ); 9588 disableStreamEncryption(); 9589 9590 aLine.setLength( 0 ); 9591 aLine.append( "\nendstream\nendobj\n\n" ); 9592 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9593 9594 if( nMaskObject ) 9595 { 9596 BitmapEmit aEmit; 9597 aEmit.m_nObject = nMaskObject; 9598 if( rObject.m_aMask.GetBitCount() == 1 ) 9599 aEmit.m_aBitmap = BitmapEx( rObject.m_aMask, rObject.m_aMask ); 9600 else if( rObject.m_aMask.GetBitCount() == 8 ) 9601 aEmit.m_aBitmap = BitmapEx( rObject.m_aMask, AlphaMask( rObject.m_aMask ) ); 9602 writeBitmapObject( aEmit, true ); 9603 } 9604 9605 return true; 9606 } 9607 9608 bool PDFWriterImpl::writeBitmapObject( BitmapEmit& rObject, bool bMask ) 9609 { 9610 CHECK_RETURN( updateObject( rObject.m_nObject ) ); 9611 9612 Bitmap aBitmap; 9613 Color aTransparentColor( COL_TRANSPARENT ); 9614 bool bWriteMask = false; 9615 if( ! bMask ) 9616 { 9617 aBitmap = rObject.m_aBitmap.GetBitmap(); 9618 if( rObject.m_aBitmap.IsAlpha() ) 9619 { 9620 if( m_aContext.Version >= PDFWriter::PDF_1_4 ) 9621 bWriteMask = true; 9622 // else draw without alpha channel 9623 } 9624 else 9625 { 9626 switch( rObject.m_aBitmap.GetTransparentType() ) 9627 { 9628 case TRANSPARENT_NONE: 9629 // comes from drawMask function 9630 if( aBitmap.GetBitCount() == 1 && rObject.m_bDrawMask ) 9631 bMask = true; 9632 break; 9633 case TRANSPARENT_COLOR: 9634 aTransparentColor = rObject.m_aBitmap.GetTransparentColor(); 9635 break; 9636 case TRANSPARENT_BITMAP: 9637 bWriteMask = true; 9638 break; 9639 } 9640 } 9641 } 9642 else 9643 { 9644 if( m_aContext.Version < PDFWriter::PDF_1_4 || ! rObject.m_aBitmap.IsAlpha() ) 9645 { 9646 aBitmap = rObject.m_aBitmap.GetMask(); 9647 aBitmap.Convert( BMP_CONVERSION_1BIT_THRESHOLD ); 9648 DBG_ASSERT( aBitmap.GetBitCount() == 1, "mask conversion failed" ); 9649 } 9650 else if( aBitmap.GetBitCount() != 8 ) 9651 { 9652 aBitmap = rObject.m_aBitmap.GetAlpha().GetBitmap(); 9653 aBitmap.Convert( BMP_CONVERSION_8BIT_GREYS ); 9654 DBG_ASSERT( aBitmap.GetBitCount() == 8, "alpha mask conversion failed" ); 9655 } 9656 } 9657 9658 BitmapReadAccess* pAccess = aBitmap.AcquireReadAccess(); 9659 AccessReleaser aReleaser( pAccess ); 9660 9661 bool bTrueColor; 9662 sal_Int32 nBitsPerComponent; 9663 switch( aBitmap.GetBitCount() ) 9664 { 9665 case 1: 9666 case 2: 9667 case 4: 9668 case 8: 9669 bTrueColor = false; 9670 nBitsPerComponent = aBitmap.GetBitCount(); 9671 break; 9672 default: 9673 bTrueColor = true; 9674 nBitsPerComponent = 8; 9675 break; 9676 } 9677 9678 sal_Int32 nStreamLengthObject = createObject(); 9679 sal_Int32 nMaskObject = 0; 9680 9681 #if OSL_DEBUG_LEVEL > 1 9682 emitComment( "PDFWriterImpl::writeBitmapObject" ); 9683 #endif 9684 OStringBuffer aLine(1024); 9685 aLine.append( rObject.m_nObject ); 9686 aLine.append( " 0 obj\n" 9687 "<</Type/XObject/Subtype/Image/Width " ); 9688 aLine.append( (sal_Int32)aBitmap.GetSizePixel().Width() ); 9689 aLine.append( "/Height " ); 9690 aLine.append( (sal_Int32)aBitmap.GetSizePixel().Height() ); 9691 aLine.append( "/BitsPerComponent " ); 9692 aLine.append( nBitsPerComponent ); 9693 aLine.append( "/Length " ); 9694 aLine.append( nStreamLengthObject ); 9695 aLine.append( " 0 R\n" ); 9696 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 9697 if( nBitsPerComponent != 1 ) 9698 { 9699 aLine.append( "/Filter/FlateDecode" ); 9700 } 9701 else 9702 { 9703 aLine.append( "/Filter/CCITTFaxDecode/DecodeParms<</K -1/BlackIs1 true/Columns " ); 9704 aLine.append( (sal_Int32)aBitmap.GetSizePixel().Width() ); 9705 aLine.append( ">>\n" ); 9706 } 9707 #endif 9708 if( ! bMask ) 9709 { 9710 aLine.append( "/ColorSpace" ); 9711 if( bTrueColor ) 9712 aLine.append( "/DeviceRGB\n" ); 9713 else if( aBitmap.HasGreyPalette() ) 9714 { 9715 aLine.append( "/DeviceGray\n" ); 9716 if( aBitmap.GetBitCount() == 1 ) 9717 { 9718 // #i47395# 1 bit bitmaps occasionally have an inverted grey palette 9719 sal_Int32 nBlackIndex = pAccess->GetBestPaletteIndex( BitmapColor( Color( COL_BLACK ) ) ); 9720 DBG_ASSERT( nBlackIndex == 0 || nBlackIndex == 1, "wrong black index" ); 9721 if( nBlackIndex == 1 ) 9722 aLine.append( "/Decode[1 0]\n" ); 9723 } 9724 } 9725 else 9726 { 9727 aLine.append( "[ /Indexed/DeviceRGB " ); 9728 aLine.append( (sal_Int32)(pAccess->GetPaletteEntryCount()-1) ); 9729 aLine.append( "\n<" ); 9730 if( m_aContext.Encryption.Encrypt() ) 9731 { 9732 enableStringEncryption( rObject.m_nObject ); 9733 //check encryption buffer size 9734 if( checkEncryptionBufferSize( pAccess->GetPaletteEntryCount()*3 ) ) 9735 { 9736 int nChar = 0; 9737 //fill the encryption buffer 9738 for( sal_uInt16 i = 0; i < pAccess->GetPaletteEntryCount(); i++ ) 9739 { 9740 const BitmapColor& rColor = pAccess->GetPaletteColor( i ); 9741 m_pEncryptionBuffer[nChar++] = rColor.GetRed(); 9742 m_pEncryptionBuffer[nChar++] = rColor.GetGreen(); 9743 m_pEncryptionBuffer[nChar++] = rColor.GetBlue(); 9744 } 9745 //encrypt the colorspace lookup table 9746 rtl_cipher_encodeARCFOUR( m_aCipher, m_pEncryptionBuffer, nChar, m_pEncryptionBuffer, nChar ); 9747 //now queue the data for output 9748 nChar = 0; 9749 for( sal_uInt16 i = 0; i < pAccess->GetPaletteEntryCount(); i++ ) 9750 { 9751 appendHex(m_pEncryptionBuffer[nChar++], aLine ); 9752 appendHex(m_pEncryptionBuffer[nChar++], aLine ); 9753 appendHex(m_pEncryptionBuffer[nChar++], aLine ); 9754 } 9755 } 9756 } 9757 else //no encryption requested (PDF/A-1a program flow drops here) 9758 { 9759 for( sal_uInt16 i = 0; i < pAccess->GetPaletteEntryCount(); i++ ) 9760 { 9761 const BitmapColor& rColor = pAccess->GetPaletteColor( i ); 9762 appendHex( rColor.GetRed(), aLine ); 9763 appendHex( rColor.GetGreen(), aLine ); 9764 appendHex( rColor.GetBlue(), aLine ); 9765 } 9766 } 9767 aLine.append( ">\n]\n" ); 9768 } 9769 } 9770 else 9771 { 9772 if( aBitmap.GetBitCount() == 1 ) 9773 { 9774 aLine.append( "/ImageMask true\n" ); 9775 sal_Int32 nBlackIndex = pAccess->GetBestPaletteIndex( BitmapColor( Color( COL_BLACK ) ) ); 9776 DBG_ASSERT( nBlackIndex == 0 || nBlackIndex == 1, "wrong black index" ); 9777 if( nBlackIndex ) 9778 aLine.append( "/Decode[ 1 0 ]\n" ); 9779 else 9780 aLine.append( "/Decode[ 0 1 ]\n" ); 9781 } 9782 else if( aBitmap.GetBitCount() == 8 ) 9783 { 9784 aLine.append( "/ColorSpace/DeviceGray\n" 9785 "/Decode [ 1 0 ]\n" ); 9786 } 9787 } 9788 9789 if( ! bMask && m_aContext.Version > PDFWriter::PDF_1_2 && !m_bIsPDF_A1 )//i59651 9790 { 9791 if( bWriteMask ) 9792 { 9793 nMaskObject = createObject(); 9794 if( rObject.m_aBitmap.IsAlpha() && m_aContext.Version > PDFWriter::PDF_1_3 ) 9795 aLine.append( "/SMask " ); 9796 else 9797 aLine.append( "/Mask " ); 9798 aLine.append( nMaskObject ); 9799 aLine.append( " 0 R\n" ); 9800 } 9801 else if( aTransparentColor != Color( COL_TRANSPARENT ) ) 9802 { 9803 aLine.append( "/Mask[ " ); 9804 if( bTrueColor ) 9805 { 9806 aLine.append( (sal_Int32)aTransparentColor.GetRed() ); 9807 aLine.append( ' ' ); 9808 aLine.append( (sal_Int32)aTransparentColor.GetRed() ); 9809 aLine.append( ' ' ); 9810 aLine.append( (sal_Int32)aTransparentColor.GetGreen() ); 9811 aLine.append( ' ' ); 9812 aLine.append( (sal_Int32)aTransparentColor.GetGreen() ); 9813 aLine.append( ' ' ); 9814 aLine.append( (sal_Int32)aTransparentColor.GetBlue() ); 9815 aLine.append( ' ' ); 9816 aLine.append( (sal_Int32)aTransparentColor.GetBlue() ); 9817 } 9818 else 9819 { 9820 sal_Int32 nIndex = pAccess->GetBestPaletteIndex( BitmapColor( aTransparentColor ) ); 9821 aLine.append( nIndex ); 9822 } 9823 aLine.append( " ]\n" ); 9824 } 9825 } 9826 else if( m_bIsPDF_A1 && (bWriteMask || aTransparentColor != Color( COL_TRANSPARENT )) ) 9827 m_aErrors.insert( PDFWriter::Warning_Transparency_Omitted_PDFA ); 9828 9829 aLine.append( ">>\n" 9830 "stream\n" ); 9831 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9832 sal_uInt64 nStartPos = 0; 9833 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nStartPos )) ); 9834 9835 checkAndEnableStreamEncryption( rObject.m_nObject ); 9836 #ifndef DEBUG_DISABLE_PDFCOMPRESSION 9837 if( nBitsPerComponent == 1 ) 9838 { 9839 writeG4Stream( pAccess ); 9840 } 9841 else 9842 #endif 9843 { 9844 beginCompression(); 9845 if( ! bTrueColor || pAccess->GetScanlineFormat() == BMP_FORMAT_24BIT_TC_RGB ) 9846 { 9847 const int nScanLineBytes = 1 + ( pAccess->GetBitCount() * ( pAccess->Width() - 1 ) / 8U ); 9848 9849 for( int i = 0; i < pAccess->Height(); i++ ) 9850 { 9851 CHECK_RETURN( writeBuffer( pAccess->GetScanline( i ), nScanLineBytes ) ); 9852 } 9853 } 9854 else 9855 { 9856 const int nScanLineBytes = pAccess->Width()*3; 9857 boost::shared_array<sal_uInt8> pCol( new sal_uInt8[ nScanLineBytes ] ); 9858 for( int y = 0; y < pAccess->Height(); y++ ) 9859 { 9860 for( int x = 0; x < pAccess->Width(); x++ ) 9861 { 9862 BitmapColor aColor = pAccess->GetColor( y, x ); 9863 pCol[3*x+0] = aColor.GetRed(); 9864 pCol[3*x+1] = aColor.GetGreen(); 9865 pCol[3*x+2] = aColor.GetBlue(); 9866 } 9867 CHECK_RETURN( writeBuffer( pCol.get(), nScanLineBytes ) ); 9868 } 9869 } 9870 endCompression(); 9871 } 9872 disableStreamEncryption(); 9873 9874 sal_uInt64 nEndPos = 0; 9875 CHECK_RETURN( (osl_File_E_None == osl_getFilePos( m_aFile, &nEndPos )) ); 9876 aLine.setLength( 0 ); 9877 aLine.append( "\nendstream\nendobj\n\n" ); 9878 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9879 CHECK_RETURN( updateObject( nStreamLengthObject ) ); 9880 aLine.setLength( 0 ); 9881 aLine.append( nStreamLengthObject ); 9882 aLine.append( " 0 obj\n" ); 9883 aLine.append( (sal_Int64)(nEndPos-nStartPos) ); 9884 aLine.append( "\nendobj\n\n" ); 9885 CHECK_RETURN( writeBuffer( aLine.getStr(), aLine.getLength() ) ); 9886 9887 if( nMaskObject ) 9888 { 9889 BitmapEmit aEmit; 9890 aEmit.m_nObject = nMaskObject; 9891 aEmit.m_aBitmap = rObject.m_aBitmap; 9892 return writeBitmapObject( aEmit, true ); 9893 } 9894 9895 return true; 9896 } 9897 9898 void PDFWriterImpl::drawJPGBitmap( SvStream& rDCTData, bool bIsTrueColor, const Size& rSizePixel, const Rectangle& rTargetArea, const Bitmap& rMask ) 9899 { 9900 MARK( "drawJPGBitmap" ); 9901 9902 OStringBuffer aLine( 80 ); 9903 updateGraphicsState(); 9904 9905 // #i40055# sanity check 9906 if( ! (rTargetArea.GetWidth() && rTargetArea.GetHeight() ) ) 9907 return; 9908 if( ! (rSizePixel.Width() && rSizePixel.Height()) ) 9909 return; 9910 9911 rDCTData.Seek( 0 ); 9912 if( bIsTrueColor && m_aContext.ColorMode == PDFWriter::DrawGreyscale ) 9913 { 9914 // need to convert to grayscale; 9915 // load stream to bitmap and draw the bitmap instead 9916 Graphic aGraphic; 9917 GraphicConverter::Import( rDCTData, aGraphic, CVT_JPG ); 9918 Bitmap aBmp( aGraphic.GetBitmap() ); 9919 if( !!rMask && rMask.GetSizePixel() == aBmp.GetSizePixel() ) 9920 { 9921 BitmapEx aBmpEx( aBmp, rMask ); 9922 drawBitmap( rTargetArea.TopLeft(), rTargetArea.GetSize(), aBmpEx ); 9923 } 9924 else 9925 drawBitmap( rTargetArea.TopLeft(), rTargetArea.GetSize(), aBmp ); 9926 return; 9927 } 9928 9929 SvMemoryStream* pStream = new SvMemoryStream; 9930 *pStream << rDCTData; 9931 pStream->Seek( STREAM_SEEK_TO_END ); 9932 9933 BitmapID aID; 9934 aID.m_aPixelSize = rSizePixel; 9935 aID.m_nSize = pStream->Tell(); 9936 pStream->Seek( STREAM_SEEK_TO_BEGIN ); 9937 aID.m_nChecksum = rtl_crc32( 0, pStream->GetData(), aID.m_nSize ); 9938 if( ! rMask.IsEmpty() ) 9939 aID.m_nMaskChecksum = rMask.GetChecksum(); 9940 9941 std::list< JPGEmit >::const_iterator it; 9942 for( it = m_aJPGs.begin(); it != m_aJPGs.end() && ! (aID == it->m_aID); ++it ) 9943 ; 9944 if( it == m_aJPGs.end() ) 9945 { 9946 m_aJPGs.push_front( JPGEmit() ); 9947 JPGEmit& rEmit = m_aJPGs.front(); 9948 rEmit.m_nObject = createObject(); 9949 rEmit.m_aID = aID; 9950 rEmit.m_pStream = pStream; 9951 rEmit.m_bTrueColor = bIsTrueColor; 9952 if( !! rMask && rMask.GetSizePixel() == rSizePixel ) 9953 rEmit.m_aMask = rMask; 9954 9955 it = m_aJPGs.begin(); 9956 } 9957 else 9958 delete pStream; 9959 9960 aLine.append( "q " ); 9961 sal_Int32 nCheckWidth = 0; 9962 m_aPages.back().appendMappedLength( (sal_Int32)rTargetArea.GetWidth(), aLine, false, &nCheckWidth ); 9963 aLine.append( " 0 0 " ); 9964 sal_Int32 nCheckHeight = 0; 9965 m_aPages.back().appendMappedLength( (sal_Int32)rTargetArea.GetHeight(), aLine, true, &nCheckHeight ); 9966 aLine.append( ' ' ); 9967 m_aPages.back().appendPoint( rTargetArea.BottomLeft(), aLine ); 9968 aLine.append( " cm\n/Im" ); 9969 aLine.append( it->m_nObject ); 9970 aLine.append( " Do Q\n" ); 9971 if( nCheckWidth == 0 || nCheckHeight == 0 ) 9972 { 9973 // #i97512# avoid invalid current matrix 9974 aLine.setLength( 0 ); 9975 aLine.append( "\n%jpeg image /Im" ); 9976 aLine.append( it->m_nObject ); 9977 aLine.append( " scaled to zero size, omitted\n" ); 9978 } 9979 writeBuffer( aLine.getStr(), aLine.getLength() ); 9980 9981 OStringBuffer aObjName( 16 ); 9982 aObjName.append( "Im" ); 9983 aObjName.append( it->m_nObject ); 9984 pushResource( ResXObject, aObjName.makeStringAndClear(), it->m_nObject ); 9985 9986 } 9987 9988 void PDFWriterImpl::drawBitmap( const Point& rDestPoint, const Size& rDestSize, const BitmapEmit& rBitmap, const Color& rFillColor ) 9989 { 9990 OStringBuffer aLine( 80 ); 9991 updateGraphicsState(); 9992 9993 aLine.append( "q " ); 9994 if( rFillColor != Color( COL_TRANSPARENT ) ) 9995 { 9996 appendNonStrokingColor( rFillColor, aLine ); 9997 aLine.append( ' ' ); 9998 } 9999 sal_Int32 nCheckWidth = 0; 10000 m_aPages.back().appendMappedLength( (sal_Int32)rDestSize.Width(), aLine, false, &nCheckWidth ); 10001 aLine.append( " 0 0 " ); 10002 sal_Int32 nCheckHeight = 0; 10003 m_aPages.back().appendMappedLength( (sal_Int32)rDestSize.Height(), aLine, true, &nCheckHeight ); 10004 aLine.append( ' ' ); 10005 m_aPages.back().appendPoint( rDestPoint + Point( 0, rDestSize.Height()-1 ), aLine ); 10006 aLine.append( " cm\n/Im" ); 10007 aLine.append( rBitmap.m_nObject ); 10008 aLine.append( " Do Q\n" ); 10009 if( nCheckWidth == 0 || nCheckHeight == 0 ) 10010 { 10011 // #i97512# avoid invalid current matrix 10012 aLine.setLength( 0 ); 10013 aLine.append( "\n%bitmap image /Im" ); 10014 aLine.append( rBitmap.m_nObject ); 10015 aLine.append( " scaled to zero size, omitted\n" ); 10016 } 10017 writeBuffer( aLine.getStr(), aLine.getLength() ); 10018 } 10019 10020 const PDFWriterImpl::BitmapEmit& PDFWriterImpl::createBitmapEmit( const BitmapEx& i_rBitmap, bool bDrawMask ) 10021 { 10022 BitmapEx aBitmap( i_rBitmap ); 10023 if( m_aContext.ColorMode == PDFWriter::DrawGreyscale ) 10024 { 10025 BmpConversion eConv = BMP_CONVERSION_8BIT_GREYS; 10026 int nDepth = aBitmap.GetBitmap().GetBitCount(); 10027 if( nDepth <= 4 ) 10028 eConv = BMP_CONVERSION_4BIT_GREYS; 10029 if( nDepth > 1 ) 10030 aBitmap.Convert( eConv ); 10031 } 10032 BitmapID aID; 10033 aID.m_aPixelSize = aBitmap.GetSizePixel(); 10034 aID.m_nSize = aBitmap.GetBitCount(); 10035 aID.m_nChecksum = aBitmap.GetBitmap().GetChecksum(); 10036 aID.m_nMaskChecksum = 0; 10037 if( aBitmap.IsAlpha() ) 10038 aID.m_nMaskChecksum = aBitmap.GetAlpha().GetChecksum(); 10039 else 10040 { 10041 Bitmap aMask = aBitmap.GetMask(); 10042 if( ! aMask.IsEmpty() ) 10043 aID.m_nMaskChecksum = aMask.GetChecksum(); 10044 } 10045 std::list< BitmapEmit >::const_iterator it; 10046 for( it = m_aBitmaps.begin(); it != m_aBitmaps.end(); ++it ) 10047 { 10048 if( aID == it->m_aID ) 10049 break; 10050 } 10051 if( it == m_aBitmaps.end() ) 10052 { 10053 m_aBitmaps.push_front( BitmapEmit() ); 10054 m_aBitmaps.front().m_aID = aID; 10055 m_aBitmaps.front().m_aBitmap = aBitmap; 10056 m_aBitmaps.front().m_nObject = createObject(); 10057 m_aBitmaps.front().m_bDrawMask = bDrawMask; 10058 it = m_aBitmaps.begin(); 10059 } 10060 10061 OStringBuffer aObjName( 16 ); 10062 aObjName.append( "Im" ); 10063 aObjName.append( it->m_nObject ); 10064 pushResource( ResXObject, aObjName.makeStringAndClear(), it->m_nObject ); 10065 10066 return *it; 10067 } 10068 10069 void PDFWriterImpl::drawBitmap( const Point& rDestPoint, const Size& rDestSize, const Bitmap& rBitmap ) 10070 { 10071 MARK( "drawBitmap (Bitmap)" ); 10072 10073 // #i40055# sanity check 10074 if( ! (rDestSize.Width() && rDestSize.Height()) ) 10075 return; 10076 10077 const BitmapEmit& rEmit = createBitmapEmit( BitmapEx( rBitmap ) ); 10078 drawBitmap( rDestPoint, rDestSize, rEmit, Color( COL_TRANSPARENT ) ); 10079 } 10080 10081 void PDFWriterImpl::drawBitmap( const Point& rDestPoint, const Size& rDestSize, const BitmapEx& rBitmap ) 10082 { 10083 MARK( "drawBitmap (BitmapEx)" ); 10084 10085 // #i40055# sanity check 10086 if( ! (rDestSize.Width() && rDestSize.Height()) ) 10087 return; 10088 10089 const BitmapEmit& rEmit = createBitmapEmit( rBitmap ); 10090 drawBitmap( rDestPoint, rDestSize, rEmit, Color( COL_TRANSPARENT ) ); 10091 } 10092 10093 void PDFWriterImpl::drawMask( const Point& rDestPoint, const Size& rDestSize, const Bitmap& rBitmap, const Color& rFillColor ) 10094 { 10095 MARK( "drawMask" ); 10096 10097 // #i40055# sanity check 10098 if( ! (rDestSize.Width() && rDestSize.Height()) ) 10099 return; 10100 10101 Bitmap aBitmap( rBitmap ); 10102 if( aBitmap.GetBitCount() > 1 ) 10103 aBitmap.Convert( BMP_CONVERSION_1BIT_THRESHOLD ); 10104 DBG_ASSERT( aBitmap.GetBitCount() == 1, "mask conversion failed" ); 10105 10106 const BitmapEmit& rEmit = createBitmapEmit( BitmapEx( aBitmap ), true ); 10107 drawBitmap( rDestPoint, rDestSize, rEmit, rFillColor ); 10108 } 10109 10110 sal_Int32 PDFWriterImpl::createGradient( const Gradient& rGradient, const Size& rSize ) 10111 { 10112 Size aPtSize( lcl_convert( m_aGraphicsStack.front().m_aMapMode, 10113 MapMode( MAP_POINT ), 10114 getReferenceDevice(), 10115 rSize ) ); 10116 // check if we already have this gradient 10117 std::list<GradientEmit>::iterator it; 10118 // rounding to point will generally lose some pixels 10119 // round up to point boundary 10120 aPtSize.Width()++; 10121 aPtSize.Height()++; 10122 for( it = m_aGradients.begin(); it != m_aGradients.end(); ++it ) 10123 { 10124 if( it->m_aGradient == rGradient ) 10125 { 10126 if( it->m_aSize == aPtSize ) 10127 break; 10128 } 10129 } 10130 if( it == m_aGradients.end() ) 10131 { 10132 m_aGradients.push_front( GradientEmit() ); 10133 m_aGradients.front().m_aGradient = rGradient; 10134 m_aGradients.front().m_nObject = createObject(); 10135 m_aGradients.front().m_aSize = aPtSize; 10136 it = m_aGradients.begin(); 10137 } 10138 10139 OStringBuffer aObjName( 16 ); 10140 aObjName.append( 'P' ); 10141 aObjName.append( it->m_nObject ); 10142 pushResource( ResShading, aObjName.makeStringAndClear(), it->m_nObject ); 10143 10144 return it->m_nObject; 10145 } 10146 10147 void PDFWriterImpl::drawGradient( const Rectangle& rRect, const Gradient& rGradient ) 10148 { 10149 MARK( "drawGradient (Rectangle)" ); 10150 10151 if( m_aContext.Version == PDFWriter::PDF_1_2 ) 10152 { 10153 drawRectangle( rRect ); 10154 return; 10155 } 10156 10157 sal_Int32 nGradient = createGradient( rGradient, rRect.GetSize() ); 10158 10159 Point aTranslate( rRect.BottomLeft() ); 10160 aTranslate += Point( 0, 1 ); 10161 10162 updateGraphicsState(); 10163 10164 OStringBuffer aLine( 80 ); 10165 aLine.append( "q 1 0 0 1 " ); 10166 m_aPages.back().appendPoint( aTranslate, aLine ); 10167 aLine.append( " cm " ); 10168 // if a stroke is appended reset the clip region before stroke 10169 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 10170 aLine.append( "q " ); 10171 aLine.append( "0 0 " ); 10172 m_aPages.back().appendMappedLength( (sal_Int32)rRect.GetWidth(), aLine, false ); 10173 aLine.append( ' ' ); 10174 m_aPages.back().appendMappedLength( (sal_Int32)rRect.GetHeight(), aLine, true ); 10175 aLine.append( " re W n\n" ); 10176 10177 aLine.append( "/P" ); 10178 aLine.append( nGradient ); 10179 aLine.append( " sh " ); 10180 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 10181 { 10182 aLine.append( "Q 0 0 " ); 10183 m_aPages.back().appendMappedLength( (sal_Int32)rRect.GetWidth(), aLine, false ); 10184 aLine.append( ' ' ); 10185 m_aPages.back().appendMappedLength( (sal_Int32)rRect.GetHeight(), aLine, true ); 10186 aLine.append( " re S " ); 10187 } 10188 aLine.append( "Q\n" ); 10189 writeBuffer( aLine.getStr(), aLine.getLength() ); 10190 } 10191 10192 void PDFWriterImpl::drawGradient( const PolyPolygon& rPolyPoly, const Gradient& rGradient ) 10193 { 10194 MARK( "drawGradient (PolyPolygon)" ); 10195 10196 if( m_aContext.Version == PDFWriter::PDF_1_2 ) 10197 { 10198 drawPolyPolygon( rPolyPoly ); 10199 return; 10200 } 10201 10202 Rectangle aBoundRect = rPolyPoly.GetBoundRect(); 10203 sal_Int32 nGradient = createGradient( rGradient, aBoundRect.GetSize() ); 10204 10205 updateGraphicsState(); 10206 10207 Point aTranslate = aBoundRect.BottomLeft(); 10208 int nPolygons = rPolyPoly.Count(); 10209 10210 OStringBuffer aLine( 80*nPolygons ); 10211 aLine.append( "q " ); 10212 // set PolyPolygon as clip path 10213 m_aPages.back().appendPolyPolygon( rPolyPoly, aLine ); 10214 aLine.append( "W* n\n" ); 10215 aLine.append( "1 0 0 1 " ); 10216 m_aPages.back().appendPoint( aTranslate, aLine ); 10217 aLine.append( " cm\n" ); 10218 aLine.append( "/P" ); 10219 aLine.append( nGradient ); 10220 aLine.append( " sh Q\n" ); 10221 if( m_aGraphicsStack.front().m_aLineColor != Color( COL_TRANSPARENT ) ) 10222 { 10223 // and draw the surrounding path 10224 m_aPages.back().appendPolyPolygon( rPolyPoly, aLine ); 10225 aLine.append( "S\n" ); 10226 } 10227 writeBuffer( aLine.getStr(), aLine.getLength() ); 10228 } 10229 10230 void PDFWriterImpl::drawHatch( const PolyPolygon& rPolyPoly, const Hatch& rHatch ) 10231 { 10232 MARK( "drawHatch" ); 10233 10234 updateGraphicsState(); 10235 10236 if( m_aGraphicsStack.front().m_aLineColor == Color( COL_TRANSPARENT ) && 10237 m_aGraphicsStack.front().m_aFillColor == Color( COL_TRANSPARENT ) ) 10238 return; 10239 if( rPolyPoly.Count() ) 10240 { 10241 PolyPolygon aPolyPoly( rPolyPoly ); 10242 10243 aPolyPoly.Optimize( POLY_OPTIMIZE_NO_SAME ); 10244 push( PUSH_LINECOLOR ); 10245 setLineColor( rHatch.GetColor() ); 10246 getReferenceDevice()->ImplDrawHatch( aPolyPoly, rHatch, sal_False ); 10247 pop(); 10248 } 10249 } 10250 10251 void PDFWriterImpl::drawWallpaper( const Rectangle& rRect, const Wallpaper& rWall ) 10252 { 10253 MARK( "drawWallpaper" ); 10254 10255 bool bDrawColor = false; 10256 bool bDrawGradient = false; 10257 bool bDrawBitmap = false; 10258 10259 BitmapEx aBitmap; 10260 Point aBmpPos = rRect.TopLeft(); 10261 Size aBmpSize; 10262 if( rWall.IsBitmap() ) 10263 { 10264 aBitmap = rWall.GetBitmap(); 10265 aBmpSize = lcl_convert( aBitmap.GetPrefMapMode(), 10266 getMapMode(), 10267 getReferenceDevice(), 10268 aBitmap.GetPrefSize() ); 10269 Rectangle aRect( rRect ); 10270 if( rWall.IsRect() ) 10271 { 10272 aRect = rWall.GetRect(); 10273 aBmpPos = aRect.TopLeft(); 10274 aBmpSize = aRect.GetSize(); 10275 } 10276 if( rWall.GetStyle() != WALLPAPER_SCALE ) 10277 { 10278 if( rWall.GetStyle() != WALLPAPER_TILE ) 10279 { 10280 bDrawBitmap = true; 10281 if( rWall.IsGradient() ) 10282 bDrawGradient = true; 10283 else 10284 bDrawColor = true; 10285 switch( rWall.GetStyle() ) 10286 { 10287 case WALLPAPER_TOPLEFT: 10288 break; 10289 case WALLPAPER_TOP: 10290 aBmpPos.X() += (aRect.GetWidth()-aBmpSize.Width())/2; 10291 break; 10292 case WALLPAPER_LEFT: 10293 aBmpPos.Y() += (aRect.GetHeight()-aBmpSize.Height())/2; 10294 break; 10295 case WALLPAPER_TOPRIGHT: 10296 aBmpPos.X() += aRect.GetWidth()-aBmpSize.Width(); 10297 break; 10298 case WALLPAPER_CENTER: 10299 aBmpPos.X() += (aRect.GetWidth()-aBmpSize.Width())/2; 10300 aBmpPos.Y() += (aRect.GetHeight()-aBmpSize.Height())/2; 10301 break; 10302 case WALLPAPER_RIGHT: 10303 aBmpPos.X() += aRect.GetWidth()-aBmpSize.Width(); 10304 aBmpPos.Y() += (aRect.GetHeight()-aBmpSize.Height())/2; 10305 break; 10306 case WALLPAPER_BOTTOMLEFT: 10307 aBmpPos.Y() += aRect.GetHeight()-aBmpSize.Height(); 10308 break; 10309 case WALLPAPER_BOTTOM: 10310 aBmpPos.X() += (aRect.GetWidth()-aBmpSize.Width())/2; 10311 aBmpPos.Y() += aRect.GetHeight()-aBmpSize.Height(); 10312 break; 10313 case WALLPAPER_BOTTOMRIGHT: 10314 aBmpPos.X() += aRect.GetWidth()-aBmpSize.Width(); 10315 aBmpPos.Y() += aRect.GetHeight()-aBmpSize.Height(); 10316 break; 10317 default: ; 10318 } 10319 } 10320 else 10321 { 10322 // push the bitmap 10323 const BitmapEmit& rEmit = createBitmapEmit( BitmapEx( aBitmap ) ); 10324 10325 // convert to page coordinates; this needs to be done here 10326 // since the emit does not know the page anymore 10327 Rectangle aConvertRect( aBmpPos, aBmpSize ); 10328 m_aPages.back().convertRect( aConvertRect ); 10329 10330 OStringBuffer aNameBuf(16); 10331 aNameBuf.append( "Im" ); 10332 aNameBuf.append( rEmit.m_nObject ); 10333 OString aImageName( aNameBuf.makeStringAndClear() ); 10334 10335 // push the pattern 10336 OStringBuffer aTilingStream( 32 ); 10337 appendFixedInt( aConvertRect.GetWidth(), aTilingStream ); 10338 aTilingStream.append( " 0 0 " ); 10339 appendFixedInt( aConvertRect.GetHeight(), aTilingStream ); 10340 aTilingStream.append( " 0 0 cm\n/" ); 10341 aTilingStream.append( aImageName ); 10342 aTilingStream.append( " Do\n" ); 10343 10344 m_aTilings.push_back( TilingEmit() ); 10345 m_aTilings.back().m_nObject = createObject(); 10346 m_aTilings.back().m_aRectangle = Rectangle( Point( 0, 0 ), aConvertRect.GetSize() ); 10347 m_aTilings.back().m_pTilingStream = new SvMemoryStream(); 10348 m_aTilings.back().m_pTilingStream->Write( aTilingStream.getStr(), aTilingStream.getLength() ); 10349 // phase the tiling so wallpaper begins on upper left 10350 m_aTilings.back().m_aTransform.matrix[2] = double(aConvertRect.Left() % aConvertRect.GetWidth()) / fDivisor; 10351 m_aTilings.back().m_aTransform.matrix[5] = double(aConvertRect.Top() % aConvertRect.GetHeight()) / fDivisor; 10352 m_aTilings.back().m_aResources.m_aXObjects[aImageName] = rEmit.m_nObject; 10353 10354 updateGraphicsState(); 10355 10356 OStringBuffer aObjName( 16 ); 10357 aObjName.append( 'P' ); 10358 aObjName.append( m_aTilings.back().m_nObject ); 10359 OString aPatternName( aObjName.makeStringAndClear() ); 10360 pushResource( ResPattern, aPatternName, m_aTilings.back().m_nObject ); 10361 10362 // fill a rRect with the pattern 10363 OStringBuffer aLine( 100 ); 10364 aLine.append( "q /Pattern cs /" ); 10365 aLine.append( aPatternName ); 10366 aLine.append( " scn " ); 10367 m_aPages.back().appendRect( rRect, aLine ); 10368 aLine.append( " f Q\n" ); 10369 writeBuffer( aLine.getStr(), aLine.getLength() ); 10370 } 10371 } 10372 else 10373 { 10374 aBmpPos = aRect.TopLeft(); 10375 aBmpSize = aRect.GetSize(); 10376 bDrawBitmap = true; 10377 } 10378 10379 if( aBitmap.IsTransparent() ) 10380 { 10381 if( rWall.IsGradient() ) 10382 bDrawGradient = true; 10383 else 10384 bDrawColor = true; 10385 } 10386 } 10387 else if( rWall.IsGradient() ) 10388 bDrawGradient = true; 10389 else 10390 bDrawColor = true; 10391 10392 if( bDrawGradient ) 10393 { 10394 drawGradient( rRect, rWall.GetGradient() ); 10395 } 10396 if( bDrawColor ) 10397 { 10398 Color aOldLineColor = m_aGraphicsStack.front().m_aLineColor; 10399 Color aOldFillColor = m_aGraphicsStack.front().m_aFillColor; 10400 setLineColor( Color( COL_TRANSPARENT ) ); 10401 setFillColor( rWall.GetColor() ); 10402 drawRectangle( rRect ); 10403 setLineColor( aOldLineColor ); 10404 setFillColor( aOldFillColor ); 10405 } 10406 if( bDrawBitmap ) 10407 { 10408 // set temporary clip region since aBmpPos and aBmpSize 10409 // may be outside rRect 10410 OStringBuffer aLine( 20 ); 10411 aLine.append( "q " ); 10412 m_aPages.back().appendRect( rRect, aLine ); 10413 aLine.append( " W n\n" ); 10414 writeBuffer( aLine.getStr(), aLine.getLength() ); 10415 drawBitmap( aBmpPos, aBmpSize, aBitmap ); 10416 writeBuffer( "Q\n", 2 ); 10417 } 10418 } 10419 10420 void PDFWriterImpl::beginPattern( const Rectangle& rCellRect ) 10421 { 10422 beginRedirect( new SvMemoryStream(), rCellRect ); 10423 } 10424 10425 sal_Int32 PDFWriterImpl::endPattern( const SvtGraphicFill::Transform& rTransform ) 10426 { 10427 Rectangle aConvertRect( getRedirectTargetRect() ); 10428 DBG_ASSERT( aConvertRect.GetWidth() != 0 && aConvertRect.GetHeight() != 0, "empty cell rectangle in pattern" ); 10429 10430 // get scaling between current mapmode and PDF output 10431 Size aScaling( lcl_convert( m_aGraphicsStack.front().m_aMapMode, m_aMapMode, getReferenceDevice(), Size( 10000, 10000 ) ) ); 10432 double fSX = (double(aScaling.Width()) / 10000.0); 10433 double fSY = (double(aScaling.Height()) / 10000.0); 10434 10435 // transform translation part of matrix 10436 Size aTranslation( (long)rTransform.matrix[2], (long)rTransform.matrix[5] ); 10437 aTranslation = lcl_convert( m_aGraphicsStack.front().m_aMapMode, m_aMapMode, getReferenceDevice(), aTranslation ); 10438 10439 sal_Int32 nTilingId = m_aTilings.size(); 10440 m_aTilings.push_back( TilingEmit() ); 10441 TilingEmit& rTile = m_aTilings.back(); 10442 rTile.m_nObject = createObject(); 10443 rTile.m_aResources = m_aOutputStreams.front().m_aResourceDict; 10444 rTile.m_aTransform.matrix[0] = rTransform.matrix[0] * fSX; 10445 rTile.m_aTransform.matrix[1] = rTransform.matrix[1] * fSY; 10446 rTile.m_aTransform.matrix[2] = aTranslation.Width(); 10447 rTile.m_aTransform.matrix[3] = rTransform.matrix[3] * fSX; 10448 rTile.m_aTransform.matrix[4] = rTransform.matrix[4] * fSY; 10449 rTile.m_aTransform.matrix[5] = -aTranslation.Height(); 10450 // caution: endRedirect pops the stream, so do this last 10451 rTile.m_pTilingStream = dynamic_cast<SvMemoryStream*>(endRedirect()); 10452 // FIXME: bound rect will not work with rotated matrix 10453 rTile.m_aRectangle = Rectangle( Point(0,0), aConvertRect.GetSize() ); 10454 rTile.m_aCellSize = aConvertRect.GetSize(); 10455 10456 OStringBuffer aObjName( 16 ); 10457 aObjName.append( 'P' ); 10458 aObjName.append( rTile.m_nObject ); 10459 pushResource( ResPattern, aObjName.makeStringAndClear(), rTile.m_nObject ); 10460 return nTilingId; 10461 } 10462 10463 void PDFWriterImpl::drawPolyPolygon( const PolyPolygon& rPolyPoly, sal_Int32 nPattern, bool bEOFill ) 10464 { 10465 if( nPattern < 0 || nPattern >= (sal_Int32)m_aTilings.size() ) 10466 return; 10467 10468 m_aPages.back().endStream(); 10469 sal_Int32 nXObject = createObject(); 10470 OStringBuffer aNameBuf( 16 ); 10471 aNameBuf.append( "Pol" ); 10472 aNameBuf.append( nXObject ); 10473 OString aObjName( aNameBuf.makeStringAndClear() ); 10474 Rectangle aObjRect; 10475 if( updateObject( nXObject ) ) 10476 { 10477 // get bounding rect of object 10478 PolyPolygon aSubDiv; 10479 rPolyPoly.AdaptiveSubdivide( aSubDiv ); 10480 aObjRect = aSubDiv.GetBoundRect(); 10481 Rectangle aConvObjRect( aObjRect ); 10482 m_aPages.back().convertRect( aConvObjRect ); 10483 10484 // move polypolygon to bottom left of page 10485 PolyPolygon aLocalPath( rPolyPoly ); 10486 sal_Int32 nPgWd = getReferenceDevice()->ImplGetDPIX() * m_aPages.back().getWidth() / 72; 10487 sal_Int32 nPgHt = getReferenceDevice()->ImplGetDPIY() * m_aPages.back().getHeight() / 72; 10488 Size aLogicPgSz = getReferenceDevice()->PixelToLogic( Size( nPgWd, nPgHt ), m_aGraphicsStack.front().m_aMapMode ); 10489 sal_Int32 nXOff = aObjRect.Left(); 10490 sal_Int32 nYOff = aLogicPgSz.Height() - aObjRect.Bottom(); 10491 aLocalPath.Move( -nXOff, nYOff ); 10492 10493 // prepare XObject's content stream 10494 OStringBuffer aStream( 512 ); 10495 aStream.append( "/Pattern cs /P" ); 10496 aStream.append( m_aTilings[ nPattern ].m_nObject ); 10497 aStream.append( " scn\n" ); 10498 m_aPages.back().appendPolyPolygon( aLocalPath, aStream ); 10499 aStream.append( bEOFill ? "f*" : "f" ); 10500 SvMemoryStream aMemStream( aStream.getLength() ); 10501 aMemStream.Write( aStream.getStr(), aStream.getLength() ); 10502 bool bDeflate = compressStream( &aMemStream ); 10503 aMemStream.Seek( STREAM_SEEK_TO_END ); 10504 sal_Int32 nStreamLen = (sal_Int32)aMemStream.Tell(); 10505 aMemStream.Seek( STREAM_SEEK_TO_BEGIN ); 10506 10507 // add new XObject to global resource dict 10508 m_aGlobalResourceDict.m_aXObjects[ aObjName ] = nXObject; 10509 10510 // write XObject 10511 OStringBuffer aLine( 512 ); 10512 aLine.append( nXObject ); 10513 aLine.append( " 0 obj\n" 10514 "<</Type/XObject/Subtype/Form/BBox[0 0 " ); 10515 appendFixedInt( aConvObjRect.GetWidth(), aLine ); 10516 aLine.append( ' ' ); 10517 appendFixedInt( aConvObjRect.GetHeight(), aLine ); 10518 aLine.append( "]/Length " ); 10519 aLine.append( nStreamLen ); 10520 if( bDeflate ) 10521 aLine.append( "/Filter/FlateDecode" ); 10522 aLine.append( ">>\n" 10523 "stream\n" ); 10524 writeBuffer( aLine.getStr(), aLine.getLength() ); 10525 checkAndEnableStreamEncryption( nXObject ); 10526 writeBuffer( aMemStream.GetData(), nStreamLen ); 10527 disableStreamEncryption(); 10528 writeBuffer( "\nendstream\nendobj\n\n", 19 ); 10529 } 10530 m_aPages.back().beginStream(); 10531 OStringBuffer aLine( 80 ); 10532 aLine.append( "q 1 0 0 1 " ); 10533 m_aPages.back().appendPoint( aObjRect.BottomLeft(), aLine ); 10534 aLine.append( " cm/" ); 10535 aLine.append( aObjName ); 10536 aLine.append( " Do Q\n" ); 10537 writeBuffer( aLine.getStr(), aLine.getLength() ); 10538 } 10539 10540 void PDFWriterImpl::updateGraphicsState() 10541 { 10542 OStringBuffer aLine( 256 ); 10543 GraphicsState& rNewState = m_aGraphicsStack.front(); 10544 // first set clip region since it might invalidate everything else 10545 10546 if( (rNewState.m_nUpdateFlags & GraphicsState::updateClipRegion) ) 10547 { 10548 rNewState.m_nUpdateFlags &= ~GraphicsState::updateClipRegion; 10549 10550 if( m_aCurrentPDFState.m_bClipRegion != rNewState.m_bClipRegion || 10551 ( rNewState.m_bClipRegion && m_aCurrentPDFState.m_aClipRegion != rNewState.m_aClipRegion ) ) 10552 { 10553 if( m_aCurrentPDFState.m_bClipRegion && m_aCurrentPDFState.m_aClipRegion.count() ) 10554 { 10555 aLine.append( "Q " ); 10556 // invalidate everything but the clip region 10557 m_aCurrentPDFState = GraphicsState(); 10558 rNewState.m_nUpdateFlags = sal::static_int_cast<sal_uInt16>(~GraphicsState::updateClipRegion); 10559 } 10560 if( rNewState.m_bClipRegion && rNewState.m_aClipRegion.count() ) 10561 { 10562 // clip region is always stored in private PDF mapmode 10563 MapMode aNewMapMode = rNewState.m_aMapMode; 10564 rNewState.m_aMapMode = m_aMapMode; 10565 getReferenceDevice()->SetMapMode( rNewState.m_aMapMode ); 10566 m_aCurrentPDFState.m_aMapMode = rNewState.m_aMapMode; 10567 10568 aLine.append( "q " ); 10569 m_aPages.back().appendPolyPolygon( rNewState.m_aClipRegion, aLine ); 10570 aLine.append( "W* n\n" ); 10571 rNewState.m_aMapMode = aNewMapMode; 10572 getReferenceDevice()->SetMapMode( rNewState.m_aMapMode ); 10573 m_aCurrentPDFState.m_aMapMode = rNewState.m_aMapMode; 10574 } 10575 } 10576 } 10577 10578 if( (rNewState.m_nUpdateFlags & GraphicsState::updateMapMode) ) 10579 { 10580 rNewState.m_nUpdateFlags &= ~GraphicsState::updateMapMode; 10581 getReferenceDevice()->SetMapMode( rNewState.m_aMapMode ); 10582 } 10583 10584 if( (rNewState.m_nUpdateFlags & GraphicsState::updateFont) ) 10585 { 10586 rNewState.m_nUpdateFlags &= ~GraphicsState::updateFont; 10587 getReferenceDevice()->SetFont( rNewState.m_aFont ); 10588 getReferenceDevice()->ImplNewFont(); 10589 } 10590 10591 if( (rNewState.m_nUpdateFlags & GraphicsState::updateLayoutMode) ) 10592 { 10593 rNewState.m_nUpdateFlags &= ~GraphicsState::updateLayoutMode; 10594 getReferenceDevice()->SetLayoutMode( rNewState.m_nLayoutMode ); 10595 } 10596 10597 if( (rNewState.m_nUpdateFlags & GraphicsState::updateDigitLanguage) ) 10598 { 10599 rNewState.m_nUpdateFlags &= ~GraphicsState::updateDigitLanguage; 10600 getReferenceDevice()->SetDigitLanguage( rNewState.m_aDigitLanguage ); 10601 } 10602 10603 if( (rNewState.m_nUpdateFlags & GraphicsState::updateLineColor) ) 10604 { 10605 rNewState.m_nUpdateFlags &= ~GraphicsState::updateLineColor; 10606 if( m_aCurrentPDFState.m_aLineColor != rNewState.m_aLineColor && 10607 rNewState.m_aLineColor != Color( COL_TRANSPARENT ) ) 10608 { 10609 appendStrokingColor( rNewState.m_aLineColor, aLine ); 10610 aLine.append( "\n" ); 10611 } 10612 } 10613 10614 if( (rNewState.m_nUpdateFlags & GraphicsState::updateFillColor) ) 10615 { 10616 rNewState.m_nUpdateFlags &= ~GraphicsState::updateFillColor; 10617 if( m_aCurrentPDFState.m_aFillColor != rNewState.m_aFillColor && 10618 rNewState.m_aFillColor != Color( COL_TRANSPARENT ) ) 10619 { 10620 appendNonStrokingColor( rNewState.m_aFillColor, aLine ); 10621 aLine.append( "\n" ); 10622 } 10623 } 10624 10625 if( (rNewState.m_nUpdateFlags & GraphicsState::updateTransparentPercent) ) 10626 { 10627 rNewState.m_nUpdateFlags &= ~GraphicsState::updateTransparentPercent; 10628 if( m_aContext.Version >= PDFWriter::PDF_1_4 && m_aCurrentPDFState.m_nTransparentPercent != rNewState.m_nTransparentPercent ) 10629 { 10630 // TODO: switch extended graphicsstate 10631 } 10632 } 10633 10634 // everything is up to date now 10635 m_aCurrentPDFState = m_aGraphicsStack.front(); 10636 if( aLine.getLength() ) 10637 writeBuffer( aLine.getStr(), aLine.getLength() ); 10638 } 10639 10640 /* #i47544# imitate OutputDevice behaviour: 10641 * if a font with a nontransparent color is set, it overwrites the current 10642 * text color. OTOH setting the text color will overwrite the color of the font. 10643 */ 10644 void PDFWriterImpl::setFont( const Font& rFont ) 10645 { 10646 Color aColor = rFont.GetColor(); 10647 if( aColor == Color( COL_TRANSPARENT ) ) 10648 aColor = m_aGraphicsStack.front().m_aFont.GetColor(); 10649 m_aGraphicsStack.front().m_aFont = rFont; 10650 m_aGraphicsStack.front().m_aFont.SetColor( aColor ); 10651 m_aGraphicsStack.front().m_nUpdateFlags |= GraphicsState::updateFont; 10652 } 10653 10654 void PDFWriterImpl::push( sal_uInt16 nFlags ) 10655 { 10656 OSL_ENSURE( m_aGraphicsStack.size() > 0, "invalid graphics stack" ); 10657 m_aGraphicsStack.push_front( m_aGraphicsStack.front() ); 10658 m_aGraphicsStack.front().m_nFlags = nFlags; 10659 } 10660 10661 void PDFWriterImpl::pop() 10662 { 10663 OSL_ENSURE( m_aGraphicsStack.size() > 1, "pop without push" ); 10664 if( m_aGraphicsStack.size() < 2 ) 10665 return; 10666 10667 GraphicsState aState = m_aGraphicsStack.front(); 10668 m_aGraphicsStack.pop_front(); 10669 GraphicsState& rOld = m_aGraphicsStack.front(); 10670 10671 // move those parameters back that were not pushed 10672 // in the first place 10673 if( ! (aState.m_nFlags & PUSH_LINECOLOR) ) 10674 setLineColor( aState.m_aLineColor ); 10675 if( ! (aState.m_nFlags & PUSH_FILLCOLOR) ) 10676 setFillColor( aState.m_aFillColor ); 10677 if( ! (aState.m_nFlags & PUSH_FONT) ) 10678 setFont( aState.m_aFont ); 10679 if( ! (aState.m_nFlags & PUSH_TEXTCOLOR) ) 10680 setTextColor( aState.m_aFont.GetColor() ); 10681 if( ! (aState.m_nFlags & PUSH_MAPMODE) ) 10682 setMapMode( aState.m_aMapMode ); 10683 if( ! (aState.m_nFlags & PUSH_CLIPREGION) ) 10684 { 10685 // do not use setClipRegion here 10686 // it would convert again assuming the current mapmode 10687 rOld.m_aClipRegion = aState.m_aClipRegion; 10688 rOld.m_bClipRegion = aState.m_bClipRegion; 10689 } 10690 if( ! (aState.m_nFlags & PUSH_TEXTLINECOLOR ) ) 10691 setTextLineColor( aState.m_aTextLineColor ); 10692 if( ! (aState.m_nFlags & PUSH_OVERLINECOLOR ) ) 10693 setOverlineColor( aState.m_aOverlineColor ); 10694 if( ! (aState.m_nFlags & PUSH_TEXTALIGN ) ) 10695 setTextAlign( aState.m_aFont.GetAlign() ); 10696 if( ! (aState.m_nFlags & PUSH_TEXTFILLCOLOR) ) 10697 setTextFillColor( aState.m_aFont.GetFillColor() ); 10698 if( ! (aState.m_nFlags & PUSH_REFPOINT) ) 10699 { 10700 // what ? 10701 } 10702 // invalidate graphics state 10703 m_aGraphicsStack.front().m_nUpdateFlags = sal::static_int_cast<sal_uInt16>(~0U); 10704 } 10705 10706 void PDFWriterImpl::setMapMode( const MapMode& rMapMode ) 10707 { 10708 m_aGraphicsStack.front().m_aMapMode = rMapMode; 10709 getReferenceDevice()->SetMapMode( rMapMode ); 10710 m_aCurrentPDFState.m_aMapMode = rMapMode; 10711 } 10712 10713 void PDFWriterImpl::setClipRegion( const basegfx::B2DPolyPolygon& rRegion ) 10714 { 10715 basegfx::B2DPolyPolygon aRegion = getReferenceDevice()->LogicToPixel( rRegion, m_aGraphicsStack.front().m_aMapMode ); 10716 aRegion = getReferenceDevice()->PixelToLogic( aRegion, m_aMapMode ); 10717 m_aGraphicsStack.front().m_aClipRegion = aRegion; 10718 m_aGraphicsStack.front().m_bClipRegion = true; 10719 m_aGraphicsStack.front().m_nUpdateFlags |= GraphicsState::updateClipRegion; 10720 } 10721 10722 void PDFWriterImpl::moveClipRegion( sal_Int32 nX, sal_Int32 nY ) 10723 { 10724 if( m_aGraphicsStack.front().m_bClipRegion && m_aGraphicsStack.front().m_aClipRegion.count() ) 10725 { 10726 Point aPoint( lcl_convert( m_aGraphicsStack.front().m_aMapMode, 10727 m_aMapMode, 10728 getReferenceDevice(), 10729 Point( nX, nY ) ) ); 10730 aPoint -= lcl_convert( m_aGraphicsStack.front().m_aMapMode, 10731 m_aMapMode, 10732 getReferenceDevice(), 10733 Point() ); 10734 basegfx::B2DHomMatrix aMat; 10735 aMat.translate( aPoint.X(), aPoint.Y() ); 10736 m_aGraphicsStack.front().m_aClipRegion.transform( aMat ); 10737 m_aGraphicsStack.front().m_nUpdateFlags |= GraphicsState::updateClipRegion; 10738 } 10739 } 10740 10741 bool PDFWriterImpl::intersectClipRegion( const Rectangle& rRect ) 10742 { 10743 basegfx::B2DPolyPolygon aRect( basegfx::tools::createPolygonFromRect( 10744 basegfx::B2DRectangle( rRect.Left(), rRect.Top(), rRect.Right(), rRect.Bottom() ) ) ); 10745 return intersectClipRegion( aRect ); 10746 } 10747 10748 10749 bool PDFWriterImpl::intersectClipRegion( const basegfx::B2DPolyPolygon& rRegion ) 10750 { 10751 basegfx::B2DPolyPolygon aRegion( getReferenceDevice()->LogicToPixel( rRegion, m_aGraphicsStack.front().m_aMapMode ) ); 10752 aRegion = getReferenceDevice()->PixelToLogic( aRegion, m_aMapMode ); 10753 m_aGraphicsStack.front().m_nUpdateFlags |= GraphicsState::updateClipRegion; 10754 if( m_aGraphicsStack.front().m_bClipRegion ) 10755 { 10756 basegfx::B2DPolyPolygon aOld( basegfx::tools::prepareForPolygonOperation( m_aGraphicsStack.front().m_aClipRegion ) ); 10757 aRegion = basegfx::tools::prepareForPolygonOperation( aRegion ); 10758 m_aGraphicsStack.front().m_aClipRegion = basegfx::tools::solvePolygonOperationAnd( aOld, aRegion ); 10759 } 10760 else 10761 { 10762 m_aGraphicsStack.front().m_aClipRegion = aRegion; 10763 m_aGraphicsStack.front().m_bClipRegion = true; 10764 } 10765 return true; 10766 } 10767 10768 void PDFWriterImpl::createNote( const Rectangle& rRect, const PDFNote& rNote, sal_Int32 nPageNr ) 10769 { 10770 if( nPageNr < 0 ) 10771 nPageNr = m_nCurrentPage; 10772 10773 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 10774 return; 10775 10776 m_aNotes.push_back( PDFNoteEntry() ); 10777 m_aNotes.back().m_nObject = createObject(); 10778 m_aNotes.back().m_aContents = rNote; 10779 m_aNotes.back().m_aRect = rRect; 10780 // convert to default user space now, since the mapmode may change 10781 m_aPages[nPageNr].convertRect( m_aNotes.back().m_aRect ); 10782 10783 // insert note to page's annotation list 10784 m_aPages[ nPageNr ].m_aAnnotations.push_back( m_aNotes.back().m_nObject ); 10785 } 10786 10787 sal_Int32 PDFWriterImpl::createLink( const Rectangle& rRect, sal_Int32 nPageNr ) 10788 { 10789 if( nPageNr < 0 ) 10790 nPageNr = m_nCurrentPage; 10791 10792 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 10793 return -1; 10794 10795 sal_Int32 nRet = m_aLinks.size(); 10796 10797 m_aLinks.push_back( PDFLink() ); 10798 m_aLinks.back().m_nObject = createObject(); 10799 m_aLinks.back().m_nPage = nPageNr; 10800 m_aLinks.back().m_aRect = rRect; 10801 // convert to default user space now, since the mapmode may change 10802 m_aPages[nPageNr].convertRect( m_aLinks.back().m_aRect ); 10803 10804 // insert link to page's annotation list 10805 m_aPages[ nPageNr ].m_aAnnotations.push_back( m_aLinks.back().m_nObject ); 10806 10807 return nRet; 10808 } 10809 10810 //--->i56629 10811 sal_Int32 PDFWriterImpl::createNamedDest( const rtl::OUString& sDestName, const Rectangle& rRect, sal_Int32 nPageNr, PDFWriter::DestAreaType eType ) 10812 { 10813 if( nPageNr < 0 ) 10814 nPageNr = m_nCurrentPage; 10815 10816 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 10817 return -1; 10818 10819 sal_Int32 nRet = m_aNamedDests.size(); 10820 10821 m_aNamedDests.push_back( PDFNamedDest() ); 10822 m_aNamedDests.back().m_aDestName = sDestName; 10823 m_aNamedDests.back().m_nPage = nPageNr; 10824 m_aNamedDests.back().m_eType = eType; 10825 m_aNamedDests.back().m_aRect = rRect; 10826 // convert to default user space now, since the mapmode may change 10827 m_aPages[nPageNr].convertRect( m_aNamedDests.back().m_aRect ); 10828 10829 return nRet; 10830 } 10831 //<---i56629 10832 10833 sal_Int32 PDFWriterImpl::createDest( const Rectangle& rRect, sal_Int32 nPageNr, PDFWriter::DestAreaType eType ) 10834 { 10835 if( nPageNr < 0 ) 10836 nPageNr = m_nCurrentPage; 10837 10838 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 10839 return -1; 10840 10841 sal_Int32 nRet = m_aDests.size(); 10842 10843 m_aDests.push_back( PDFDest() ); 10844 m_aDests.back().m_nPage = nPageNr; 10845 m_aDests.back().m_eType = eType; 10846 m_aDests.back().m_aRect = rRect; 10847 // convert to default user space now, since the mapmode may change 10848 m_aPages[nPageNr].convertRect( m_aDests.back().m_aRect ); 10849 10850 return nRet; 10851 } 10852 10853 sal_Int32 PDFWriterImpl::registerDestReference( sal_Int32 nDestId, const Rectangle& rRect, sal_Int32 nPageNr, PDFWriter::DestAreaType eType ) 10854 { 10855 return m_aDestinationIdTranslation[ nDestId ] = createDest( rRect, nPageNr, eType ); 10856 } 10857 10858 sal_Int32 PDFWriterImpl::setLinkDest( sal_Int32 nLinkId, sal_Int32 nDestId ) 10859 { 10860 if( nLinkId < 0 || nLinkId >= (sal_Int32)m_aLinks.size() ) 10861 return -1; 10862 if( nDestId < 0 || nDestId >= (sal_Int32)m_aDests.size() ) 10863 return -2; 10864 10865 m_aLinks[ nLinkId ].m_nDest = nDestId; 10866 10867 return 0; 10868 } 10869 10870 sal_Int32 PDFWriterImpl::setLinkURL( sal_Int32 nLinkId, const OUString& rURL ) 10871 { 10872 if( nLinkId < 0 || nLinkId >= (sal_Int32)m_aLinks.size() ) 10873 return -1; 10874 10875 m_aLinks[ nLinkId ].m_nDest = -1; 10876 10877 using namespace ::com::sun::star; 10878 10879 if (!m_xTrans.is()) 10880 { 10881 uno::Reference< lang::XMultiServiceFactory > xFact( comphelper::getProcessServiceFactory() ); 10882 if( xFact.is() ) 10883 { 10884 m_xTrans = uno::Reference < util::XURLTransformer >( 10885 xFact->createInstance( OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.util.URLTransformer" ) ) ), uno::UNO_QUERY ); 10886 } 10887 } 10888 10889 util::URL aURL; 10890 aURL.Complete = rURL; 10891 10892 if (m_xTrans.is()) 10893 m_xTrans->parseStrict( aURL ); 10894 10895 m_aLinks[ nLinkId ].m_aURL = aURL.Complete; 10896 10897 return 0; 10898 } 10899 10900 void PDFWriterImpl::setLinkPropertyId( sal_Int32 nLinkId, sal_Int32 nPropertyId ) 10901 { 10902 m_aLinkPropertyMap[ nPropertyId ] = nLinkId; 10903 } 10904 10905 sal_Int32 PDFWriterImpl::createOutlineItem( sal_Int32 nParent, const OUString& rText, sal_Int32 nDestID ) 10906 { 10907 // create new item 10908 sal_Int32 nNewItem = m_aOutline.size(); 10909 m_aOutline.push_back( PDFOutlineEntry() ); 10910 10911 // set item attributes 10912 setOutlineItemParent( nNewItem, nParent ); 10913 setOutlineItemText( nNewItem, rText ); 10914 setOutlineItemDest( nNewItem, nDestID ); 10915 10916 return nNewItem; 10917 } 10918 10919 sal_Int32 PDFWriterImpl::setOutlineItemParent( sal_Int32 nItem, sal_Int32 nNewParent ) 10920 { 10921 if( nItem < 1 || nItem >= (sal_Int32)m_aOutline.size() ) 10922 return -1; 10923 10924 int nRet = 0; 10925 10926 if( nNewParent < 0 || nNewParent >= (sal_Int32)m_aOutline.size() || nNewParent == nItem ) 10927 { 10928 nNewParent = 0; 10929 nRet = -2; 10930 } 10931 // remove item from previous parent 10932 sal_Int32 nParentID = m_aOutline[ nItem ].m_nParentID; 10933 if( nParentID >= 0 && nParentID < (sal_Int32)m_aOutline.size() ) 10934 { 10935 PDFOutlineEntry& rParent = m_aOutline[ nParentID ]; 10936 10937 for( std::vector<sal_Int32>::iterator it = rParent.m_aChildren.begin(); 10938 it != rParent.m_aChildren.end(); ++it ) 10939 { 10940 if( *it == nItem ) 10941 { 10942 rParent.m_aChildren.erase( it ); 10943 break; 10944 } 10945 } 10946 } 10947 10948 // insert item to new parent's list of children 10949 m_aOutline[ nNewParent ].m_aChildren.push_back( nItem ); 10950 10951 return nRet; 10952 } 10953 10954 sal_Int32 PDFWriterImpl::setOutlineItemText( sal_Int32 nItem, const OUString& rText ) 10955 { 10956 if( nItem < 1 || nItem >= (sal_Int32)m_aOutline.size() ) 10957 return -1; 10958 10959 m_aOutline[ nItem ].m_aTitle = psp::WhitespaceToSpace( rText ); 10960 return 0; 10961 } 10962 10963 sal_Int32 PDFWriterImpl::setOutlineItemDest( sal_Int32 nItem, sal_Int32 nDestID ) 10964 { 10965 if( nItem < 1 || nItem >= (sal_Int32)m_aOutline.size() ) // item does not exist 10966 return -1; 10967 if( nDestID < 0 || nDestID >= (sal_Int32)m_aDests.size() ) // dest does not exist 10968 return -2; 10969 m_aOutline[nItem].m_nDestID = nDestID; 10970 return 0; 10971 } 10972 10973 const sal_Char* PDFWriterImpl::getStructureTag( PDFWriter::StructElement eType ) 10974 { 10975 static std::map< PDFWriter::StructElement, const char* > aTagStrings; 10976 if( aTagStrings.empty() ) 10977 { 10978 aTagStrings[ PDFWriter::NonStructElement] = "NonStruct"; 10979 aTagStrings[ PDFWriter::Document ] = "Document"; 10980 aTagStrings[ PDFWriter::Part ] = "Part"; 10981 aTagStrings[ PDFWriter::Article ] = "Art"; 10982 aTagStrings[ PDFWriter::Section ] = "Sect"; 10983 aTagStrings[ PDFWriter::Division ] = "Div"; 10984 aTagStrings[ PDFWriter::BlockQuote ] = "BlockQuote"; 10985 aTagStrings[ PDFWriter::Caption ] = "Caption"; 10986 aTagStrings[ PDFWriter::TOC ] = "TOC"; 10987 aTagStrings[ PDFWriter::TOCI ] = "TOCI"; 10988 aTagStrings[ PDFWriter::Index ] = "Index"; 10989 aTagStrings[ PDFWriter::Paragraph ] = "P"; 10990 aTagStrings[ PDFWriter::Heading ] = "H"; 10991 aTagStrings[ PDFWriter::H1 ] = "H1"; 10992 aTagStrings[ PDFWriter::H2 ] = "H2"; 10993 aTagStrings[ PDFWriter::H3 ] = "H3"; 10994 aTagStrings[ PDFWriter::H4 ] = "H4"; 10995 aTagStrings[ PDFWriter::H5 ] = "H5"; 10996 aTagStrings[ PDFWriter::H6 ] = "H6"; 10997 aTagStrings[ PDFWriter::List ] = "L"; 10998 aTagStrings[ PDFWriter::ListItem ] = "LI"; 10999 aTagStrings[ PDFWriter::LILabel ] = "Lbl"; 11000 aTagStrings[ PDFWriter::LIBody ] = "LBody"; 11001 aTagStrings[ PDFWriter::Table ] = "Table"; 11002 aTagStrings[ PDFWriter::TableRow ] = "TR"; 11003 aTagStrings[ PDFWriter::TableHeader ] = "TH"; 11004 aTagStrings[ PDFWriter::TableData ] = "TD"; 11005 aTagStrings[ PDFWriter::Span ] = "Span"; 11006 aTagStrings[ PDFWriter::Quote ] = "Quote"; 11007 aTagStrings[ PDFWriter::Note ] = "Note"; 11008 aTagStrings[ PDFWriter::Reference ] = "Reference"; 11009 aTagStrings[ PDFWriter::BibEntry ] = "BibEntry"; 11010 aTagStrings[ PDFWriter::Code ] = "Code"; 11011 aTagStrings[ PDFWriter::Link ] = "Link"; 11012 aTagStrings[ PDFWriter::Figure ] = "Figure"; 11013 aTagStrings[ PDFWriter::Formula ] = "Formula"; 11014 aTagStrings[ PDFWriter::Form ] = "Form"; 11015 } 11016 11017 std::map< PDFWriter::StructElement, const char* >::const_iterator it = aTagStrings.find( eType ); 11018 11019 return it != aTagStrings.end() ? it->second : "Div"; 11020 } 11021 11022 void PDFWriterImpl::beginStructureElementMCSeq() 11023 { 11024 if( m_bEmitStructure && 11025 m_nCurrentStructElement > 0 && // StructTreeRoot 11026 ! m_aStructure[ m_nCurrentStructElement ].m_bOpenMCSeq // already opened sequence 11027 ) 11028 { 11029 PDFStructureElement& rEle = m_aStructure[ m_nCurrentStructElement ]; 11030 OStringBuffer aLine( 128 ); 11031 sal_Int32 nMCID = m_aPages[ m_nCurrentPage ].m_aMCIDParents.size(); 11032 aLine.append( "/" ); 11033 if( rEle.m_aAlias.getLength() > 0 ) 11034 aLine.append( rEle.m_aAlias ); 11035 else 11036 aLine.append( getStructureTag( rEle.m_eType ) ); 11037 aLine.append( "<</MCID " ); 11038 aLine.append( nMCID ); 11039 aLine.append( ">>BDC\n" ); 11040 writeBuffer( aLine.getStr(), aLine.getLength() ); 11041 11042 // update the element's content list 11043 #if OSL_DEBUG_LEVEL > 1 11044 fprintf( stderr, "beginning marked content id %" SAL_PRIdINT32 " on page object %" SAL_PRIdINT32 ", structure first page = %" SAL_PRIdINT32 "\n", 11045 nMCID, 11046 m_aPages[ m_nCurrentPage ].m_nPageObject, 11047 rEle.m_nFirstPageObject ); 11048 #endif 11049 rEle.m_aKids.push_back( PDFStructureElementKid( nMCID, m_aPages[m_nCurrentPage].m_nPageObject ) ); 11050 // update the page's mcid parent list 11051 m_aPages[ m_nCurrentPage ].m_aMCIDParents.push_back( rEle.m_nObject ); 11052 // mark element MC sequence as open 11053 rEle.m_bOpenMCSeq = true; 11054 } 11055 // handle artifacts 11056 else if( ! m_bEmitStructure && m_aContext.Tagged && 11057 m_nCurrentStructElement > 0 && 11058 m_aStructure[ m_nCurrentStructElement ].m_eType == PDFWriter::NonStructElement && 11059 ! m_aStructure[ m_nCurrentStructElement ].m_bOpenMCSeq // already opened sequence 11060 ) 11061 { 11062 OStringBuffer aLine( 128 ); 11063 aLine.append( "/Artifact BMC\n" ); 11064 writeBuffer( aLine.getStr(), aLine.getLength() ); 11065 // mark element MC sequence as open 11066 m_aStructure[ m_nCurrentStructElement ].m_bOpenMCSeq = true; 11067 } 11068 } 11069 11070 void PDFWriterImpl::endStructureElementMCSeq() 11071 { 11072 if( m_nCurrentStructElement > 0 && // StructTreeRoot 11073 ( m_bEmitStructure || m_aStructure[ m_nCurrentStructElement ].m_eType == PDFWriter::NonStructElement ) && 11074 m_aStructure[ m_nCurrentStructElement ].m_bOpenMCSeq // must have an opened MC sequence 11075 ) 11076 { 11077 writeBuffer( "EMC\n", 4 ); 11078 m_aStructure[ m_nCurrentStructElement ].m_bOpenMCSeq = false; 11079 } 11080 } 11081 11082 bool PDFWriterImpl::checkEmitStructure() 11083 { 11084 bool bEmit = false; 11085 if( m_aContext.Tagged ) 11086 { 11087 bEmit = true; 11088 sal_Int32 nEle = m_nCurrentStructElement; 11089 while( nEle > 0 && nEle < sal_Int32(m_aStructure.size()) ) 11090 { 11091 if( m_aStructure[ nEle ].m_eType == PDFWriter::NonStructElement ) 11092 { 11093 bEmit = false; 11094 break; 11095 } 11096 nEle = m_aStructure[ nEle ].m_nParentElement; 11097 } 11098 } 11099 return bEmit; 11100 } 11101 11102 sal_Int32 PDFWriterImpl::beginStructureElement( PDFWriter::StructElement eType, const rtl::OUString& rAlias ) 11103 { 11104 if( m_nCurrentPage < 0 ) 11105 return -1; 11106 11107 if( ! m_aContext.Tagged ) 11108 return -1; 11109 11110 // close eventual current MC sequence 11111 endStructureElementMCSeq(); 11112 11113 if( m_nCurrentStructElement == 0 && 11114 eType != PDFWriter::Document && eType != PDFWriter::NonStructElement ) 11115 { 11116 // struct tree root hit, but not beginning document 11117 // this might happen with setCurrentStructureElement 11118 // silently insert structure into document again if one properly exists 11119 if( ! m_aStructure[ 0 ].m_aChildren.empty() ) 11120 { 11121 PDFWriter::StructElement childType = PDFWriter::NonStructElement; 11122 sal_Int32 nNewCurElement = 0; 11123 const std::list< sal_Int32 >& rRootChildren = m_aStructure[0].m_aChildren; 11124 for( std::list< sal_Int32 >::const_iterator it = rRootChildren.begin(); 11125 childType != PDFWriter::Document && it != rRootChildren.end(); ++it ) 11126 { 11127 nNewCurElement = *it; 11128 childType = m_aStructure[ nNewCurElement ].m_eType; 11129 } 11130 if( childType == PDFWriter::Document ) 11131 { 11132 m_nCurrentStructElement = nNewCurElement; 11133 DBG_ASSERT( 0, "Structure element inserted to StructTreeRoot that is not a document" ); 11134 } 11135 else { 11136 DBG_ERROR( "document structure in disorder !" ); 11137 } 11138 } 11139 else { 11140 DBG_ERROR( "PDF document structure MUST be contained in a Document element" ); 11141 } 11142 } 11143 11144 sal_Int32 nNewId = sal_Int32(m_aStructure.size()); 11145 m_aStructure.push_back( PDFStructureElement() ); 11146 PDFStructureElement& rEle = m_aStructure.back(); 11147 rEle.m_eType = eType; 11148 rEle.m_nOwnElement = nNewId; 11149 rEle.m_nParentElement = m_nCurrentStructElement; 11150 rEle.m_nFirstPageObject = m_aPages[ m_nCurrentPage ].m_nPageObject; 11151 m_aStructure[ m_nCurrentStructElement ].m_aChildren.push_back( nNewId ); 11152 m_nCurrentStructElement = nNewId; 11153 11154 // handle alias names 11155 if( rAlias.getLength() && eType != PDFWriter::NonStructElement ) 11156 { 11157 OStringBuffer aNameBuf( rAlias.getLength() ); 11158 appendName( rAlias, aNameBuf ); 11159 OString aAliasName( aNameBuf.makeStringAndClear() ); 11160 rEle.m_aAlias = aAliasName; 11161 m_aRoleMap[ aAliasName ] = getStructureTag( eType ); 11162 } 11163 11164 #if OSL_DEBUG_LEVEL > 1 11165 OStringBuffer aLine( "beginStructureElement " ); 11166 aLine.append( m_nCurrentStructElement ); 11167 aLine.append( ": " ); 11168 aLine.append( getStructureTag( eType ) ); 11169 if( rEle.m_aAlias.getLength() ) 11170 { 11171 aLine.append( " aliased as \"" ); 11172 aLine.append( rEle.m_aAlias ); 11173 aLine.append( '\"' ); 11174 } 11175 emitComment( aLine.getStr() ); 11176 #endif 11177 11178 // check whether to emit structure henceforth 11179 m_bEmitStructure = checkEmitStructure(); 11180 11181 if( m_bEmitStructure ) // don't create nonexistant objects 11182 { 11183 rEle.m_nObject = createObject(); 11184 // update parent's kids list 11185 m_aStructure[ rEle.m_nParentElement ].m_aKids.push_back( rEle.m_nObject ); 11186 } 11187 return nNewId; 11188 } 11189 11190 void PDFWriterImpl::endStructureElement() 11191 { 11192 if( m_nCurrentPage < 0 ) 11193 return; 11194 11195 if( ! m_aContext.Tagged ) 11196 return; 11197 11198 if( m_nCurrentStructElement == 0 ) 11199 { 11200 // hit the struct tree root, that means there is an endStructureElement 11201 // without corresponding beginStructureElement 11202 return; 11203 } 11204 11205 // end the marked content sequence 11206 endStructureElementMCSeq(); 11207 11208 #if OSL_DEBUG_LEVEL > 1 11209 OStringBuffer aLine( "endStructureElement " ); 11210 aLine.append( m_nCurrentStructElement ); 11211 aLine.append( ": " ); 11212 aLine.append( getStructureTag( m_aStructure[ m_nCurrentStructElement ].m_eType ) ); 11213 if( m_aStructure[ m_nCurrentStructElement ].m_aAlias.getLength() ) 11214 { 11215 aLine.append( " aliased as \"" ); 11216 aLine.append( m_aStructure[ m_nCurrentStructElement ].m_aAlias ); 11217 aLine.append( '\"' ); 11218 } 11219 #endif 11220 11221 // "end" the structure element, the parent becomes current element 11222 m_nCurrentStructElement = m_aStructure[ m_nCurrentStructElement ].m_nParentElement; 11223 11224 // check whether to emit structure henceforth 11225 m_bEmitStructure = checkEmitStructure(); 11226 11227 #if OSL_DEBUG_LEVEL > 1 11228 if( m_bEmitStructure ) 11229 emitComment( aLine.getStr() ); 11230 #endif 11231 } 11232 11233 //---> i94258 11234 /* 11235 * This function adds an internal structure list container to overcome the 8191 elements array limitation 11236 * in kids element emission. 11237 * Recursive function 11238 * 11239 */ 11240 void PDFWriterImpl::addInternalStructureContainer( PDFStructureElement& rEle ) 11241 { 11242 if( rEle.m_eType == PDFWriter::NonStructElement && 11243 rEle.m_nOwnElement != rEle.m_nParentElement ) 11244 return; 11245 11246 for( std::list< sal_Int32 >::const_iterator it = rEle.m_aChildren.begin(); it != rEle.m_aChildren.end(); ++it ) 11247 { 11248 if( *it > 0 && *it < sal_Int32(m_aStructure.size()) ) 11249 { 11250 PDFStructureElement& rChild = m_aStructure[ *it ]; 11251 if( rChild.m_eType != PDFWriter::NonStructElement ) 11252 { 11253 //triggered when a child of the rEle element is found 11254 if( rChild.m_nParentElement == rEle.m_nOwnElement ) 11255 addInternalStructureContainer( rChild );//examine the child 11256 else 11257 { 11258 DBG_ERROR( "PDFWriterImpl::addInternalStructureContainer: invalid child structure element" ); 11259 #if OSL_DEBUG_LEVEL > 1 11260 fprintf( stderr, "PDFWriterImpl::addInternalStructureContainer: invalid child structure elemnt with id %" SAL_PRIdINT32 "\n", *it ); 11261 #endif 11262 } 11263 } 11264 } 11265 else 11266 { 11267 DBG_ERROR( "PDFWriterImpl::emitStructure: invalid child structure id" ); 11268 #if OSL_DEBUG_LEVEL > 1 11269 fprintf( stderr, "PDFWriterImpl::addInternalStructureContainer: invalid child structure id %" SAL_PRIdINT32 "\n", *it ); 11270 #endif 11271 } 11272 } 11273 11274 if( rEle.m_nOwnElement != rEle.m_nParentElement ) 11275 { 11276 if( !rEle.m_aKids.empty() ) 11277 { 11278 if( rEle.m_aKids.size() > ncMaxPDFArraySize ) { 11279 //then we need to add the containers for the kids elements 11280 // a list to be used for the new kid element 11281 std::list< PDFStructureElementKid > aNewKids; 11282 std::list< sal_Int32 > aNewChildren; 11283 11284 // add Div in RoleMap, in case no one else did (TODO: is it needed? Is it dangerous?) 11285 OStringBuffer aNameBuf( "Div" ); 11286 OString aAliasName( aNameBuf.makeStringAndClear() ); 11287 m_aRoleMap[ aAliasName ] = getStructureTag( PDFWriter::Division ); 11288 11289 while( rEle.m_aKids.size() > ncMaxPDFArraySize ) 11290 { 11291 sal_Int32 nCurrentStructElement = rEle.m_nOwnElement; 11292 sal_Int32 nNewId = sal_Int32(m_aStructure.size()); 11293 m_aStructure.push_back( PDFStructureElement() ); 11294 PDFStructureElement& rEleNew = m_aStructure.back(); 11295 rEleNew.m_aAlias = aAliasName; 11296 rEleNew.m_eType = PDFWriter::Division; // a new Div type container 11297 rEleNew.m_nOwnElement = nNewId; 11298 rEleNew.m_nParentElement = nCurrentStructElement; 11299 //inherit the same page as the first child to be reparented 11300 rEleNew.m_nFirstPageObject = m_aStructure[ rEle.m_aChildren.front() ].m_nFirstPageObject; 11301 rEleNew.m_nObject = createObject();//assign a PDF object number 11302 //add the object to the kid list of the parent 11303 aNewKids.push_back( PDFStructureElementKid( rEleNew.m_nObject ) ); 11304 aNewChildren.push_back( nNewId ); 11305 11306 std::list< sal_Int32 >::iterator aChildEndIt( rEle.m_aChildren.begin() ); 11307 std::list< PDFStructureElementKid >::iterator aKidEndIt( rEle.m_aKids.begin() ); 11308 advance( aChildEndIt, ncMaxPDFArraySize ); 11309 advance( aKidEndIt, ncMaxPDFArraySize ); 11310 11311 rEleNew.m_aKids.splice( rEleNew.m_aKids.begin(), 11312 rEle.m_aKids, 11313 rEle.m_aKids.begin(), 11314 aKidEndIt ); 11315 rEleNew.m_aChildren.splice( rEleNew.m_aChildren.begin(), 11316 rEle.m_aChildren, 11317 rEle.m_aChildren.begin(), 11318 aChildEndIt ); 11319 // set the kid's new parent 11320 for( std::list< sal_Int32 >::const_iterator it = rEleNew.m_aChildren.begin(); 11321 it != rEleNew.m_aChildren.end(); ++it ) 11322 { 11323 m_aStructure[ *it ].m_nParentElement = nNewId; 11324 } 11325 } 11326 //finally add the new kids resulting from the container added 11327 rEle.m_aKids.insert( rEle.m_aKids.begin(), aNewKids.begin(), aNewKids.end() ); 11328 rEle.m_aChildren.insert( rEle.m_aChildren.begin(), aNewChildren.begin(), aNewChildren.end() ); 11329 } 11330 } 11331 } 11332 } 11333 //<--- i94258 11334 11335 bool PDFWriterImpl::setCurrentStructureElement( sal_Int32 nEle ) 11336 { 11337 bool bSuccess = false; 11338 11339 if( m_aContext.Tagged && nEle >= 0 && nEle < sal_Int32(m_aStructure.size()) ) 11340 { 11341 // end eventual previous marked content sequence 11342 endStructureElementMCSeq(); 11343 11344 m_nCurrentStructElement = nEle; 11345 m_bEmitStructure = checkEmitStructure(); 11346 #if OSL_DEBUG_LEVEL > 1 11347 OStringBuffer aLine( "setCurrentStructureElement " ); 11348 aLine.append( m_nCurrentStructElement ); 11349 aLine.append( ": " ); 11350 aLine.append( getStructureTag( m_aStructure[ m_nCurrentStructElement ].m_eType ) ); 11351 if( m_aStructure[ m_nCurrentStructElement ].m_aAlias.getLength() ) 11352 { 11353 aLine.append( " aliased as \"" ); 11354 aLine.append( m_aStructure[ m_nCurrentStructElement ].m_aAlias ); 11355 aLine.append( '\"' ); 11356 } 11357 if( ! m_bEmitStructure ) 11358 aLine.append( " (inside NonStruct)" ); 11359 emitComment( aLine.getStr() ); 11360 #endif 11361 bSuccess = true; 11362 } 11363 11364 return bSuccess; 11365 } 11366 11367 sal_Int32 PDFWriterImpl::getCurrentStructureElement() 11368 { 11369 return m_nCurrentStructElement; 11370 } 11371 11372 bool PDFWriterImpl::setStructureAttribute( enum PDFWriter::StructAttribute eAttr, enum PDFWriter::StructAttributeValue eVal ) 11373 { 11374 if( !m_aContext.Tagged ) 11375 return false; 11376 11377 bool bInsert = false; 11378 if( m_nCurrentStructElement > 0 && m_bEmitStructure ) 11379 { 11380 PDFWriter::StructElement eType = m_aStructure[ m_nCurrentStructElement ].m_eType; 11381 switch( eAttr ) 11382 { 11383 case PDFWriter::Placement: 11384 if( eVal == PDFWriter::Block || 11385 eVal == PDFWriter::Inline || 11386 eVal == PDFWriter::Before || 11387 eVal == PDFWriter::Start || 11388 eVal == PDFWriter::End ) 11389 bInsert = true; 11390 break; 11391 case PDFWriter::WritingMode: 11392 if( eVal == PDFWriter::LrTb || 11393 eVal == PDFWriter::RlTb || 11394 eVal == PDFWriter::TbRl ) 11395 { 11396 bInsert = true; 11397 } 11398 break; 11399 case PDFWriter::TextAlign: 11400 if( eVal == PDFWriter::Start || 11401 eVal == PDFWriter::Center || 11402 eVal == PDFWriter::End || 11403 eVal == PDFWriter::Justify ) 11404 { 11405 if( eType == PDFWriter::Paragraph || 11406 eType == PDFWriter::Heading || 11407 eType == PDFWriter::H1 || 11408 eType == PDFWriter::H2 || 11409 eType == PDFWriter::H3 || 11410 eType == PDFWriter::H4 || 11411 eType == PDFWriter::H5 || 11412 eType == PDFWriter::H6 || 11413 eType == PDFWriter::List || 11414 eType == PDFWriter::ListItem || 11415 eType == PDFWriter::LILabel || 11416 eType == PDFWriter::LIBody || 11417 eType == PDFWriter::Table || 11418 eType == PDFWriter::TableRow || 11419 eType == PDFWriter::TableHeader || 11420 eType == PDFWriter::TableData ) 11421 { 11422 bInsert = true; 11423 } 11424 } 11425 break; 11426 case PDFWriter::Width: 11427 case PDFWriter::Height: 11428 if( eVal == PDFWriter::Auto ) 11429 { 11430 if( eType == PDFWriter::Figure || 11431 eType == PDFWriter::Formula || 11432 eType == PDFWriter::Form || 11433 eType == PDFWriter::Table || 11434 eType == PDFWriter::TableHeader || 11435 eType == PDFWriter::TableData ) 11436 { 11437 bInsert = true; 11438 } 11439 } 11440 break; 11441 case PDFWriter::BlockAlign: 11442 if( eVal == PDFWriter::Before || 11443 eVal == PDFWriter::Middle || 11444 eVal == PDFWriter::After || 11445 eVal == PDFWriter::Justify ) 11446 { 11447 if( eType == PDFWriter::TableHeader || 11448 eType == PDFWriter::TableData ) 11449 { 11450 bInsert = true; 11451 } 11452 } 11453 break; 11454 case PDFWriter::InlineAlign: 11455 if( eVal == PDFWriter::Start || 11456 eVal == PDFWriter::Center || 11457 eVal == PDFWriter::End ) 11458 { 11459 if( eType == PDFWriter::TableHeader || 11460 eType == PDFWriter::TableData ) 11461 { 11462 bInsert = true; 11463 } 11464 } 11465 break; 11466 case PDFWriter::LineHeight: 11467 if( eVal == PDFWriter::Normal || 11468 eVal == PDFWriter::Auto ) 11469 { 11470 // only for ILSE and BLSE 11471 if( eType == PDFWriter::Paragraph || 11472 eType == PDFWriter::Heading || 11473 eType == PDFWriter::H1 || 11474 eType == PDFWriter::H2 || 11475 eType == PDFWriter::H3 || 11476 eType == PDFWriter::H4 || 11477 eType == PDFWriter::H5 || 11478 eType == PDFWriter::H6 || 11479 eType == PDFWriter::List || 11480 eType == PDFWriter::ListItem || 11481 eType == PDFWriter::LILabel || 11482 eType == PDFWriter::LIBody || 11483 eType == PDFWriter::Table || 11484 eType == PDFWriter::TableRow || 11485 eType == PDFWriter::TableHeader || 11486 eType == PDFWriter::TableData || 11487 eType == PDFWriter::Span || 11488 eType == PDFWriter::Quote || 11489 eType == PDFWriter::Note || 11490 eType == PDFWriter::Reference || 11491 eType == PDFWriter::BibEntry || 11492 eType == PDFWriter::Code || 11493 eType == PDFWriter::Link ) 11494 { 11495 bInsert = true; 11496 } 11497 } 11498 break; 11499 case PDFWriter::TextDecorationType: 11500 if( eVal == PDFWriter::NONE || 11501 eVal == PDFWriter::Underline || 11502 eVal == PDFWriter::Overline || 11503 eVal == PDFWriter::LineThrough ) 11504 { 11505 // only for ILSE and BLSE 11506 if( eType == PDFWriter::Paragraph || 11507 eType == PDFWriter::Heading || 11508 eType == PDFWriter::H1 || 11509 eType == PDFWriter::H2 || 11510 eType == PDFWriter::H3 || 11511 eType == PDFWriter::H4 || 11512 eType == PDFWriter::H5 || 11513 eType == PDFWriter::H6 || 11514 eType == PDFWriter::List || 11515 eType == PDFWriter::ListItem || 11516 eType == PDFWriter::LILabel || 11517 eType == PDFWriter::LIBody || 11518 eType == PDFWriter::Table || 11519 eType == PDFWriter::TableRow || 11520 eType == PDFWriter::TableHeader || 11521 eType == PDFWriter::TableData || 11522 eType == PDFWriter::Span || 11523 eType == PDFWriter::Quote || 11524 eType == PDFWriter::Note || 11525 eType == PDFWriter::Reference || 11526 eType == PDFWriter::BibEntry || 11527 eType == PDFWriter::Code || 11528 eType == PDFWriter::Link ) 11529 { 11530 bInsert = true; 11531 } 11532 } 11533 break; 11534 case PDFWriter::ListNumbering: 11535 if( eVal == PDFWriter::NONE || 11536 eVal == PDFWriter::Disc || 11537 eVal == PDFWriter::Circle || 11538 eVal == PDFWriter::Square || 11539 eVal == PDFWriter::Decimal || 11540 eVal == PDFWriter::UpperRoman || 11541 eVal == PDFWriter::LowerRoman || 11542 eVal == PDFWriter::UpperAlpha || 11543 eVal == PDFWriter::LowerAlpha ) 11544 { 11545 if( eType == PDFWriter::List ) 11546 bInsert = true; 11547 } 11548 break; 11549 default: break; 11550 } 11551 } 11552 11553 if( bInsert ) 11554 m_aStructure[ m_nCurrentStructElement ].m_aAttributes[ eAttr ] = PDFStructureAttribute( eVal ); 11555 #if OSL_DEBUG_LEVEL > 1 11556 else if( m_nCurrentStructElement > 0 && m_bEmitStructure ) 11557 fprintf( stderr, "rejecting setStructureAttribute( %s, %s ) on %s (%s) element\n", 11558 getAttributeTag( eAttr ), 11559 getAttributeValueTag( eVal ), 11560 getStructureTag( m_aStructure[ m_nCurrentStructElement ].m_eType ), 11561 m_aStructure[ m_nCurrentStructElement ].m_aAlias.getStr() 11562 ); 11563 #endif 11564 11565 return bInsert; 11566 } 11567 11568 bool PDFWriterImpl::setStructureAttributeNumerical( enum PDFWriter::StructAttribute eAttr, sal_Int32 nValue ) 11569 { 11570 if( ! m_aContext.Tagged ) 11571 return false; 11572 11573 bool bInsert = false; 11574 if( m_nCurrentStructElement > 0 && m_bEmitStructure ) 11575 { 11576 if( eAttr == PDFWriter::Language ) 11577 { 11578 m_aStructure[ m_nCurrentStructElement ].m_aLocale = MsLangId::convertLanguageToLocale( (LanguageType)nValue ); 11579 return true; 11580 } 11581 11582 PDFWriter::StructElement eType = m_aStructure[ m_nCurrentStructElement ].m_eType; 11583 switch( eAttr ) 11584 { 11585 case PDFWriter::SpaceBefore: 11586 case PDFWriter::SpaceAfter: 11587 case PDFWriter::StartIndent: 11588 case PDFWriter::EndIndent: 11589 // just for BLSE 11590 if( eType == PDFWriter::Paragraph || 11591 eType == PDFWriter::Heading || 11592 eType == PDFWriter::H1 || 11593 eType == PDFWriter::H2 || 11594 eType == PDFWriter::H3 || 11595 eType == PDFWriter::H4 || 11596 eType == PDFWriter::H5 || 11597 eType == PDFWriter::H6 || 11598 eType == PDFWriter::List || 11599 eType == PDFWriter::ListItem || 11600 eType == PDFWriter::LILabel || 11601 eType == PDFWriter::LIBody || 11602 eType == PDFWriter::Table || 11603 eType == PDFWriter::TableRow || 11604 eType == PDFWriter::TableHeader || 11605 eType == PDFWriter::TableData ) 11606 { 11607 bInsert = true; 11608 } 11609 break; 11610 case PDFWriter::TextIndent: 11611 // paragraph like BLSE and additional elements 11612 if( eType == PDFWriter::Paragraph || 11613 eType == PDFWriter::Heading || 11614 eType == PDFWriter::H1 || 11615 eType == PDFWriter::H2 || 11616 eType == PDFWriter::H3 || 11617 eType == PDFWriter::H4 || 11618 eType == PDFWriter::H5 || 11619 eType == PDFWriter::H6 || 11620 eType == PDFWriter::LILabel || 11621 eType == PDFWriter::LIBody || 11622 eType == PDFWriter::TableHeader || 11623 eType == PDFWriter::TableData ) 11624 { 11625 bInsert = true; 11626 } 11627 break; 11628 case PDFWriter::Width: 11629 case PDFWriter::Height: 11630 if( eType == PDFWriter::Figure || 11631 eType == PDFWriter::Formula || 11632 eType == PDFWriter::Form || 11633 eType == PDFWriter::Table || 11634 eType == PDFWriter::TableHeader || 11635 eType == PDFWriter::TableData ) 11636 { 11637 bInsert = true; 11638 } 11639 break; 11640 case PDFWriter::LineHeight: 11641 case PDFWriter::BaselineShift: 11642 // only for ILSE and BLSE 11643 if( eType == PDFWriter::Paragraph || 11644 eType == PDFWriter::Heading || 11645 eType == PDFWriter::H1 || 11646 eType == PDFWriter::H2 || 11647 eType == PDFWriter::H3 || 11648 eType == PDFWriter::H4 || 11649 eType == PDFWriter::H5 || 11650 eType == PDFWriter::H6 || 11651 eType == PDFWriter::List || 11652 eType == PDFWriter::ListItem || 11653 eType == PDFWriter::LILabel || 11654 eType == PDFWriter::LIBody || 11655 eType == PDFWriter::Table || 11656 eType == PDFWriter::TableRow || 11657 eType == PDFWriter::TableHeader || 11658 eType == PDFWriter::TableData || 11659 eType == PDFWriter::Span || 11660 eType == PDFWriter::Quote || 11661 eType == PDFWriter::Note || 11662 eType == PDFWriter::Reference || 11663 eType == PDFWriter::BibEntry || 11664 eType == PDFWriter::Code || 11665 eType == PDFWriter::Link ) 11666 { 11667 bInsert = true; 11668 } 11669 break; 11670 case PDFWriter::RowSpan: 11671 case PDFWriter::ColSpan: 11672 // only for table cells 11673 if( eType == PDFWriter::TableHeader || 11674 eType == PDFWriter::TableData ) 11675 { 11676 bInsert = true; 11677 } 11678 break; 11679 case PDFWriter::LinkAnnotation: 11680 if( eType == PDFWriter::Link ) 11681 bInsert = true; 11682 break; 11683 default: break; 11684 } 11685 } 11686 11687 if( bInsert ) 11688 m_aStructure[ m_nCurrentStructElement ].m_aAttributes[ eAttr ] = PDFStructureAttribute( nValue ); 11689 #if OSL_DEBUG_LEVEL > 1 11690 else if( m_nCurrentStructElement > 0 && m_bEmitStructure ) 11691 fprintf( stderr, "rejecting setStructureAttributeNumerical( %s, %d ) on %s (%s) element\n", 11692 getAttributeTag( eAttr ), 11693 (int)nValue, 11694 getStructureTag( m_aStructure[ m_nCurrentStructElement ].m_eType ), 11695 m_aStructure[ m_nCurrentStructElement ].m_aAlias.getStr() ); 11696 #endif 11697 11698 return bInsert; 11699 } 11700 11701 void PDFWriterImpl::setStructureBoundingBox( const Rectangle& rRect ) 11702 { 11703 sal_Int32 nPageNr = m_nCurrentPage; 11704 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() || !m_aContext.Tagged ) 11705 return; 11706 11707 11708 if( m_nCurrentStructElement > 0 && m_bEmitStructure ) 11709 { 11710 PDFWriter::StructElement eType = m_aStructure[ m_nCurrentStructElement ].m_eType; 11711 if( eType == PDFWriter::Figure || 11712 eType == PDFWriter::Formula || 11713 eType == PDFWriter::Form || 11714 eType == PDFWriter::Table ) 11715 { 11716 m_aStructure[ m_nCurrentStructElement ].m_aBBox = rRect; 11717 // convert to default user space now, since the mapmode may change 11718 m_aPages[nPageNr].convertRect( m_aStructure[ m_nCurrentStructElement ].m_aBBox ); 11719 } 11720 } 11721 } 11722 11723 void PDFWriterImpl::setActualText( const String& rText ) 11724 { 11725 if( m_aContext.Tagged && m_nCurrentStructElement > 0 && m_bEmitStructure ) 11726 { 11727 m_aStructure[ m_nCurrentStructElement ].m_aActualText = rText; 11728 } 11729 } 11730 11731 void PDFWriterImpl::setAlternateText( const String& rText ) 11732 { 11733 if( m_aContext.Tagged && m_nCurrentStructElement > 0 && m_bEmitStructure ) 11734 { 11735 m_aStructure[ m_nCurrentStructElement ].m_aAltText = rText; 11736 } 11737 } 11738 11739 void PDFWriterImpl::setAutoAdvanceTime( sal_uInt32 nSeconds, sal_Int32 nPageNr ) 11740 { 11741 if( nPageNr < 0 ) 11742 nPageNr = m_nCurrentPage; 11743 11744 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 11745 return; 11746 11747 m_aPages[ nPageNr ].m_nDuration = nSeconds; 11748 } 11749 11750 void PDFWriterImpl::setPageTransition( PDFWriter::PageTransition eType, sal_uInt32 nMilliSec, sal_Int32 nPageNr ) 11751 { 11752 if( nPageNr < 0 ) 11753 nPageNr = m_nCurrentPage; 11754 11755 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 11756 return; 11757 11758 m_aPages[ nPageNr ].m_eTransition = eType; 11759 m_aPages[ nPageNr ].m_nTransTime = nMilliSec; 11760 } 11761 11762 void PDFWriterImpl::ensureUniqueRadioOnValues() 11763 { 11764 // loop over radio groups 11765 for( std::map<sal_Int32,sal_Int32>::const_iterator group = m_aRadioGroupWidgets.begin(); 11766 group != m_aRadioGroupWidgets.end(); ++group ) 11767 { 11768 PDFWidget& rGroupWidget = m_aWidgets[ group->second ]; 11769 // check whether all kids have a unique OnValue 11770 std::hash_map< OUString, sal_Int32, OUStringHash > aOnValues; 11771 int nChildren = rGroupWidget.m_aKidsIndex.size(); 11772 bool bIsUnique = true; 11773 for( int nKid = 0; nKid < nChildren && bIsUnique; nKid++ ) 11774 { 11775 int nKidIndex = rGroupWidget.m_aKidsIndex[nKid]; 11776 const OUString& rVal = m_aWidgets[nKidIndex].m_aOnValue; 11777 #if OSL_DEBUG_LEVEL > 1 11778 fprintf( stderr, "OnValue: %s\n", OUStringToOString( rVal, RTL_TEXTENCODING_UTF8 ).getStr() ); 11779 #endif 11780 if( aOnValues.find( rVal ) == aOnValues.end() ) 11781 { 11782 aOnValues[ rVal ] = 1; 11783 } 11784 else 11785 { 11786 bIsUnique = false; 11787 } 11788 } 11789 if( ! bIsUnique ) 11790 { 11791 #if OSL_DEBUG_LEVEL > 1 11792 fprintf( stderr, "enforcing unique OnValues\n" ); 11793 #endif 11794 // make unique by using ascending OnValues 11795 for( int nKid = 0; nKid < nChildren; nKid++ ) 11796 { 11797 int nKidIndex = rGroupWidget.m_aKidsIndex[nKid]; 11798 PDFWidget& rKid = m_aWidgets[nKidIndex]; 11799 rKid.m_aOnValue = OUString::valueOf( sal_Int32(nKid+1) ); 11800 if( ! rKid.m_aValue.equalsAscii( "Off" ) ) 11801 rKid.m_aValue = rKid.m_aOnValue; 11802 } 11803 } 11804 // finally move the "Yes" appearance to the OnValue appearance 11805 for( int nKid = 0; nKid < nChildren; nKid++ ) 11806 { 11807 int nKidIndex = rGroupWidget.m_aKidsIndex[nKid]; 11808 PDFWidget& rKid = m_aWidgets[nKidIndex]; 11809 PDFAppearanceMap::iterator app_it = rKid.m_aAppearances.find( "N" ); 11810 if( app_it != rKid.m_aAppearances.end() ) 11811 { 11812 PDFAppearanceStreams::iterator stream_it = app_it->second.find( "Yes" ); 11813 if( stream_it != app_it->second.end() ) 11814 { 11815 SvMemoryStream* pStream = stream_it->second; 11816 app_it->second.erase( stream_it ); 11817 OStringBuffer aBuf( rKid.m_aOnValue.getLength()*2 ); 11818 appendName( rKid.m_aOnValue, aBuf ); 11819 (app_it->second)[ aBuf.makeStringAndClear() ] = pStream; 11820 } 11821 #if OSL_DEBUG_LEVEL > 1 11822 else 11823 fprintf( stderr, "error: RadioButton without \"Yes\" stream\n" ); 11824 #endif 11825 } 11826 // update selected radio button 11827 if( ! rKid.m_aValue.equalsAscii( "Off" ) ) 11828 { 11829 rGroupWidget.m_aValue = rKid.m_aValue; 11830 } 11831 } 11832 } 11833 } 11834 11835 sal_Int32 PDFWriterImpl::findRadioGroupWidget( const PDFWriter::RadioButtonWidget& rBtn ) 11836 { 11837 sal_Int32 nRadioGroupWidget = -1; 11838 11839 std::map< sal_Int32, sal_Int32 >::const_iterator it = m_aRadioGroupWidgets.find( rBtn.RadioGroup ); 11840 11841 if( it == m_aRadioGroupWidgets.end() ) 11842 { 11843 m_aRadioGroupWidgets[ rBtn.RadioGroup ] = nRadioGroupWidget = 11844 sal_Int32(m_aWidgets.size()); 11845 11846 // new group, insert the radiobutton 11847 m_aWidgets.push_back( PDFWidget() ); 11848 m_aWidgets.back().m_nObject = createObject(); 11849 m_aWidgets.back().m_nPage = m_nCurrentPage; 11850 m_aWidgets.back().m_eType = PDFWriter::RadioButton; 11851 m_aWidgets.back().m_nRadioGroup = rBtn.RadioGroup; 11852 m_aWidgets.back().m_nFlags |= 0x0000C000; // NoToggleToOff and Radio bits 11853 11854 createWidgetFieldName( sal_Int32(m_aWidgets.size()-1), rBtn ); 11855 } 11856 else 11857 nRadioGroupWidget = it->second; 11858 11859 return nRadioGroupWidget; 11860 } 11861 11862 sal_Int32 PDFWriterImpl::createControl( const PDFWriter::AnyWidget& rControl, sal_Int32 nPageNr ) 11863 { 11864 if( nPageNr < 0 ) 11865 nPageNr = m_nCurrentPage; 11866 11867 if( nPageNr < 0 || nPageNr >= (sal_Int32)m_aPages.size() ) 11868 return -1; 11869 11870 sal_Int32 nNewWidget = m_aWidgets.size(); 11871 m_aWidgets.push_back( PDFWidget() ); 11872 11873 m_aWidgets.back().m_nObject = createObject(); 11874 m_aWidgets.back().m_aRect = rControl.Location; 11875 m_aWidgets.back().m_nPage = nPageNr; 11876 m_aWidgets.back().m_eType = rControl.getType(); 11877 11878 sal_Int32 nRadioGroupWidget = -1; 11879 // for unknown reasons the radio buttons of a radio group must not have a 11880 // field name, else the buttons are in fact check boxes - 11881 // that is multiple buttons of the radio group can be selected 11882 if( rControl.getType() == PDFWriter::RadioButton ) 11883 nRadioGroupWidget = findRadioGroupWidget( static_cast<const PDFWriter::RadioButtonWidget&>(rControl) ); 11884 else 11885 { 11886 createWidgetFieldName( nNewWidget, rControl ); 11887 } 11888 11889 // caution: m_aWidgets must not be changed after here or rNewWidget may be invalid 11890 PDFWidget& rNewWidget = m_aWidgets[nNewWidget]; 11891 rNewWidget.m_aDescription = rControl.Description; 11892 rNewWidget.m_aText = rControl.Text; 11893 rNewWidget.m_nTextStyle = rControl.TextStyle & 11894 ( TEXT_DRAW_LEFT | TEXT_DRAW_CENTER | TEXT_DRAW_RIGHT | TEXT_DRAW_TOP | 11895 TEXT_DRAW_VCENTER | TEXT_DRAW_BOTTOM | 11896 TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK ); 11897 rNewWidget.m_nTabOrder = rControl.TabOrder; 11898 11899 // various properties are set via the flags (/Ff) property of the field dict 11900 if( rControl.ReadOnly ) 11901 rNewWidget.m_nFlags |= 1; 11902 if( rControl.getType() == PDFWriter::PushButton ) 11903 { 11904 const PDFWriter::PushButtonWidget& rBtn = static_cast<const PDFWriter::PushButtonWidget&>(rControl); 11905 if( rNewWidget.m_nTextStyle == 0 ) 11906 rNewWidget.m_nTextStyle = 11907 TEXT_DRAW_CENTER | TEXT_DRAW_VCENTER | 11908 TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK; 11909 11910 rNewWidget.m_nFlags |= 0x00010000; 11911 if( rBtn.URL.getLength() ) 11912 rNewWidget.m_aListEntries.push_back( rBtn.URL ); 11913 rNewWidget.m_bSubmit = rBtn.Submit; 11914 rNewWidget.m_bSubmitGet = rBtn.SubmitGet; 11915 rNewWidget.m_nDest = rBtn.Dest; 11916 createDefaultPushButtonAppearance( rNewWidget, rBtn ); 11917 } 11918 else if( rControl.getType() == PDFWriter::RadioButton ) 11919 { 11920 const PDFWriter::RadioButtonWidget& rBtn = static_cast<const PDFWriter::RadioButtonWidget&>(rControl); 11921 if( rNewWidget.m_nTextStyle == 0 ) 11922 rNewWidget.m_nTextStyle = 11923 TEXT_DRAW_VCENTER | TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK; 11924 /* PDF sees a RadioButton group as one radio button with 11925 * children which are in turn check boxes 11926 * 11927 * so we need to create a radio button on demand for a new group 11928 * and insert a checkbox for each RadioButtonWidget as its child 11929 */ 11930 rNewWidget.m_eType = PDFWriter::CheckBox; 11931 rNewWidget.m_nRadioGroup = rBtn.RadioGroup; 11932 11933 DBG_ASSERT( nRadioGroupWidget >= 0 && nRadioGroupWidget < (sal_Int32)m_aWidgets.size(), "no radio group parent" ); 11934 11935 PDFWidget& rRadioButton = m_aWidgets[nRadioGroupWidget]; 11936 rRadioButton.m_aKids.push_back( rNewWidget.m_nObject ); 11937 rRadioButton.m_aKidsIndex.push_back( nNewWidget ); 11938 rNewWidget.m_nParent = rRadioButton.m_nObject; 11939 11940 rNewWidget.m_aValue = OUString( RTL_CONSTASCII_USTRINGPARAM( "Off" ) ); 11941 rNewWidget.m_aOnValue = rBtn.OnValue; 11942 if( ! rRadioButton.m_aValue.getLength() && rBtn.Selected ) 11943 { 11944 rNewWidget.m_aValue = rNewWidget.m_aOnValue; 11945 rRadioButton.m_aValue = rNewWidget.m_aOnValue; 11946 } 11947 createDefaultRadioButtonAppearance( rNewWidget, rBtn ); 11948 11949 // union rect of radio group 11950 Rectangle aRect = rNewWidget.m_aRect; 11951 m_aPages[ nPageNr ].convertRect( aRect ); 11952 rRadioButton.m_aRect.Union( aRect ); 11953 } 11954 else if( rControl.getType() == PDFWriter::CheckBox ) 11955 { 11956 const PDFWriter::CheckBoxWidget& rBox = static_cast<const PDFWriter::CheckBoxWidget&>(rControl); 11957 if( rNewWidget.m_nTextStyle == 0 ) 11958 rNewWidget.m_nTextStyle = 11959 TEXT_DRAW_VCENTER | TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK; 11960 11961 rNewWidget.m_aValue = OUString::createFromAscii( rBox.Checked ? "Yes" : "Off" ); 11962 // create default appearance before m_aRect gets transformed 11963 createDefaultCheckBoxAppearance( rNewWidget, rBox ); 11964 } 11965 else if( rControl.getType() == PDFWriter::ListBox ) 11966 { 11967 if( rNewWidget.m_nTextStyle == 0 ) 11968 rNewWidget.m_nTextStyle = TEXT_DRAW_VCENTER; 11969 11970 const PDFWriter::ListBoxWidget& rLstBox = static_cast<const PDFWriter::ListBoxWidget&>(rControl); 11971 rNewWidget.m_aListEntries = rLstBox.Entries; 11972 rNewWidget.m_aSelectedEntries = rLstBox.SelectedEntries; 11973 rNewWidget.m_aValue = rLstBox.Text; 11974 if( rLstBox.DropDown ) 11975 rNewWidget.m_nFlags |= 0x00020000; 11976 if( rLstBox.Sort ) 11977 rNewWidget.m_nFlags |= 0x00080000; 11978 if( rLstBox.MultiSelect && !rLstBox.DropDown && (int)m_aContext.Version > (int)PDFWriter::PDF_1_3 ) 11979 rNewWidget.m_nFlags |= 0x00200000; 11980 11981 createDefaultListBoxAppearance( rNewWidget, rLstBox ); 11982 } 11983 else if( rControl.getType() == PDFWriter::ComboBox ) 11984 { 11985 if( rNewWidget.m_nTextStyle == 0 ) 11986 rNewWidget.m_nTextStyle = TEXT_DRAW_VCENTER; 11987 11988 const PDFWriter::ComboBoxWidget& rBox = static_cast<const PDFWriter::ComboBoxWidget&>(rControl); 11989 rNewWidget.m_aValue = rBox.Text; 11990 rNewWidget.m_aListEntries = rBox.Entries; 11991 rNewWidget.m_nFlags |= 0x00060000; // combo and edit flag 11992 if( rBox.Sort ) 11993 rNewWidget.m_nFlags |= 0x00080000; 11994 11995 PDFWriter::ListBoxWidget aLBox; 11996 aLBox.Name = rBox.Name; 11997 aLBox.Description = rBox.Description; 11998 aLBox.Text = rBox.Text; 11999 aLBox.TextStyle = rBox.TextStyle; 12000 aLBox.ReadOnly = rBox.ReadOnly; 12001 aLBox.Border = rBox.Border; 12002 aLBox.BorderColor = rBox.BorderColor; 12003 aLBox.Background = rBox.Background; 12004 aLBox.BackgroundColor = rBox.BackgroundColor; 12005 aLBox.TextFont = rBox.TextFont; 12006 aLBox.TextColor = rBox.TextColor; 12007 aLBox.DropDown = true; 12008 aLBox.Sort = rBox.Sort; 12009 aLBox.MultiSelect = false; 12010 aLBox.Entries = rBox.Entries; 12011 12012 createDefaultListBoxAppearance( rNewWidget, aLBox ); 12013 } 12014 else if( rControl.getType() == PDFWriter::Edit ) 12015 { 12016 if( rNewWidget.m_nTextStyle == 0 ) 12017 rNewWidget.m_nTextStyle = TEXT_DRAW_LEFT | TEXT_DRAW_VCENTER; 12018 12019 const PDFWriter::EditWidget& rEdit = static_cast<const PDFWriter::EditWidget&>(rControl); 12020 if( rEdit.MultiLine ) 12021 { 12022 rNewWidget.m_nFlags |= 0x00001000; 12023 rNewWidget.m_nTextStyle |= TEXT_DRAW_MULTILINE | TEXT_DRAW_WORDBREAK; 12024 } 12025 if( rEdit.Password ) 12026 rNewWidget.m_nFlags |= 0x00002000; 12027 if( rEdit.FileSelect && m_aContext.Version > PDFWriter::PDF_1_3 ) 12028 rNewWidget.m_nFlags |= 0x00100000; 12029 rNewWidget.m_nMaxLen = rEdit.MaxLen; 12030 rNewWidget.m_aValue = rEdit.Text; 12031 12032 createDefaultEditAppearance( rNewWidget, rEdit ); 12033 } 12034 12035 // convert to default user space now, since the mapmode may change 12036 // note: create default appearances before m_aRect gets transformed 12037 m_aPages[ nPageNr ].convertRect( rNewWidget.m_aRect ); 12038 12039 // insert widget to page's annotation list 12040 m_aPages[ nPageNr ].m_aAnnotations.push_back( rNewWidget.m_nObject ); 12041 12042 // mark page as having widgets 12043 m_aPages[ nPageNr ].m_bHasWidgets = true; 12044 12045 return nNewWidget; 12046 } 12047 12048 void PDFWriterImpl::beginControlAppearance( sal_Int32 nControl ) 12049 { 12050 if( nControl < 0 || nControl >= (sal_Int32)m_aWidgets.size() ) 12051 return; 12052 12053 PDFWidget& rWidget = m_aWidgets[ nControl ]; 12054 m_nCurrentControl = nControl; 12055 12056 SvMemoryStream* pControlStream = new SvMemoryStream( 1024, 1024 ); 12057 // back conversion of control rect to current MapMode; necessary because 12058 // MapMode between createControl and beginControlAppearance 12059 // could have changed; therefore the widget rectangle is 12060 // already converted 12061 Rectangle aBack( Point( rWidget.m_aRect.Left(), pointToPixel(m_aPages[m_nCurrentPage].getHeight()) - rWidget.m_aRect.Top() - rWidget.m_aRect.GetHeight() ), 12062 rWidget.m_aRect.GetSize() ); 12063 aBack = lcl_convert( m_aMapMode, 12064 m_aGraphicsStack.front().m_aMapMode, 12065 getReferenceDevice(), 12066 aBack ); 12067 beginRedirect( pControlStream, aBack ); 12068 writeBuffer( "/Tx BMC\n", 8 ); 12069 } 12070 12071 bool PDFWriterImpl::endControlAppearance( PDFWriter::WidgetState eState ) 12072 { 12073 bool bRet = false; 12074 if( ! m_aOutputStreams.empty() ) 12075 writeBuffer( "\nEMC\n", 5 ); 12076 SvMemoryStream* pAppearance = static_cast<SvMemoryStream*>(endRedirect()); 12077 if( pAppearance && m_nCurrentControl >= 0 && m_nCurrentControl < (sal_Int32)m_aWidgets.size() ) 12078 { 12079 PDFWidget& rWidget = m_aWidgets[ m_nCurrentControl ]; 12080 OString aState, aStyle; 12081 switch( rWidget.m_eType ) 12082 { 12083 case PDFWriter::PushButton: 12084 if( eState == PDFWriter::Up || eState == PDFWriter::Down ) 12085 { 12086 aState = (eState == PDFWriter::Up) ? "N" : "D"; 12087 aStyle = "Standard"; 12088 } 12089 break; 12090 case PDFWriter::CheckBox: 12091 if( eState == PDFWriter::Up || eState == PDFWriter::Down ) 12092 { 12093 aState = "N"; 12094 aStyle = (eState == PDFWriter::Up) ? "Off" : "Yes"; 12095 /* cf PDFReference 3rd ed. V1.4 p539: 12096 recommended name for on state is "Yes", 12097 recommended name for off state is "Off" 12098 */ 12099 } 12100 break; 12101 case PDFWriter::RadioButton: 12102 if( eState == PDFWriter::Up || eState == PDFWriter::Down ) 12103 { 12104 aState = "N"; 12105 if( eState == PDFWriter::Up ) 12106 aStyle = "Off"; 12107 else 12108 { 12109 OStringBuffer aBuf( rWidget.m_aOnValue.getLength()*2 ); 12110 appendName( rWidget.m_aOnValue, aBuf ); 12111 aStyle = aBuf.makeStringAndClear(); 12112 } 12113 } 12114 break; 12115 case PDFWriter::Edit: 12116 aState = "N"; 12117 aStyle = "Standard"; 12118 break; 12119 case PDFWriter::ListBox: 12120 case PDFWriter::ComboBox: 12121 case PDFWriter::Hierarchy: 12122 break; 12123 } 12124 if( aState.getLength() && aStyle.getLength() ) 12125 { 12126 // delete eventual existing stream 12127 PDFAppearanceStreams::iterator it = 12128 rWidget.m_aAppearances[ aState ].find( aStyle ); 12129 if( it != rWidget.m_aAppearances[ aState ].end() ) 12130 delete it->second; 12131 rWidget.m_aAppearances[ aState ][ aStyle ] = pAppearance; 12132 bRet = true; 12133 } 12134 } 12135 12136 if( ! bRet ) 12137 delete pAppearance; 12138 12139 m_nCurrentControl = -1; 12140 12141 return bRet; 12142 } 12143 12144 void PDFWriterImpl::addStream( const String& rMimeType, PDFOutputStream* pStream, bool bCompress ) 12145 { 12146 if( pStream ) 12147 { 12148 m_aAdditionalStreams.push_back( PDFAddStream() ); 12149 PDFAddStream& rStream = m_aAdditionalStreams.back(); 12150 rStream.m_aMimeType = rMimeType.Len() 12151 ? OUString( rMimeType ) 12152 : OUString( RTL_CONSTASCII_USTRINGPARAM( "application/octet-stream" ) ); 12153 rStream.m_pStream = pStream; 12154 rStream.m_bCompress = bCompress; 12155 } 12156 } 12157 12158 12159 12160