summaryrefslogtreecommitdiff
path: root/filter/source/svg/svgfontexport.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'filter/source/svg/svgfontexport.cxx')
-rw-r--r--filter/source/svg/svgfontexport.cxx258
1 files changed, 258 insertions, 0 deletions
diff --git a/filter/source/svg/svgfontexport.cxx b/filter/source/svg/svgfontexport.cxx
new file mode 100644
index 000000000000..8fa9db03eacf
--- /dev/null
+++ b/filter/source/svg/svgfontexport.cxx
@@ -0,0 +1,258 @@
+ /*************************************************************************
+ *
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * Copyright 2000, 2010 Oracle and/or its affiliates.
+ *
+ * OpenOffice.org - a multi-platform office productivity suite
+ *
+ * This file is part of OpenOffice.org.
+ *
+ * OpenOffice.org is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Lesser General Public License version 3
+ * only, as published by the Free Software Foundation.
+ *
+ * OpenOffice.org is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU Lesser General Public License version 3 for more details
+ * (a copy is included in the LICENSE file that accompanied this code).
+ *
+ * You should have received a copy of the GNU Lesser General Public License
+ * version 3 along with OpenOffice.org. If not, see
+ * <http://www.openoffice.org/license.html>
+ * for a copy of the LGPLv3 License.
+ *
+ ************************************************************************/
+
+// MARKER(update_precomp.py): autogen include statement, do not remove
+#include "precompiled_filter.hxx"
+
+
+#include "svgfontexport.hxx"
+
+static const sal_Int32 nFontEM = 2048;
+
+// -----------------
+// - SVGFontExport -
+// -----------------
+
+SVGFontExport::SVGFontExport( SvXMLExport& rExport, const ::std::vector< ObjectRepresentation >& rObjects ) :
+ mrExport( rExport ),
+ maObjects( rObjects ),
+ mnCurFontId( 0 )
+{
+}
+
+// -----------------------------------------------------------------------------
+
+SVGFontExport::~SVGFontExport()
+{
+}
+
+// -----------------------------------------------------------------------------
+
+void SVGFontExport::implCollectGlyphs()
+{
+ VirtualDevice aVDev;
+ ObjectVector::const_iterator aIter( maObjects.begin() );
+
+ aVDev.EnableOutput( sal_False );
+
+ while( aIter != maObjects.end() )
+ {
+ if( (*aIter).HasRepresentation() )
+ {
+ const GDIMetaFile& rMtf = (*aIter).GetRepresentation();
+
+ aVDev.Push();
+
+ for( sal_uInt32 i = 0, nCount = rMtf.GetActionCount(); i < nCount; ++i )
+ {
+ ::rtl::OUString aText;
+ MetaAction* pAction = rMtf.GetAction( i );
+ const USHORT nType = pAction->GetType();
+
+ switch( nType )
+ {
+ case( META_TEXT_ACTION ):
+ {
+ const MetaTextAction* pA = (const MetaTextAction*) pAction;
+ aText = String( pA->GetText(), pA->GetIndex(), pA->GetLen() );
+ }
+ break;
+
+ case( META_TEXTRECT_ACTION ):
+ {
+ const MetaTextRectAction* pA = (const MetaTextRectAction*) pAction;
+ aText = pA->GetText();
+ }
+ break;
+
+ case( META_TEXTARRAY_ACTION ):
+ {
+ const MetaTextArrayAction* pA = (const MetaTextArrayAction*) pAction;
+ aText = String( pA->GetText(), pA->GetIndex(), pA->GetLen() );
+ }
+ break;
+
+ case( META_STRETCHTEXT_ACTION ):
+ {
+ const MetaStretchTextAction* pA = (const MetaStretchTextAction*) pAction;
+ aText = String( pA->GetText(), pA->GetIndex(), pA->GetLen() );
+ }
+ break;
+
+ default:
+ pAction->Execute( &aVDev );
+ break;
+ }
+
+ if( aText.getLength() )
+ {
+ const String& rFontName = aVDev.GetFont().GetName();
+ const sal_Unicode* pStr = aText.getStr();
+
+ for( sal_uInt32 j = 0, nLen = aText.getLength(); j < nLen; ++j )
+ maGlyphs[ rFontName ].insert( pStr[ j ] );
+ }
+ }
+
+ aVDev.Pop();
+ }
+
+ ++aIter;
+ }
+}
+
+// -----------------------------------------------------------------------------
+
+void SVGFontExport::implEmbedFont( const ::rtl::OUString& rFontName, const ::std::set< sal_Unicode >& rGlyphs )
+{
+#ifdef _SVG_EMBED_FONTS
+ ::std::set< sal_Unicode >::const_iterator aIter( rGlyphs.begin() );
+ const ::rtl::OUString aEmbeddedFontStr( B2UCONST( "EmbeddedFont_" ) );
+
+ {
+ SvXMLElementExport aExp( mrExport, XML_NAMESPACE_NONE, "defs", TRUE, TRUE );
+ ::rtl::OUString aCurIdStr( aEmbeddedFontStr );
+ ::rtl::OUString aUnitsPerEM( SVGActionWriter::GetValueString( nFontEM ) );
+ VirtualDevice aVDev;
+ Font aFont( rFontName, Size( 0, nFontEM ) );
+
+ aVDev.SetMapMode( MAP_100TH_MM );
+ aFont.SetAlign( ALIGN_BASELINE );
+ aVDev.SetFont( aFont );
+
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "id", aCurIdStr += SVGActionWriter::GetValueString( ++mnCurFontId ) );
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "horiz-adv-x", aUnitsPerEM );
+
+ {
+ SvXMLElementExport aExp2( mrExport, XML_NAMESPACE_NONE, "font", TRUE, TRUE );
+ Point aPos;
+ Size aSize( nFontEM, nFontEM );
+ PolyPolygon aMissingGlyphPolyPoly( Rectangle( aPos, aSize ) );
+
+ aMissingGlyphPolyPoly.Move( 0, -nFontEM );
+ aMissingGlyphPolyPoly.Scale( 1.0, -1.0 );
+
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "font-family", GetMappedFontName( rFontName ) );
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "units-per-em", aUnitsPerEM );
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "ascent", SVGActionWriter::GetValueString( aVDev.GetFontMetric().GetAscent() ) );
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "descent", SVGActionWriter::GetValueString( aVDev.GetFontMetric().GetDescent() ) );
+
+ {
+ SvXMLElementExport aExp3( mrExport, XML_NAMESPACE_NONE, "font-face", TRUE, TRUE );
+ }
+
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "horiz-adv-x", SVGActionWriter::GetValueString( aSize.Width() ) );
+
+ {
+ SvXMLElementExport aExp3( mrExport, XML_NAMESPACE_NONE, "missing-glyph", TRUE, TRUE );
+
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "style", B2UCONST( "fill:none;stroke:black;stroke-width:33" ) );
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "d", SVGActionWriter::GetPathString( aMissingGlyphPolyPoly, sal_False ) );
+
+ {
+ SvXMLElementExport aExp4( mrExport, XML_NAMESPACE_NONE, "path", TRUE, TRUE );
+ }
+ }
+
+ while( aIter != rGlyphs.end() )
+ {
+ implEmbedGlyph( aVDev, ::rtl::OUString( *aIter ) );
+ ++aIter;
+ }
+ }
+ }
+#endif
+}
+
+// -----------------------------------------------------------------------------
+
+void SVGFontExport::implEmbedGlyph( OutputDevice& rOut, const ::rtl::OUString& rGlyphs )
+{
+ PolyPolygon aPolyPoly;
+ ::rtl::OUString aStr( rGlyphs );
+ const sal_Unicode nSpace = ' ';
+
+ if( rOut.GetTextOutline( aPolyPoly, aStr ) )
+ {
+ Rectangle aBoundRect;
+
+ aPolyPoly.Scale( 1.0, -1.0 );
+
+ if( !rOut.GetTextBoundRect( aBoundRect, aStr ) )
+ aBoundRect = Rectangle( Point( 0, 0 ), Size( rOut.GetTextWidth( aStr ), 0 ) );
+
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "unicode", aStr );
+
+ if( rGlyphs[ 0 ] == nSpace )
+ aBoundRect = Rectangle( Point( 0, 0 ), Size( rOut.GetTextWidth( sal_Unicode( 'x' ) ), 0 ) );
+
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "horiz-adv-x", SVGActionWriter::GetValueString( aBoundRect.GetWidth() ) );
+
+ {
+ SvXMLElementExport aExp( mrExport, XML_NAMESPACE_NONE, "glyph", TRUE, TRUE );
+ const ::rtl::OUString aPathString( SVGActionWriter::GetPathString( aPolyPoly, sal_False ) );
+
+ if( aPathString.getLength() )
+ {
+ mrExport.AddAttribute( XML_NAMESPACE_NONE, "d", aPathString );
+
+ {
+ SvXMLElementExport aElem( mrExport, XML_NAMESPACE_NONE, B2UCONST( "path" ), TRUE, TRUE );
+ }
+ }
+ }
+ }
+}
+
+// -----------------------------------------------------------------------------
+
+void SVGFontExport::EmbedFonts()
+{
+ implCollectGlyphs();
+
+ GlyphMap::const_iterator aIter( maGlyphs.begin() );
+
+ while( aIter != maGlyphs.end() )
+ {
+ implEmbedFont( (*aIter).first, (*aIter).second );
+ ++aIter;
+ }
+}
+
+// -----------------------------------------------------------------------------
+
+::rtl::OUString SVGFontExport::GetMappedFontName( const ::rtl::OUString& rFontName ) const
+{
+ ::rtl::OUString aRet( String( rFontName ).GetToken( 0, ';' ) );
+
+#ifdef _SVG_EMBED_FONTS
+ if( mnCurFontId )
+ aRet += B2UCONST( " embedded" );
+#endif
+
+ return aRet;
+}