| /************************************************************** |
| * |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, |
| * software distributed under the License is distributed on an |
| * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY |
| * KIND, either express or implied. See the License for the |
| * specific language governing permissions and limitations |
| * under the License. |
| * |
| *************************************************************/ |
| |
| |
| |
| // MARKER(update_precomp.py): autogen include statement, do not remove |
| #include "precompiled_filter.hxx" |
| |
| #include <stdio.h> |
| |
| // |
| #include <vcl/sv.h> |
| #include <vcl/svapp.hxx> |
| #include <vcl/bitmap.hxx> |
| #include <vcl/bitmapex.hxx> |
| #include <vcl/animate.hxx> |
| #include <vcl/gdimtf.hxx> |
| #include <vcl/graph.h> |
| #include <vcl/window.hxx> |
| #include <vcl/graph.hxx> |
| #include <vcl/metaact.hxx> |
| #include <vcl/gdimtf.hxx> |
| #include <vcl/virdev.hxx> |
| #include <vcl/cvtgrf.hxx> |
| #include <vcl/bmpacc.hxx> |
| #include <svtools/fltcall.hxx> |
| #include <tools/urlobj.hxx> |
| #include <tools/tempfile.hxx> |
| #include <osl/process.h> |
| #include <osl/file.hxx> |
| |
| /************************************************************************* |
| |* |
| |* ImpSearchEntry() |
| |* |
| |* Beschreibung Prueft ob im Speicherbereich pSource der nComp Bytes |
| |* gross ist eine Zeichenkette(pDest) mit der l�nge nSize |
| |* liegt. Geprueft wird NON-CASE-SENSITIVE und der Rueck- |
| |* gabewert ist die Adresse an der die Zeichekette gefunden |
| |* wurde oder NULL |
| |* |
| |* Ersterstellung SJ 04.03.98 ( und das an meinem Geburtstag ) |
| |* Letzte Aenderung SJ 04.03.98 |
| |* |
| *************************************************************************/ |
| |
| static sal_uInt8* ImplSearchEntry( sal_uInt8* pSource, sal_uInt8* pDest, sal_uLong nComp, sal_uLong nSize ) |
| { |
| while ( nComp-- >= nSize ) |
| { |
| sal_uLong i; |
| for ( i = 0; i < nSize; i++ ) |
| { |
| if ( ( pSource[i]&~0x20 ) != ( pDest[i]&~0x20 ) ) |
| break; |
| } |
| if ( i == nSize ) |
| return pSource; |
| pSource++; |
| } |
| return NULL; |
| } |
| |
| //-------------------------------------------------------------------------- |
| // SecurityCount is the buffersize of the buffer in which we will parse for a number |
| static long ImplGetNumber( sal_uInt8 **pBuf, int& nSecurityCount ) |
| { |
| sal_Bool bValid = sal_True; |
| sal_Bool bNegative = sal_False; |
| long nRetValue = 0; |
| while ( ( --nSecurityCount ) && ( ( **pBuf == ' ' ) || ( **pBuf == 0x9 ) ) ) |
| (*pBuf)++; |
| sal_uInt8 nByte = **pBuf; |
| while ( nSecurityCount && ( nByte != ' ' ) && ( nByte != 0x9 ) && ( nByte != 0xd ) && ( nByte != 0xa ) ) |
| { |
| switch ( nByte ) |
| { |
| case '.' : |
| // we'll only use the integer format |
| bValid = sal_False; |
| break; |
| case '-' : |
| bNegative = sal_True; |
| break; |
| default : |
| if ( ( nByte < '0' ) || ( nByte > '9' ) ) |
| nSecurityCount = 1; // error parsing the bounding box values |
| else if ( bValid ) |
| { |
| nRetValue *= 10; |
| nRetValue += nByte - '0'; |
| } |
| break; |
| } |
| nSecurityCount--; |
| nByte = *(++(*pBuf)); |
| } |
| if ( bNegative ) |
| nRetValue = -nRetValue; |
| return nRetValue; |
| } |
| |
| //-------------------------------------------------------------------------- |
| |
| static int ImplGetLen( sal_uInt8* pBuf, int nMax ) |
| { |
| int nLen = 0; |
| while( nLen != nMax ) |
| { |
| sal_uInt8 nDat = *pBuf++; |
| if ( nDat == 0x0a || nDat == 0x25 ) |
| break; |
| nLen++; |
| } |
| return nLen; |
| } |
| |
| static void MakeAsMeta(Graphic &rGraphic) |
| { |
| VirtualDevice aVDev; |
| GDIMetaFile aMtf; |
| Bitmap aBmp( rGraphic.GetBitmap() ); |
| Size aSize = aBmp.GetPrefSize(); |
| |
| if( !aSize.Width() || !aSize.Height() ) |
| aSize = Application::GetDefaultDevice()->PixelToLogic( |
| aBmp.GetSizePixel(), MAP_100TH_MM ); |
| else |
| aSize = Application::GetDefaultDevice()->LogicToLogic( aSize, |
| aBmp.GetPrefMapMode(), MAP_100TH_MM ); |
| |
| aVDev.EnableOutput( sal_False ); |
| aMtf.Record( &aVDev ); |
| aVDev.DrawBitmap( Point(), aSize, rGraphic.GetBitmap() ); |
| aMtf.Stop(); |
| aMtf.WindStart(); |
| aMtf.SetPrefMapMode( MAP_100TH_MM ); |
| aMtf.SetPrefSize( aSize ); |
| rGraphic = aMtf; |
| } |
| |
| static oslProcessError runProcessWithPathSearch(const rtl::OUString &rProgName, |
| rtl_uString* pArgs[], sal_uInt32 nArgs, oslProcess *pProcess, |
| oslFileHandle *pIn, oslFileHandle *pOut, oslFileHandle *pErr) |
| { |
| #ifdef WNT |
| /* |
| * ooo#72096 |
| * On Window the underlying SearchPath searches in order of... |
| * The directory from which the application loaded. |
| * The current directory. |
| * The Windows system directory. |
| * The Windows directory. |
| * The directories that are listed in the PATH environment variable. |
| * |
| * Because one of our programs is called "convert" and there is a convert |
| * in the windows system directory, we want to explicitly search the PATH |
| * to avoid picking up on that one if ImageMagick's convert preceeds it in |
| * PATH. |
| * |
| */ |
| rtl::OUString url; |
| rtl::OUString path(reinterpret_cast<const sal_Unicode*>(_wgetenv(L"PATH"))); |
| |
| oslFileError err = osl_searchFileURL(rProgName.pData, path.pData, &url.pData); |
| if (err != osl_File_E_None) |
| return osl_Process_E_NotFound; |
| return osl_executeProcess_WithRedirectedIO(url.pData, |
| pArgs, nArgs, osl_Process_HIDDEN, |
| osl_getCurrentSecurity(), 0, 0, 0, pProcess, pIn, pOut, pErr); |
| #else |
| return osl_executeProcess_WithRedirectedIO(rProgName.pData, |
| pArgs, nArgs, osl_Process_SEARCHPATH | osl_Process_HIDDEN, |
| osl_getCurrentSecurity(), 0, 0, 0, pProcess, pIn, pOut, pErr); |
| #endif |
| } |
| |
| #if defined(WNT) || defined(OS2) |
| # define EXESUFFIX ".exe" |
| #else |
| # define EXESUFFIX "" |
| #endif |
| |
| static bool RenderAsEMF(const sal_uInt8* pBuf, sal_uInt32 nBytesRead, Graphic &rGraphic) |
| { |
| TempFile aTemp; |
| aTemp.EnableKillingFile(); |
| rtl::OUString fileName = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( "pstoedit" EXESUFFIX)); |
| rtl::OUString arg1 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-f")); |
| rtl::OUString arg2 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("emf:-OO")); |
| rtl::OUString arg3 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-")); |
| rtl::OUString output; |
| osl::FileBase::getSystemPathFromFileURL(aTemp.GetName(), output); |
| rtl_uString *args[] = |
| { |
| arg1.pData, arg2.pData, arg3.pData, output.pData |
| }; |
| oslProcess aProcess; |
| oslFileHandle pIn = NULL; |
| oslFileHandle pOut = NULL; |
| oslFileHandle pErr = NULL; |
| oslProcessError eErr = runProcessWithPathSearch(fileName, |
| args, sizeof(args)/sizeof(rtl_uString *), |
| &aProcess, &pIn, &pOut, &pErr); |
| |
| if (eErr!=osl_Process_E_None) |
| return false; |
| |
| bool bRet = false; |
| sal_uInt64 nCount; |
| osl_writeFile(pIn, pBuf, nBytesRead, &nCount); |
| if (pIn) osl_closeFile(pIn); |
| bool bEMFSupported=true; |
| if (pOut) |
| { |
| rtl::ByteSequence seq; |
| if (osl_File_E_None == osl_readLine(pOut, (sal_Sequence **)&seq)) |
| { |
| rtl::OString line( (const sal_Char *) seq.getConstArray(), seq.getLength() ); |
| if (line.indexOf(rtl::OString("Unsupported output format")) == 0) |
| bEMFSupported=false; |
| } |
| osl_closeFile(pOut); |
| } |
| if (pErr) osl_closeFile(pErr); |
| if (nCount == nBytesRead && bEMFSupported) |
| { |
| SvFileStream aFile(output, STREAM_READ); |
| if (GraphicConverter::Import(aFile, rGraphic, CVT_EMF) == ERRCODE_NONE) |
| bRet = true; |
| } |
| osl_joinProcess(aProcess); |
| osl_freeProcessHandle(aProcess); |
| return bRet; |
| } |
| |
| static bool RenderAsPNGThroughHelper(const sal_uInt8* pBuf, sal_uInt32 nBytesRead, |
| Graphic &rGraphic, rtl::OUString &rProgName, rtl_uString *pArgs[], size_t nArgs) |
| { |
| oslProcess aProcess; |
| oslFileHandle pIn = NULL; |
| oslFileHandle pOut = NULL; |
| oslFileHandle pErr = NULL; |
| oslProcessError eErr = runProcessWithPathSearch(rProgName, |
| pArgs, nArgs, |
| &aProcess, &pIn, &pOut, &pErr); |
| if (eErr!=osl_Process_E_None) |
| return false; |
| |
| bool bRet = false; |
| sal_uInt64 nCount; |
| osl_writeFile(pIn, pBuf, nBytesRead, &nCount); |
| if (pIn) osl_closeFile(pIn); |
| if (nCount == nBytesRead) |
| { |
| SvMemoryStream aMemStm; |
| sal_uInt8 aBuf[32000]; |
| oslFileError eFileErr = osl_readFile(pOut, aBuf, 32000, &nCount); |
| while (eFileErr == osl_File_E_None && nCount) |
| { |
| aMemStm.Write(aBuf, sal::static_int_cast< sal_Size >(nCount)); |
| eFileErr = osl_readFile(pOut, aBuf, 32000, &nCount); |
| } |
| |
| aMemStm.Seek(0); |
| if ( |
| aMemStm.GetEndOfData() && |
| GraphicConverter::Import(aMemStm, rGraphic, CVT_PNG) == ERRCODE_NONE |
| ) |
| { |
| MakeAsMeta(rGraphic); |
| bRet = true; |
| } |
| } |
| if (pOut) osl_closeFile(pOut); |
| if (pErr) osl_closeFile(pErr); |
| osl_joinProcess(aProcess); |
| osl_freeProcessHandle(aProcess); |
| return bRet; |
| } |
| |
| static bool RenderAsPNGThroughConvert(const sal_uInt8* pBuf, sal_uInt32 nBytesRead, |
| Graphic &rGraphic) |
| { |
| rtl::OUString fileName = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( "convert" EXESUFFIX)); |
| // density in pixel/inch |
| rtl::OUString arg1 = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-density")); |
| // since the preview is also used for PDF-Export & printing on non-PS-printers, |
| // use some better quality - 300x300 should allow some resizing as well |
| rtl::OUString arg2 = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("300x300")); |
| // read eps from STDIN |
| rtl::OUString arg3 = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("eps:-")); |
| // write png to STDOUT |
| rtl::OUString arg4 = rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("png:-")); |
| rtl_uString *args[] = |
| { |
| arg1.pData, arg2.pData, arg3.pData, arg4.pData |
| }; |
| return RenderAsPNGThroughHelper(pBuf, nBytesRead, rGraphic, fileName, args, |
| sizeof(args)/sizeof(rtl_uString *)); |
| } |
| |
| static bool RenderAsPNGThroughGS(const sal_uInt8* pBuf, sal_uInt32 nBytesRead, |
| Graphic &rGraphic) |
| { |
| #ifdef WNT |
| rtl::OUString fileName = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( "gswin32c" EXESUFFIX)); |
| #else |
| rtl::OUString fileName = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM( "gs" EXESUFFIX)); |
| #endif |
| rtl::OUString arg1 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-q")); |
| rtl::OUString arg2 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-dBATCH")); |
| rtl::OUString arg3 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-dNOPAUSE")); |
| rtl::OUString arg4 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-dPARANOIDSAFER")); |
| rtl::OUString arg5 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-dEPSCrop")); |
| rtl::OUString arg6 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-dTextAlphaBits=4")); |
| rtl::OUString arg7 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-dGraphicsAlphaBits=4")); |
| rtl::OUString arg8 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-r300x300")); |
| rtl::OUString arg9 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-sDEVICE=png256")); |
| rtl::OUString arg10 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-sOutputFile=-")); |
| rtl::OUString arg11 = |
| rtl::OUString(RTL_CONSTASCII_USTRINGPARAM("-")); |
| rtl_uString *args[] = |
| { |
| arg1.pData, arg2.pData, arg3.pData, arg4.pData, arg5.pData, |
| arg6.pData, arg7.pData, arg8.pData, arg9.pData, arg10.pData, |
| arg11.pData |
| }; |
| return RenderAsPNGThroughHelper(pBuf, nBytesRead, rGraphic, fileName, args, |
| sizeof(args)/sizeof(rtl_uString *)); |
| } |
| |
| static bool RenderAsPNG(const sal_uInt8* pBuf, sal_uInt32 nBytesRead, Graphic &rGraphic) |
| { |
| if (RenderAsPNGThroughConvert(pBuf, nBytesRead, rGraphic)) |
| return true; |
| else |
| return RenderAsPNGThroughGS(pBuf, nBytesRead, rGraphic); |
| } |
| |
| // this method adds a replacement action containing the original wmf or tiff replacement, |
| // so the original eps can be written when storing to ODF. |
| void CreateMtfReplacementAction( GDIMetaFile& rMtf, SvStream& rStrm, sal_uInt32 nOrigPos, sal_uInt32 nPSSize, |
| sal_uInt32 nPosWMF, sal_uInt32 nSizeWMF, sal_uInt32 nPosTIFF, sal_uInt32 nSizeTIFF ) |
| { |
| ByteString aComment( (const sal_Char*)"EPSReplacementGraphic" ); |
| if ( nSizeWMF || nSizeTIFF ) |
| { |
| SvMemoryStream aReplacement( nSizeWMF + nSizeTIFF + 28 ); |
| sal_uInt32 nMagic = 0xc6d3d0c5; |
| sal_uInt32 nPPos = 28 + nSizeWMF + nSizeTIFF; |
| sal_uInt32 nWPos = nSizeWMF ? 28 : 0; |
| sal_uInt32 nTPos = nSizeTIFF ? 28 + nSizeWMF : 0; |
| |
| aReplacement << nMagic << nPPos << nPSSize |
| << nWPos << nSizeWMF |
| << nTPos << nSizeTIFF; |
| if ( nSizeWMF ) |
| { |
| sal_uInt8* pBuf = new sal_uInt8[ nSizeWMF ]; |
| rStrm.Seek( nOrigPos + nPosWMF ); |
| rStrm.Read( pBuf, nSizeWMF ); |
| aReplacement.Write( pBuf, nSizeWMF ); |
| delete[] pBuf; |
| } |
| if ( nSizeTIFF ) |
| { |
| sal_uInt8* pBuf = new sal_uInt8[ nSizeTIFF ]; |
| rStrm.Seek( nOrigPos + nPosTIFF ); |
| rStrm.Read( pBuf, nSizeTIFF ); |
| aReplacement.Write( pBuf, nSizeTIFF ); |
| delete[] pBuf; |
| } |
| rMtf.AddAction( (MetaAction*)( new MetaCommentAction( aComment, 0, (const sal_uInt8*)aReplacement.GetData(), aReplacement.Tell() ) ) ); |
| } |
| else |
| rMtf.AddAction( (MetaAction*)( new MetaCommentAction( aComment, 0, NULL, 0 ) ) ); |
| } |
| |
| //there is no preview -> make a red box |
| void MakePreview(sal_uInt8* pBuf, sal_uInt32 nBytesRead, |
| long nWidth, long nHeight, Graphic &rGraphic) |
| { |
| GDIMetaFile aMtf; |
| VirtualDevice aVDev; |
| Font aFont; |
| |
| aVDev.EnableOutput( sal_False ); |
| aMtf.Record( &aVDev ); |
| aVDev.SetLineColor( Color( COL_RED ) ); |
| aVDev.SetFillColor(); |
| |
| aFont.SetColor( COL_LIGHTRED ); |
| // aFont.SetSize( Size( 0, 32 ) ); |
| |
| aVDev.Push( PUSH_FONT ); |
| aVDev.SetFont( aFont ); |
| |
| Rectangle aRect( Point( 1, 1 ), Size( nWidth - 2, nHeight - 2 ) ); |
| aVDev.DrawRect( aRect ); |
| |
| String aString; |
| int nLen; |
| sal_uInt8* pDest = ImplSearchEntry( pBuf, (sal_uInt8*)"%%Title:", nBytesRead - 32, 8 ); |
| if ( pDest ) |
| { |
| pDest += 8; |
| if ( *pDest == ' ' ) |
| pDest++; |
| nLen = ImplGetLen( pDest, 32 ); |
| sal_uInt8 aOldValue(pDest[ nLen ]); pDest[ nLen ] = 0; |
| if ( strcmp( (const char*)pDest, "none" ) != 0 ) |
| { |
| aString.AppendAscii( " Title:" ); |
| aString.AppendAscii( (char*)pDest ); |
| aString.AppendAscii( "\n" ); |
| } |
| pDest[ nLen ] = aOldValue; |
| } |
| pDest = ImplSearchEntry( pBuf, (sal_uInt8*)"%%Creator:", nBytesRead - 32, 10 ); |
| if ( pDest ) |
| { |
| pDest += 10; |
| if ( *pDest == ' ' ) |
| pDest++; |
| nLen = ImplGetLen( pDest, 32 ); |
| sal_uInt8 aOldValue(pDest[ nLen ]); pDest[ nLen ] = 0; |
| aString.AppendAscii( " Creator:" ); |
| aString.AppendAscii( (char*)pDest ); |
| aString.AppendAscii( "\n" ); |
| pDest[ nLen ] = aOldValue; |
| } |
| pDest = ImplSearchEntry( pBuf, (sal_uInt8*)"%%CreationDate:", nBytesRead - 32, 15 ); |
| if ( pDest ) |
| { |
| pDest += 15; |
| if ( *pDest == ' ' ) |
| pDest++; |
| nLen = ImplGetLen( pDest, 32 ); |
| sal_uInt8 aOldValue(pDest[ nLen ]); pDest[ nLen ] = 0; |
| if ( strcmp( (const char*)pDest, "none" ) != 0 ) |
| { |
| aString.AppendAscii( " CreationDate:" ); |
| aString.AppendAscii( (char*)pDest ); |
| aString.AppendAscii( "\n" ); |
| } |
| pDest[ nLen ] = aOldValue; |
| } |
| pDest = ImplSearchEntry( pBuf, (sal_uInt8*)"%%LanguageLevel:", nBytesRead - 4, 16 ); |
| if ( pDest ) |
| { |
| pDest += 16; |
| int nCount = 4; |
| long nNumber = ImplGetNumber( &pDest, nCount ); |
| if ( nCount && ( (sal_uInt32)nNumber < 10 ) ) |
| { |
| aString.AppendAscii( " LanguageLevel:" ); |
| aString.Append( UniString::CreateFromInt32( nNumber ) ); |
| } |
| } |
| aVDev.DrawText( aRect, aString, TEXT_DRAW_CLIP | TEXT_DRAW_MULTILINE ); |
| aVDev.Pop(); |
| aMtf.Stop(); |
| aMtf.WindStart(); |
| aMtf.SetPrefMapMode( MAP_POINT ); |
| aMtf.SetPrefSize( Size( nWidth, nHeight ) ); |
| rGraphic = aMtf; |
| } |
| |
| |
| //================== GraphicImport - die exportierte Funktion ================ |
| |
| #ifdef WNT |
| extern "C" sal_Bool _cdecl GraphicImport(SvStream & rStream, Graphic & rGraphic, FilterConfigItem*, sal_Bool) |
| #else |
| extern "C" sal_Bool GraphicImport(SvStream & rStream, Graphic & rGraphic, FilterConfigItem*, sal_Bool) |
| #endif |
| { |
| if ( rStream.GetError() ) |
| return sal_False; |
| |
| Graphic aGraphic; |
| sal_Bool bRetValue = sal_False; |
| sal_Bool bHasPreview = sal_False; |
| sal_Bool bGraphicLinkCreated = sal_False; |
| sal_uInt32 nSignature, nPSStreamPos, nPSSize; |
| sal_uInt32 nSizeWMF = 0; |
| sal_uInt32 nPosWMF = 0; |
| sal_uInt32 nSizeTIFF = 0; |
| sal_uInt32 nPosTIFF = 0; |
| sal_uInt32 nOrigPos = nPSStreamPos = rStream.Tell(); |
| sal_uInt16 nOldFormat = rStream.GetNumberFormatInt(); |
| rStream.SetNumberFormatInt( NUMBERFORMAT_INT_LITTLEENDIAN ); |
| rStream >> nSignature; |
| if ( nSignature == 0xc6d3d0c5 ) |
| { |
| rStream >> nPSStreamPos >> nPSSize >> nPosWMF >> nSizeWMF; |
| |
| // first we try to get the metafile grafix |
| |
| if ( nSizeWMF ) |
| { |
| if ( nPosWMF != 0 ) |
| { |
| rStream.Seek( nOrigPos + nPosWMF ); |
| if ( GraphicConverter::Import( rStream, aGraphic, CVT_WMF ) == ERRCODE_NONE ) |
| bHasPreview = bRetValue = sal_True; |
| } |
| } |
| else |
| { |
| rStream >> nPosTIFF >> nSizeTIFF; |
| |
| // else we have to get the tiff grafix |
| |
| if ( nPosTIFF && nSizeTIFF ) |
| { |
| rStream.Seek( nOrigPos + nPosTIFF ); |
| if ( GraphicConverter::Import( rStream, aGraphic, CVT_TIF ) == ERRCODE_NONE ) |
| { |
| MakeAsMeta(aGraphic); |
| rStream.Seek( nOrigPos + nPosTIFF ); |
| bHasPreview = bRetValue = sal_True; |
| } |
| } |
| } |
| } |
| else |
| { |
| nPSStreamPos = nOrigPos; // no preview available _>so we must get the size manually |
| nPSSize = rStream.Seek( STREAM_SEEK_TO_END ) - nOrigPos; |
| } |
| sal_uInt8* pHeader = new sal_uInt8[ 22 ]; |
| rStream.Seek( nPSStreamPos ); |
| rStream.Read( pHeader, 22 ); // check PostScript header |
| if ( ImplSearchEntry( pHeader, (sal_uInt8*)"%!PS-Adobe", 10, 10 ) && |
| ImplSearchEntry( &pHeader[ 15 ], (sal_uInt8*)"EPS", 3, 3 ) ) |
| { |
| rStream.Seek( nPSStreamPos ); |
| sal_uInt8* pBuf = new sal_uInt8[ nPSSize ]; |
| if ( pBuf ) |
| { |
| sal_uInt32 nBufStartPos = rStream.Tell(); |
| sal_uInt32 nBytesRead = rStream.Read( pBuf, nPSSize ); |
| if ( nBytesRead == nPSSize ) |
| { |
| int nSecurityCount = 32; |
| if ( !bHasPreview ) // if there is no tiff/wmf preview, we will parse for an preview in the eps prolog |
| { |
| sal_uInt8* pDest = ImplSearchEntry( pBuf, (sal_uInt8*)"%%BeginPreview:", nBytesRead - 32, 15 ); |
| if ( pDest ) |
| { |
| pDest += 15; |
| long nWidth = ImplGetNumber( &pDest, nSecurityCount ); |
| long nHeight = ImplGetNumber( &pDest, nSecurityCount ); |
| long nBitDepth = ImplGetNumber( &pDest, nSecurityCount ); |
| long nScanLines = ImplGetNumber( &pDest, nSecurityCount ); |
| pDest = ImplSearchEntry( pDest, (sal_uInt8*)"%", 16, 1 ); // go to the first Scanline |
| if ( nSecurityCount && pDest && nWidth && nHeight && ( ( nBitDepth == 1 ) || ( nBitDepth == 8 ) ) && nScanLines ) |
| { |
| rStream.Seek( nBufStartPos + ( pDest - pBuf ) ); |
| |
| Bitmap aBitmap( Size( nWidth, nHeight ), 1 ); |
| BitmapWriteAccess* pAcc = aBitmap.AcquireWriteAccess(); |
| if ( pAcc ) |
| { |
| int nBitsLeft; |
| sal_Bool bIsValid = sal_True; |
| sal_uInt8 nDat = 0; |
| char nByte; |
| for ( long y = 0; bIsValid && ( y < nHeight ); y++ ) |
| { |
| nBitsLeft = 0; |
| for ( long x = 0; x < nWidth; x++ ) |
| { |
| if ( --nBitsLeft < 0 ) |
| { |
| while ( bIsValid && ( nBitsLeft != 7 ) ) |
| { |
| rStream >> nByte; |
| switch ( nByte ) |
| { |
| case 0x0a : |
| if ( --nScanLines < 0 ) |
| bIsValid = sal_False; |
| case 0x09 : |
| case 0x0d : |
| case 0x20 : |
| case 0x25 : |
| break; |
| default: |
| { |
| if ( nByte >= '0' ) |
| { |
| if ( nByte > '9' ) |
| { |
| nByte &=~0x20; // case none sensitive for hexadezimal values |
| nByte -= ( 'A' - 10 ); |
| if ( nByte > 15 ) |
| bIsValid = sal_False; |
| } |
| else |
| nByte -= '0'; |
| nBitsLeft += 4; |
| nDat <<= 4; |
| nDat |= ( nByte ^ 0xf ); // in epsi a zero bit represents white color |
| } |
| else |
| bIsValid = sal_False; |
| } |
| break; |
| } |
| } |
| } |
| if ( nBitDepth == 1 ) |
| pAcc->SetPixelIndex( y, x, static_cast<sal_uInt8>(nDat >> nBitsLeft) & 1 ); |
| else |
| { |
| pAcc->SetPixelIndex( y, x, nDat ? 1 : 0 ); // nBitDepth == 8 |
| nBitsLeft = 0; |
| } |
| } |
| } |
| if ( bIsValid ) |
| { |
| VirtualDevice aVDev; |
| GDIMetaFile aMtf; |
| Size aSize; |
| aVDev.EnableOutput( sal_False ); |
| aMtf.Record( &aVDev ); |
| aSize = aBitmap.GetPrefSize(); |
| if( !aSize.Width() || !aSize.Height() ) |
| aSize = Application::GetDefaultDevice()->PixelToLogic( aBitmap.GetSizePixel(), MAP_100TH_MM ); |
| else |
| aSize = Application::GetDefaultDevice()->LogicToLogic( aSize, aBitmap.GetPrefMapMode(), MAP_100TH_MM ); |
| aVDev.DrawBitmap( Point(), aSize, aBitmap ); |
| aMtf.Stop(); |
| aMtf.WindStart(); |
| aMtf.SetPrefMapMode( MAP_100TH_MM ); |
| aMtf.SetPrefSize( aSize ); |
| aGraphic = aMtf; |
| bHasPreview = bRetValue = sal_True; |
| } |
| aBitmap.ReleaseAccess( pAcc ); |
| } |
| } |
| } |
| } |
| |
| sal_uInt8* pDest = ImplSearchEntry( pBuf, (sal_uInt8*)"%%BoundingBox:", nBytesRead, 14 ); |
| if ( pDest ) |
| { |
| nSecurityCount = 100; |
| long nNumb[4]; |
| nNumb[0] = nNumb[1] = nNumb[2] = nNumb[3] = 0; |
| pDest += 14; |
| for ( int i = 0; ( i < 4 ) && nSecurityCount; i++ ) |
| { |
| nNumb[ i ] = ImplGetNumber( &pDest, nSecurityCount ); |
| } |
| if ( nSecurityCount) |
| { |
| bGraphicLinkCreated = sal_True; |
| GfxLink aGfxLink( pBuf, nPSSize, GFX_LINK_TYPE_EPS_BUFFER, sal_True ) ; |
| GDIMetaFile aMtf; |
| |
| long nWidth = nNumb[2] - nNumb[0] + 1; |
| long nHeight = nNumb[3] - nNumb[1] + 1; |
| |
| // if there is no preview -> try with gs to make one |
| if( !bHasPreview ) |
| { |
| bHasPreview = RenderAsEMF(pBuf, nBytesRead, aGraphic); |
| if (!bHasPreview) |
| bHasPreview = RenderAsPNG(pBuf, nBytesRead, aGraphic); |
| } |
| |
| // if there is no preview -> make a red box |
| if( !bHasPreview ) |
| { |
| MakePreview(pBuf, nBytesRead, nWidth, nHeight, |
| aGraphic); |
| } |
| |
| aMtf.AddAction( (MetaAction*)( new MetaEPSAction( Point(), Size( nWidth, nHeight ), |
| aGfxLink, aGraphic.GetGDIMetaFile() ) ) ); |
| CreateMtfReplacementAction( aMtf, rStream, nOrigPos, nPSSize, nPosWMF, nSizeWMF, nPosTIFF, nSizeTIFF ); |
| aMtf.WindStart(); |
| aMtf.SetPrefMapMode( MAP_POINT ); |
| aMtf.SetPrefSize( Size( nWidth, nHeight ) ); |
| rGraphic = aMtf; |
| bRetValue = sal_True; |
| } |
| } |
| } |
| } |
| if ( !bGraphicLinkCreated ) |
| delete[] pBuf; |
| } |
| delete[] pHeader; |
| rStream.SetNumberFormatInt(nOldFormat); |
| rStream.Seek( nOrigPos ); |
| return ( bRetValue ); |
| } |
| |