Impress Remote 1.0.5, tag sdremote-1.0.5
[LibreOffice.git] / vcl / generic / print / text_gfx.cxx
blobe5856672780eebd4f92eb01ce3bce198b1a9839f
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
21 #include <math.h>
23 #include "psputil.hxx"
24 #include "glyphset.hxx"
26 #include "generic/printergfx.hxx"
27 #include "vcl/fontmanager.hxx"
28 #include "vcl/helper.hxx"
30 #include "osl/thread.h"
32 #include "sal/alloca.h"
34 using namespace psp ;
36 namespace psp {
38 container for a font and its helper fonts:
39 1st font is the font substitute e.g. helvetica substitutes arial on the printer
40 2nd is the font itself
41 3rd is a fallback font, usually a font with unicode glyph repertoir (e.g. andale)
42 symbol fonts (adobe-fontspecific) may need special glyphmapping
43 (symbol page vc. latin page)
45 class Font3
47 private:
49 #define Font3Size 3
51 fontID mpFont [Font3Size];
52 bool mbSymbol;
54 public:
56 fontID GetFont (int nIdx) const
57 { return nIdx < Font3Size ? mpFont[nIdx] : -1 ; }
58 bool IsSymbolFont () const
59 { return mbSymbol; }
61 Font3 (const PrinterGfx &rGfx);
62 ~Font3 () {}
65 Font3::Font3(const PrinterGfx &rGfx)
67 mpFont[0] = rGfx.getFontSubstitute();
68 mpFont[1] = rGfx.GetFontID();
69 mpFont[2] = rGfx.getFallbackID();
70 // mpFont[2] = rGfx.GetFontID();
72 PrintFontManager &rMgr = PrintFontManager::get();
73 mbSymbol = mpFont[1] != -1 ?
74 rMgr.getFontEncoding(mpFont[1]) == RTL_TEXTENCODING_SYMBOL : false;
77 } // namespace psp
79 static int getVerticalDeltaAngle( sal_Unicode nChar )
81 int nRotation = GetVerticalFlags(nChar);
82 if (nRotation == GF_ROTR)
83 return -900;
84 if (nRotation == GF_ROTL)
85 return 900;
86 return 0;
89 void
90 PrinterGfx::PSUploadPS1Font (sal_Int32 nFontID)
92 std::list< sal_Int32 >::iterator aFont;
93 // already in the document header ?
94 for (aFont = maPS1Font.begin(); aFont != maPS1Font.end(); ++aFont )
95 if( nFontID == *aFont )
96 return;
98 // no occurrenc yet, mark for download
99 // add the fontid to the list
100 maPS1Font.push_back (nFontID);
104 * implement text handling printer routines,
107 sal_uInt16
108 PrinterGfx::SetFont(
109 sal_Int32 nFontID,
110 sal_Int32 nHeight,
111 sal_Int32 nWidth,
112 sal_Int32 nAngle,
113 bool bVertical,
114 bool bArtItalic,
115 bool bArtBold
118 // font and encoding will be set by drawText again immediately
119 // before PSShowText
120 mnFontID = nFontID;
121 maVirtualStatus.maFont = rtl::OString();
122 maVirtualStatus.maEncoding = RTL_TEXTENCODING_DONTKNOW;
123 maVirtualStatus.mnTextHeight = nHeight;
124 maVirtualStatus.mnTextWidth = nWidth;
125 maVirtualStatus.mbArtItalic = bArtItalic;
126 maVirtualStatus.mbArtBold = bArtBold;
127 mnTextAngle = nAngle;
128 mbTextVertical = bVertical;
130 return 0;
133 void PrinterGfx::drawGlyphs(
134 const Point& rPoint,
135 sal_uInt32* pGlyphIds,
136 sal_Unicode* pUnicodes,
137 sal_Int16 nLen,
138 sal_Int32* pDeltaArray
142 // draw the string
143 // search for a glyph set matching the set font
144 std::list< GlyphSet >::iterator aIter;
145 for (aIter = maPS3Font.begin(); aIter != maPS3Font.end(); ++aIter)
146 if ( ((*aIter).GetFontID() == mnFontID)
147 && ((*aIter).IsVertical() == mbTextVertical))
149 (*aIter).DrawGlyphs (*this, rPoint, pGlyphIds, pUnicodes, nLen, pDeltaArray);
150 break;
153 // not found ? create a new one
154 if (aIter == maPS3Font.end())
156 maPS3Font.push_back (GlyphSet(mnFontID, mbTextVertical));
157 maPS3Font.back().DrawGlyphs (*this, rPoint, pGlyphIds, pUnicodes, nLen, pDeltaArray);
161 void PrinterGfx::DrawGlyphs(
162 const Point& rPoint,
163 sal_GlyphId* pGlyphIds,
164 sal_Unicode* pUnicodes,
165 sal_Int16 nLen,
166 sal_Int32* pDeltaArray
169 if( nLen <= 0 )
170 return;
172 if ( !mrFontMgr.isFontDownloadingAllowed( mnFontID ) )
174 LicenseWarning(rPoint, pUnicodes, nLen, pDeltaArray);
175 return;
178 if( mrFontMgr.getFontType( mnFontID ) != fonttype::TrueType )
180 DrawText( rPoint, pUnicodes, nLen, pDeltaArray );
181 return;
184 // move and rotate the user coordinate system
185 // avoid the gsave/grestore for the simple cases since it allows
186 // reuse of the current font if it hasn't changed
187 sal_Int32 nCurrentTextAngle = mnTextAngle;
188 Point aPoint( rPoint );
190 if (nCurrentTextAngle != 0)
192 PSGSave ();
193 PSTranslate (rPoint);
194 PSRotate (nCurrentTextAngle);
195 mnTextAngle = 0;
196 aPoint = Point( 0, 0 );
199 if( mbTextVertical )
201 // vertical glyphs can have an additional rotation ... sigh.
202 // so break up text in chunks of normal glyphs and print out
203 // specially rotated glyphs extra
204 sal_uInt32* pTempGlyphIds = (sal_uInt32*)alloca(sizeof(sal_Int32)*nLen);
205 sal_Int32* pTempDelta = (sal_Int32*)alloca(sizeof(sal_Int32)*nLen);
206 sal_Unicode* pTempUnicodes = (sal_Unicode*)alloca(sizeof(sal_Unicode)*nLen);
207 sal_Int16 nTempLen = 0;
208 sal_Int32 nTempFirstDelta = 0;
209 Point aRotPoint;
210 sal_Int32 nTextHeight = maVirtualStatus.mnTextHeight;
211 sal_Int32 nTextWidth = maVirtualStatus.mnTextWidth ? maVirtualStatus.mnTextWidth : maVirtualStatus.mnTextHeight;
212 sal_Int32 nAscend = mrFontMgr.getFontAscend( mnFontID );
213 sal_Int32 nDescend = mrFontMgr.getFontDescend( mnFontID );
215 nDescend = nDescend * nTextHeight / 1000;
216 nAscend = nAscend * nTextHeight / 1000;
218 for( sal_Int16 i = 0; i < nLen; i++ )
220 const sal_GlyphId nRot = pGlyphIds[i] & GF_ROTMASK;
221 if( nRot == GF_NONE )
223 pTempUnicodes[nTempLen] = pUnicodes[i];
224 pTempGlyphIds[nTempLen] = pGlyphIds[i];
225 if( nTempLen > 0 )
226 pTempDelta[nTempLen-1] = pDeltaArray[i-1]-nTempFirstDelta;
227 else
229 // the first element in pDeltaArray shows
230 // the offset of the second character
231 // so if the first glyph is normal
232 // then we do not need to move the delta indices
233 // else we have to move them down by one and
234 // recalculate aPoint and all deltas
235 if( i != 0 )
236 nTempFirstDelta = pDeltaArray[ i-1 ];
238 nTempLen++;
240 else
242 sal_Int32 nOffset = i > 0 ? pDeltaArray[i-1] : 0;
243 sal_Int32 nRotAngle = 0;
244 switch( nRot )
246 case GF_ROTR:
247 nRotAngle = 2700;
248 aRotPoint = Point( -nAscend*nTextWidth/nTextHeight, -nDescend*nTextWidth/nTextHeight - nOffset );
249 break;
250 case GF_VERT:
251 nRotAngle = 1800;
252 aRotPoint = Point( -nOffset, (nAscend+nDescend) );
253 break;
254 case GF_ROTL:
255 nRotAngle = 900;
256 aRotPoint = Point( -nDescend*nTextWidth/nTextHeight, nOffset + nAscend*nTextWidth/nTextHeight );
257 break;
259 sal_GlyphId nRotGlyphId = pGlyphIds[i];
260 sal_Unicode nRotUnicode = pUnicodes[i];
261 sal_Int32 nRotDelta = 0;
263 // transform matrix to new individual direction
264 PSGSave ();
265 GraphicsStatus aSaveStatus = maVirtualStatus;
266 if( nRot != 2 ) // switch font aspect
268 maVirtualStatus.mnTextWidth = nTextHeight;
269 maVirtualStatus.mnTextHeight = nTextWidth;
271 if( aPoint.X() || aPoint.Y() )
272 PSTranslate( aPoint );
273 PSRotate (nRotAngle);
274 // draw the rotated glyph
275 drawGlyphs( aRotPoint, &nRotGlyphId, &nRotUnicode, 1, &nRotDelta );
277 // restore previous state
278 maVirtualStatus = aSaveStatus;
279 PSGRestore();
283 pGlyphIds = pTempGlyphIds;
284 pUnicodes = pTempUnicodes;
285 pDeltaArray = pTempDelta;
286 nLen = nTempLen;
288 aPoint.X() += nTempFirstDelta;
291 if( nLen > 0 )
292 drawGlyphs( aPoint, pGlyphIds, pUnicodes, nLen, pDeltaArray );
294 // restore the user coordinate system
295 if (nCurrentTextAngle != 0)
297 PSGRestore ();
298 mnTextAngle = nCurrentTextAngle;
302 void
303 PrinterGfx::DrawText (
304 const Point& rPoint,
305 const sal_Unicode* pStr,
306 sal_Int16 nLen,
307 const sal_Int32* pDeltaArray
310 fontID nRestoreFont = mnFontID;
312 // setup font[substitutes] and map the string into the symbol area in case of
313 // symbol font
314 Font3 aFont(*this);
315 sal_Unicode *pEffectiveStr;
316 if ( aFont.IsSymbolFont() )
318 pEffectiveStr = (sal_Unicode*)alloca(nLen * sizeof(pStr[0]));
319 for (int i = 0; i < nLen; i++)
320 pEffectiveStr[i] = pStr[i] < 256 ? pStr[i] + 0xF000 : pStr[i];
322 else
324 pEffectiveStr = const_cast<sal_Unicode*>(pStr);
327 fontID *pFontMap = (fontID*) alloca(nLen * sizeof(fontID));
328 sal_Int32 *pCharWidth = (sal_Int32*) alloca(nLen * sizeof(sal_Int32));
330 for( int n = 0; n < nLen; n++ )
332 CharacterMetric aBBox;
333 pFontMap[n] = getCharMetric (aFont, pEffectiveStr[n], &aBBox);
334 pCharWidth[n] = getCharWidth (mbTextVertical, pEffectiveStr[n], &aBBox);
337 // setup a new delta array, use virtual resolution of 1000
338 sal_Int32* pNewDeltaArray = (sal_Int32*)alloca( sizeof( sal_Int32 )*nLen );
339 if ( pDeltaArray != 0)
341 for (int i = 0; i < nLen - 1; i++)
342 pNewDeltaArray[i] = 1000 * pDeltaArray[i];
343 pNewDeltaArray[nLen - 1] = 0;
345 else
347 pNewDeltaArray[0] = pCharWidth[0];
348 for (int i = 1; i < nLen; i++)
349 pNewDeltaArray[i] = pNewDeltaArray[i-1] + pCharWidth[i];
352 // move and rotate the user coordinate system
353 // avoid the gsave/grestore for the simple cases since it allows
354 // reuse of the current font if it hasn't changed
355 sal_Int32 nCurrentTextAngle = mnTextAngle;
356 sal_Int32 nCurrentPointX;
357 sal_Int32 nCurrentPointY;
359 if (nCurrentTextAngle != 0)
361 PSGSave ();
362 PSTranslate (rPoint);
363 PSRotate (nCurrentTextAngle);
364 mnTextAngle = 0;
366 nCurrentPointX = 0;
367 nCurrentPointY = 0;
369 else
371 nCurrentPointX = rPoint.X();
372 nCurrentPointY = rPoint.Y();
375 // draw the string
376 sal_Int32 nDelta = 0;
377 for (int nTo = 0; nTo < nLen; )
379 int nFrom = nTo;
380 fontID nFont = pFontMap[ nFrom ];
382 while ((nTo < nLen) && (nFont == pFontMap[nTo]))
384 pNewDeltaArray[ nTo ] = (sal_Int32)(((0.5 + pNewDeltaArray[ nTo ]) / 1000.0) - nDelta);
385 nTo++ ;
388 SetFont( nFont,
389 maVirtualStatus.mnTextHeight, maVirtualStatus.mnTextWidth,
390 mnTextAngle,
391 mbTextVertical,
392 maVirtualStatus.mbArtItalic,
393 maVirtualStatus.mbArtBold
396 if (mbTextVertical)
398 drawVerticalizedText(
399 Point(nCurrentPointX + nDelta, nCurrentPointY),
400 pEffectiveStr + nFrom, nTo - nFrom,
401 pNewDeltaArray + nFrom );
403 else
405 drawText(
406 Point(nCurrentPointX + nDelta, nCurrentPointY),
407 pEffectiveStr + nFrom, nTo - nFrom,
408 pDeltaArray == NULL ? NULL : pNewDeltaArray + nFrom );
410 nDelta += pNewDeltaArray[ nTo - 1 ];
413 // restore the user coordinate system
414 if (nCurrentTextAngle != 0)
416 PSGRestore ();
417 mnTextAngle = nCurrentTextAngle;
420 // restore the original font settings
421 SetFont( nRestoreFont,
422 maVirtualStatus.mnTextHeight, maVirtualStatus.mnTextWidth,
423 mnTextAngle, mbTextVertical,
424 maVirtualStatus.mbArtItalic,
425 maVirtualStatus.mbArtBold
429 void PrinterGfx::drawVerticalizedText(
430 const Point& rPoint,
431 const sal_Unicode* pStr,
432 sal_Int16 nLen,
433 const sal_Int32* pDeltaArray
436 sal_Int32* pDelta = (sal_Int32*)alloca( nLen * sizeof(sal_Int32) );
438 int nTextScale = maVirtualStatus.mnTextWidth ? maVirtualStatus.mnTextWidth : maVirtualStatus.mnTextHeight;
439 int nNormalAngle = mnTextAngle;
440 int nDeltaAngle, nLastPos = 0;
442 double fSin = sin( -2.0*M_PI*nNormalAngle/3600 );
443 double fCos = cos( -2.0*M_PI*nNormalAngle/3600 );
445 PrintFontManager &rMgr = PrintFontManager::get();
446 PrintFontInfo aInfo;
447 rMgr.getFontInfo( mnFontID, aInfo );
449 bool* pGsubFlags = (bool*)alloca( nLen * sizeof(bool) );
450 rMgr.hasVerticalSubstitutions( mnFontID, pStr, nLen, pGsubFlags );
452 Point aPoint( rPoint );
453 for( int i = 0; i < nLen; )
455 while( ( nDeltaAngle = getVerticalDeltaAngle( pStr[i] ) ) == 0 && i < nLen )
456 i++;
457 if( i <= nLen && i > nLastPos )
459 for( int n = nLastPos; n < i; n++ )
460 pDelta[n] = pDeltaArray[n] - (aPoint.X() - rPoint.X() );
462 SetFont( mnFontID,
463 maVirtualStatus.mnTextHeight, maVirtualStatus.mnTextWidth,
464 nNormalAngle, mbTextVertical,
465 maVirtualStatus.mbArtItalic,
466 maVirtualStatus.mbArtBold );
467 drawText( aPoint, pStr + nLastPos, i - nLastPos, pDelta + nLastPos );
469 aPoint.X() = (sal_Int32)(rPoint.X() + ((double)pDeltaArray[i-1] * fCos));
470 aPoint.Y() = (sal_Int32)(rPoint.Y() + ((double)pDeltaArray[i-1] * fSin));
472 if( i < nLen )
474 int nOldWidth = maVirtualStatus.mnTextWidth;
475 int nOldHeight = maVirtualStatus.mnTextHeight;
476 SetFont( mnFontID,
477 nTextScale,
478 maVirtualStatus.mnTextHeight,
479 nNormalAngle + nDeltaAngle,
480 mbTextVertical,
481 maVirtualStatus.mbArtItalic,
482 maVirtualStatus.mbArtBold );
484 double nA = nTextScale * aInfo.m_nAscend / 1000.0;
485 double nD = nTextScale * aInfo.m_nDescend / 1000.0;
486 double fStretch = (double)maVirtualStatus.mnTextWidth / maVirtualStatus.mnTextHeight;
487 if( !pGsubFlags[i] )
488 nD *= fStretch;
490 Point aPos( aPoint );
491 switch( nDeltaAngle )
493 case +900:
494 aPos.X() += (sal_Int32)(+nA * fCos + nD * fSin);
495 aPos.Y() += (sal_Int32)(-nA * fSin + nD * fCos);
496 break;
497 case -900:
498 aPos.X() += (sal_Int32)(+nA * fSin + nD * fCos);
499 aPos.Y() += (sal_Int32)(-(nTextScale*fStretch - nD) * fCos);
500 break;
502 drawText( aPos, pStr+i, 1, NULL );
503 if( i < nLen-1 && pDeltaArray )
505 aPoint.X() = (sal_Int32)(rPoint.X() + ((double)pDeltaArray[i] * fCos));
506 aPoint.Y() = (sal_Int32)(rPoint.Y() + ((double)pDeltaArray[i] * fSin));
509 // swap text width/height again
510 SetFont( mnFontID,
511 nOldHeight,
512 nOldWidth,
513 nNormalAngle,
514 mbTextVertical,
515 maVirtualStatus.mbArtItalic,
516 maVirtualStatus.mbArtBold );
518 i++;
519 nLastPos = i;
521 mnTextAngle = nNormalAngle;
524 void
525 PrinterGfx::LicenseWarning(const Point& rPoint, const sal_Unicode* pStr,
526 sal_Int16 nLen, const sal_Int32* pDeltaArray)
528 // treat it like a builtin font in case a user has that font also in the
529 // printer. This is not so unlikely as it may seem; no print embedding
530 // licensed fonts are often used (or so they say) in companies:
531 // they are installed on displays and printers, but get not embedded in
532 // they are installed on displays and printers, but get not embedded in
533 // print files or documents because they are not licensed for use outside
534 // the company.
535 rtl::OString aMessage( "The font " );
536 aMessage += rtl::OUStringToOString( mrFontMgr.getPSName(mnFontID),
537 RTL_TEXTENCODING_ASCII_US );
538 aMessage += " could not be downloaded\nbecause its license does not allow for that";
539 PSComment( aMessage.getStr() );
541 rtl::OString aFontName = rtl::OUStringToOString(
542 mrFontMgr.getPSName(mnFontID),
543 RTL_TEXTENCODING_ASCII_US);
544 PSSetFont (aFontName, RTL_TEXTENCODING_ISO_8859_1);
546 sal_Size nSize = 4 * nLen;
547 sal_uChar* pBuffer = (sal_uChar*)alloca (nSize* sizeof(sal_uChar));
549 ConverterFactory &rCvt = GetConverterFactory ();
550 nSize = rCvt.Convert (pStr, nLen, pBuffer, nSize, RTL_TEXTENCODING_ISO_8859_1);
552 PSMoveTo (rPoint);
553 PSShowText (pBuffer, nLen, nSize, pDeltaArray);
556 void
557 PrinterGfx::drawText(
558 const Point& rPoint,
559 const sal_Unicode* pStr,
560 sal_Int16 nLen,
561 const sal_Int32* pDeltaArray
564 if (!(nLen > 0))
565 return;
567 fonttype::type eType = mrFontMgr.getFontType (mnFontID);
569 if (eType == fonttype::Type1)
570 PSUploadPS1Font (mnFontID);
572 if ( eType == fonttype::TrueType
573 && !mrFontMgr.isFontDownloadingAllowed(mnFontID))
575 LicenseWarning(rPoint, pStr, nLen, pDeltaArray);
576 return;
579 if( mrFontMgr.getUseOnlyFontEncoding( mnFontID ) )
581 GlyphSet aGSet( mnFontID, mbTextVertical );
582 aGSet.DrawText( *this, rPoint, pStr, nLen, pDeltaArray );
583 return;
586 // search for a glyph set matching the set font
587 std::list< GlyphSet >::iterator aIter;
588 for (aIter = maPS3Font.begin(); aIter != maPS3Font.end(); ++aIter)
589 if ( ((*aIter).GetFontID() == mnFontID)
590 && ((*aIter).IsVertical() == mbTextVertical))
592 (*aIter).DrawText (*this, rPoint, pStr, nLen, pDeltaArray);
593 break;
596 // not found ? create a new one
597 if (aIter == maPS3Font.end())
599 maPS3Font.push_back (GlyphSet(mnFontID, mbTextVertical));
600 maPS3Font.back().DrawText (*this, rPoint, pStr, nLen, pDeltaArray);
605 PrinterGfx::getCharWidth (sal_Bool b_vert, sal_Unicode n_char, CharacterMetric *p_bbox)
607 b_vert = b_vert && (getVerticalDeltaAngle(n_char) != 0);
608 int w = b_vert ? p_bbox->height : p_bbox->width;
609 w *= maVirtualStatus.mnTextWidth ? maVirtualStatus.mnTextWidth : maVirtualStatus.mnTextHeight;
610 return w;
613 fontID
614 PrinterGfx::getCharMetric (const Font3 &rFont, sal_Unicode n_char, CharacterMetric *p_bbox)
616 p_bbox->width = -1;
617 p_bbox->height = -1;
619 for (fontID n = 0; n < 3; n++)
621 fontID n_font = rFont.GetFont(n);
622 if (n_font != -1)
623 mrFontMgr.getMetrics( n_font, n_char, n_char, p_bbox );
624 if (p_bbox->width >= 0 && p_bbox->height >= 0)
625 return n_font;
627 if (n_char != '?')
628 return getCharMetric (rFont, '?', p_bbox);
630 return rFont.GetFont(0) != -1 ? rFont.GetFont(0) : rFont.GetFont(1);
633 fontID
634 PrinterGfx::getFontSubstitute () const
636 if( mpFontSubstitutes )
638 ::boost::unordered_map< fontID, fontID >::const_iterator it =
639 mpFontSubstitutes->find( mnFontID );
640 if( it != mpFontSubstitutes->end() )
641 return it->second;
644 return -1;
647 sal_Int32
648 PrinterGfx::GetCharWidth (sal_Unicode nFrom, sal_Unicode nTo, long *pWidthArray)
650 Font3 aFont(*this);
651 if (aFont.IsSymbolFont() && (nFrom < 256) && (nTo < 256))
653 nFrom += 0xF000;
654 nTo += 0xF000;
657 for( int n = 0; n < (nTo - nFrom + 1); n++ )
659 CharacterMetric aBBox;
660 getCharMetric (aFont, n + nFrom, &aBBox);
661 pWidthArray[n] = getCharWidth (mbTextVertical, n + nFrom, &aBBox);
664 // returned metrics have postscript precision
665 return 1000;
668 const ::std::list< KernPair >& PrinterGfx::getKernPairs( bool bVertical ) const
671 * Note: this is only a 80% solution: if a font is only
672 * partially substituted in a string due to missing glyphs
673 * the results may not be perfect; the more so the more the
674 * substitution differs from the original metricwise. But
675 * vcl only asks for KernPairs for each font once and NOT
676 * in a string context this is the best we can do.
677 * In future the kerning should be done on a per string basis.
679 fontID nFont = mnFontID;
680 if( mpFontSubstitutes )
682 ::boost::unordered_map< fontID, fontID >::const_iterator it =
683 mpFontSubstitutes->find( mnFontID );
684 if( it != mpFontSubstitutes->end() )
685 nFont = it->second;
687 return mrFontMgr.getKernPairs( nFont, bVertical );
691 * spool the converted truetype fonts to the page header after the page body is
692 * complete
693 * for Type1 fonts spool additional reencoding vectors that are necessary to access the
694 * whole font
697 void
698 PrinterGfx::OnEndPage ()
702 void
703 PrinterGfx::OnEndJob ()
705 maPS3Font.clear();
706 maPS1Font.clear();
709 void
710 PrinterGfx::writeResources( osl::File* pFile, std::list< rtl::OString >& rSuppliedFonts, std::list< rtl::OString >& rNeededFonts )
712 // write all type 1 fonts
713 std::list< sal_Int32 >::iterator aFont;
714 // already in the document header ?
715 for (aFont = maPS1Font.begin(); aFont != maPS1Font.end(); ++aFont)
717 const rtl::OString& rSysPath (mrFontMgr.getFontFileSysPath(*aFont) );
718 rtl::OUString aUNCPath;
719 osl::File::getFileURLFromSystemPath (OStringToOUString (rSysPath, osl_getThreadTextEncoding()), aUNCPath);
720 osl::File aFontFile (aUNCPath);
722 // provide the pfb or pfa font as a (pfa-)font resource
723 rtl::OString aPostScriptName =
724 rtl::OUStringToOString ( mrFontMgr.getPSName(*aFont),
725 RTL_TEXTENCODING_ASCII_US );
727 WritePS (pFile, "%%BeginResource: font ");
728 WritePS (pFile, aPostScriptName.getStr());
729 WritePS (pFile, "\n");
731 osl::File::RC nError = aFontFile.open(osl_File_OpenFlag_Read);
732 if (nError == osl::File::E_None)
734 convertPfbToPfa (aFontFile, *pFile);
735 aFontFile.close ();
737 char lastchar = '\n';
739 if (pFile->setPos(osl_Pos_Current, -1) == osl::FileBase::E_None)
741 sal_uInt64 uBytes(1);
742 pFile->read((void *)(&lastchar), uBytes, uBytes);
745 if (lastchar != '\n')
746 WritePS (pFile, "\n");
748 WritePS (pFile, "%%EndResource\n");
749 rSuppliedFonts.push_back( aPostScriptName );
752 // write glyphsets and reencodings
753 std::list< GlyphSet >::iterator aIter;
754 for (aIter = maPS3Font.begin(); aIter != maPS3Font.end(); ++aIter)
756 if (aIter->GetFontType() == fonttype::TrueType)
758 aIter->PSUploadFont (*pFile, *this, mbUploadPS42Fonts ? true : false, rSuppliedFonts );
760 else
761 // ( aIter->GetFontType() == fonttype::Type1
762 // || aIter->GetFontType() == fonttype::Builtin )
764 aIter->PSUploadEncoding (pFile, *this);
765 if( aIter->GetFontType() == fonttype::Builtin )
766 rNeededFonts.push_back(
767 rtl::OUStringToOString(
768 mrFontMgr.getPSName( aIter->GetFontID() ),
769 RTL_TEXTENCODING_ASCII_US ) );
773 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */