/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: graph.cxx,v $ * * $Revision: 1.12 $ * * last change: $Author: obo $ $Date: 2006-09-17 12:01:20 $ * * The Contents of this file are made available subject to * the terms of GNU Lesser General Public License Version 2.1. * * * GNU Lesser General Public License Version 2.1 * ============================================= * Copyright 2005 by Sun Microsystems, Inc. * 901 San Antonio Road, Palo Alto, CA 94303, USA * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License version 2.1, as published by the Free Software Foundation. * * This library 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 for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_vcl.hxx" #ifndef _SV_IMPGRAPH_HXX #include #endif #ifndef _SV_OUTDEV_HXX #include #endif #ifndef _SV_SVAPP_HXX #include #endif #include #ifndef _COMPHELPER_PROCESSFACTORY_HXX_ #include #endif #ifndef _COM_SUN_STAR_LANG_XMULTISERVICEFACTORY_HPP_ #include #endif #ifndef _COM_SUN_STAR_GRAPHIC_XGRAPHICPROVIDER_HPP_ #include #endif #ifndef _COM_SUN_STAR_LANG_XUNOTUNNEL_HPP_ #include #endif #ifndef _COM_SUN_STAR_LANG_XTYPEPROVIDER_HPP_ #include #endif #ifndef _COM_SUN_STAR_GRAPHIC_XGRAPHIC_HPP_ #include #endif // ----------------------- // - Compression defines - // ----------------------- #define COMPRESS_OWN ('S'|('D'<<8UL)) #define COMPRESS_NONE ( 0UL ) #define RLE_8 ( 1UL ) #define RLE_4 ( 2UL ) #define BITFIELDS ( 3UL ) #define ZCOMPRESS ( COMPRESS_OWN | 0x01000000UL ) /* == 'SD01' (binary) */ using namespace ::com::sun::star; // ----------------------- // - Default-Drawmethode - // ----------------------- static void ImplDrawDefault( OutputDevice* pOutDev, const UniString* pText, Font* pFont, const Bitmap* pBitmap, const Point& rDestPt, const Size& rDestSize ) { USHORT nPixel = (USHORT) pOutDev->PixelToLogic( Size( 1, 1 ) ).Width(); USHORT nPixelWidth = nPixel; Point aPoint( rDestPt.X() + nPixelWidth, rDestPt.Y() + nPixelWidth ); Size aSize( rDestSize.Width() - ( nPixelWidth << 1 ), rDestSize.Height() - ( nPixelWidth << 1 ) ); BOOL bFilled = ( pBitmap != NULL || pFont != NULL ); Rectangle aBorderRect( aPoint, aSize ); pOutDev->Push(); pOutDev->SetFillColor(); // Auf dem Drucker ein schwarzes Rechteck und auf dem Bildschirm eins mit 3D-Effekt if ( pOutDev->GetOutDevType() == OUTDEV_PRINTER ) pOutDev->SetLineColor( COL_BLACK ); else { aBorderRect.Left() += nPixel; aBorderRect.Top() += nPixel; pOutDev->SetLineColor( COL_LIGHTGRAY ); pOutDev->DrawRect( aBorderRect ); aBorderRect.Left() -= nPixel; aBorderRect.Top() -= nPixel; aBorderRect.Right() -= nPixel; aBorderRect.Bottom() -= nPixel; pOutDev->SetLineColor( COL_GRAY ); } pOutDev->DrawRect( aBorderRect ); aPoint.X() += nPixelWidth + 2*nPixel; aPoint.Y() += nPixelWidth + 2*nPixel; aSize.Width() -= 2*nPixelWidth + 4*nPixel; aSize.Height() -= 2*nPixelWidth + 4*nPixel; if( aSize.Width() > 0 && aSize.Height() > 0 && pBitmap && !!*pBitmap ) { Size aBitmapSize( pOutDev->PixelToLogic(pBitmap->GetSizePixel() ) ); if( aSize.Height() > aBitmapSize.Height() && aSize.Width() > aBitmapSize.Width() ) { pOutDev->DrawBitmap( aPoint, *pBitmap ); aPoint.X() += aBitmapSize.Width() + 2*nPixel; aSize.Width() -= aBitmapSize.Width() + 2*nPixel; } } if ( aSize.Width() > 0 && aSize.Height() > 0 && pFont && pText && pText->Len() && !(!pOutDev->IsOutputEnabled() /*&& pOutDev->GetConnectMetaFile() */) ) { MapMode aMapMode( MAP_POINT ); Size aSz = pOutDev->LogicToLogic( Size( 0, 12 ), &aMapMode, NULL ); long nThreshold = aSz.Height() / 2; long nStep = nThreshold / 3; if ( !nStep ) nStep = aSz.Height() - nThreshold; for(;; aSz.Height() -= nStep ) { pFont->SetSize( aSz ); pOutDev->SetFont( *pFont ); long nTextHeight = pOutDev->GetTextHeight(); long nTextWidth = pOutDev->GetTextWidth( *pText ); if ( nTextHeight ) { // Die N"aherung ber"ucksichtigt keine Ungenauigkeiten durch // Wortumbr"uche long nLines = aSize.Height() / nTextHeight; long nWidth = aSize.Width() * nLines; // N"aherung!!! if ( nTextWidth <= nWidth || aSz.Height() <= nThreshold ) { USHORT nStart = 0; USHORT nLen = 0; while( nStart < pText->Len() && pText->GetChar( nStart ) == ' ' ) nStart++; while( nStart+nLen < pText->Len() && pText->GetChar( nStart+nLen ) != ' ' ) nLen++; while( nStart < pText->Len() && nLines-- ) { USHORT nNext = nLen; do { while ( nStart+nNext < pText->Len() && pText->GetChar( nStart+nNext ) == ' ' ) nNext++; while ( nStart+nNext < pText->Len() && pText->GetChar( nStart+nNext ) != ' ' ) nNext++; nTextWidth = pOutDev->GetTextWidth( *pText, nStart, nNext ); if ( nTextWidth > aSize.Width() ) break; nLen = nNext; } while ( nStart+nNext < pText->Len() ); USHORT n = nLen; nTextWidth = pOutDev->GetTextWidth( *pText, nStart, n ); while( nTextWidth > aSize.Width() ) nTextWidth = pOutDev->GetTextWidth( *pText, nStart, --n ); pOutDev->DrawText( aPoint, *pText, nStart, n ); aPoint.Y() += nTextHeight; nStart = sal::static_int_cast(nStart + nLen); nLen = nNext-nLen; while( nStart < pText->Len() && pText->GetChar( nStart ) == ' ' ) { nStart++; nLen--; } } break; } } else break; } } // Falls die Default-Graphik keinen Inhalt hat, // malen wir ein rotes Kreuz if( !bFilled ) { aBorderRect.Left()++; aBorderRect.Top()++; aBorderRect.Right()--; aBorderRect.Bottom()--; pOutDev->SetLineColor( COL_LIGHTRED ); pOutDev->DrawLine( aBorderRect.TopLeft(), aBorderRect.BottomRight() ); pOutDev->DrawLine( aBorderRect.TopRight(), aBorderRect.BottomLeft() ); } pOutDev->Pop(); } // ----------- // - Graphic - // ----------- TYPEINIT1_AUTOFACTORY( Graphic, SvDataCopyStream ); // ------------------------------------------------------------------------ Graphic::Graphic() { mpImpGraphic = new ImpGraphic; } // ------------------------------------------------------------------------ Graphic::Graphic( const Graphic& rGraphic ) : SvDataCopyStream() { if( rGraphic.IsAnimated() ) mpImpGraphic = new ImpGraphic( *rGraphic.mpImpGraphic ); else { mpImpGraphic = rGraphic.mpImpGraphic; mpImpGraphic->mnRefCount++; } } // ------------------------------------------------------------------------ Graphic::Graphic( const Bitmap& rBmp ) { mpImpGraphic = new ImpGraphic( rBmp ); } // ------------------------------------------------------------------------ Graphic::Graphic( const BitmapEx& rBmpEx ) { mpImpGraphic = new ImpGraphic( rBmpEx ); } // ------------------------------------------------------------------------ Graphic::Graphic( const Animation& rAnimation ) { mpImpGraphic = new ImpGraphic( rAnimation ); } // ------------------------------------------------------------------------ Graphic::Graphic( const GDIMetaFile& rMtf ) { mpImpGraphic = new ImpGraphic( rMtf ); } // ------------------------------------------------------------------------ Graphic::Graphic( const ::com::sun::star::uno::Reference< ::com::sun::star::graphic::XGraphic >& rxGraphic ) { uno::Reference< lang::XUnoTunnel > xTunnel( rxGraphic, uno::UNO_QUERY ); uno::Reference< lang::XTypeProvider > xProv( rxGraphic, uno::UNO_QUERY ); const ::Graphic* pGraphic = ( ( xTunnel.is() && xProv.is() ) ? reinterpret_cast< ::Graphic* >( xTunnel->getSomething( xProv->getImplementationId() ) ) : NULL ); if( pGraphic ) { if( pGraphic->IsAnimated() ) mpImpGraphic = new ImpGraphic( *pGraphic->mpImpGraphic ); else { mpImpGraphic = pGraphic->mpImpGraphic; mpImpGraphic->mnRefCount++; } } else mpImpGraphic = new ImpGraphic; } // ------------------------------------------------------------------------ Graphic::~Graphic() { if( mpImpGraphic->mnRefCount == 1UL ) delete mpImpGraphic; else mpImpGraphic->mnRefCount--; } // ------------------------------------------------------------------------ void Graphic::ImplTestRefCount() { if( mpImpGraphic->mnRefCount > 1UL ) { mpImpGraphic->mnRefCount--; mpImpGraphic = new ImpGraphic( *mpImpGraphic ); } } // ------------------------------------------------------------------------ Graphic& Graphic::operator=( const Graphic& rGraphic ) { if( &rGraphic != this ) { if( rGraphic.IsAnimated() ) { if( mpImpGraphic->mnRefCount == 1UL ) delete mpImpGraphic; else mpImpGraphic->mnRefCount--; mpImpGraphic = new ImpGraphic( *rGraphic.mpImpGraphic ); } else { rGraphic.mpImpGraphic->mnRefCount++; if( mpImpGraphic->mnRefCount == 1UL ) delete mpImpGraphic; else mpImpGraphic->mnRefCount--; mpImpGraphic = rGraphic.mpImpGraphic; } } return *this; } // ------------------------------------------------------------------------ BOOL Graphic::operator==( const Graphic& rGraphic ) const { return( *mpImpGraphic == *rGraphic.mpImpGraphic ); } // ------------------------------------------------------------------------ BOOL Graphic::operator!=( const Graphic& rGraphic ) const { return( *mpImpGraphic != *rGraphic.mpImpGraphic ); } // ------------------------------------------------------------------------ BOOL Graphic::operator!() const { return( GRAPHIC_NONE == mpImpGraphic->ImplGetType() ); } // ------------------------------------------------------------------------ void Graphic::Load( SvStream& rIStm ) { rIStm >> *this; } // ------------------------------------------------------------------------ void Graphic::Save( SvStream& rOStm ) { rOStm << *this; } // ------------------------------------------------------------------------ void Graphic::Assign( const SvDataCopyStream& rCopyStream ) { *this = (const Graphic& ) rCopyStream; } // ------------------------------------------------------------------------ void Graphic::Clear() { ImplTestRefCount(); mpImpGraphic->ImplClear(); } // ------------------------------------------------------------------------ GraphicType Graphic::GetType() const { return mpImpGraphic->ImplGetType(); } // ------------------------------------------------------------------------ void Graphic::SetDefaultType() { ImplTestRefCount(); mpImpGraphic->ImplSetDefaultType(); } // ------------------------------------------------------------------------ BOOL Graphic::IsSupportedGraphic() const { return mpImpGraphic->ImplIsSupportedGraphic(); } // ------------------------------------------------------------------------ BOOL Graphic::IsTransparent() const { return mpImpGraphic->ImplIsTransparent(); } // ------------------------------------------------------------------------ BOOL Graphic::IsAlpha() const { return mpImpGraphic->ImplIsAlpha(); } // ------------------------------------------------------------------------ BOOL Graphic::IsAnimated() const { return mpImpGraphic->ImplIsAnimated(); } // ------------------------------------------------------------------------ Bitmap Graphic::GetBitmap() const { return GetBitmap( NULL ); } // ------------------------------------------------------------------------ BitmapEx Graphic::GetBitmapEx() const { return GetBitmapEx( NULL ); } // ------------------------------------------------------------------------ Bitmap Graphic::GetBitmap( const Size* pSizePixel ) const { return mpImpGraphic->ImplGetBitmap( pSizePixel, FALSE ); } // ------------------------------------------------------------------------ BitmapEx Graphic::GetBitmapEx( const Size* pSizePixel ) const { return mpImpGraphic->ImplGetBitmapEx( pSizePixel, FALSE ); } // ------------------------------------------------------------------------ Bitmap Graphic::GetUnlimitedBitmap( const Size* pSizePixel ) const { return mpImpGraphic->ImplGetBitmap( pSizePixel, TRUE ) ; } // ------------------------------------------------------------------------ BitmapEx Graphic::GetUnlimitedBitmapEx( const Size* pSizePixel ) const { return mpImpGraphic->ImplGetBitmapEx( pSizePixel, TRUE ) ; } // ------------------------------------------------------------------------ Animation Graphic::GetAnimation() const { return mpImpGraphic->ImplGetAnimation(); } // ------------------------------------------------------------------------ const GDIMetaFile& Graphic::GetGDIMetaFile() const { return mpImpGraphic->ImplGetGDIMetaFile(); } // ------------------------------------------------------------------------ uno::Reference< graphic::XGraphic > Graphic::GetXGraphic() const { uno::Reference< graphic::XGraphic > xRet; if( GetType() != GRAPHIC_NONE ) { uno::Reference < lang::XMultiServiceFactory > xMSF( ::comphelper::getProcessServiceFactory() ); if( xMSF.is() ) { uno::Reference< graphic::XGraphicProvider > xProv( xMSF->createInstance( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.graphic.GraphicProvider" ) ) ), uno::UNO_QUERY ); if( xProv.is() ) { uno::Sequence< beans::PropertyValue > aLoadProps( 1 ); ::rtl::OUString aURL( RTL_CONSTASCII_USTRINGPARAM( "private:memorygraphic/" ) ); aLoadProps[ 0 ].Name = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "URL" ) ); aLoadProps[ 0 ].Value <<= ( aURL += ::rtl::OUString::valueOf( reinterpret_cast< sal_Int64 >( this ) ) ); xRet = xProv->queryGraphic( aLoadProps ); } } } return xRet; } // ------------------------------------------------------------------------ Size Graphic::GetPrefSize() const { return mpImpGraphic->ImplGetPrefSize(); } // ------------------------------------------------------------------------ void Graphic::SetPrefSize( const Size& rPrefSize ) { ImplTestRefCount(); mpImpGraphic->ImplSetPrefSize( rPrefSize ); } // ------------------------------------------------------------------------ MapMode Graphic::GetPrefMapMode() const { return mpImpGraphic->ImplGetPrefMapMode(); } // ------------------------------------------------------------------------ void Graphic::SetPrefMapMode( const MapMode& rPrefMapMode ) { ImplTestRefCount(); mpImpGraphic->ImplSetPrefMapMode( rPrefMapMode ); } // ------------------------------------------------------------------ Size Graphic::GetSizePixel( const OutputDevice* pRefDevice ) const { Size aRet; if( GRAPHIC_BITMAP == mpImpGraphic->ImplGetType() ) aRet = mpImpGraphic->ImplGetBitmapEx( NULL, FALSE ).GetSizePixel(); else aRet = ( pRefDevice ? pRefDevice : Application::GetDefaultDevice() )->LogicToPixel( GetPrefSize(), GetPrefMapMode() ); return aRet; } // ------------------------------------------------------------------ ULONG Graphic::GetSizeBytes() const { return mpImpGraphic->ImplGetSizeBytes(); } // ------------------------------------------------------------------------ void Graphic::Draw( OutputDevice* pOutDev, const Point& rDestPt ) const { mpImpGraphic->ImplDraw( pOutDev, rDestPt ); } // ------------------------------------------------------------------------ void Graphic::Draw( OutputDevice* pOutDev, const Point& rDestPt, const Size& rDestSz ) const { if( GRAPHIC_DEFAULT == mpImpGraphic->ImplGetType() ) ImplDrawDefault( pOutDev, NULL, NULL, NULL, rDestPt, rDestSz ); else mpImpGraphic->ImplDraw( pOutDev, rDestPt, rDestSz ); } // ------------------------------------------------------------------------ void Graphic::Draw( OutputDevice* pOutDev, const String& rText, Font& rFont, const Bitmap& rBitmap, const Point& rDestPt, const Size& rDestSz ) { ImplDrawDefault( pOutDev, &rText, &rFont, &rBitmap, rDestPt, rDestSz ); } // ------------------------------------------------------------------------ void Graphic::StartAnimation( OutputDevice* pOutDev, const Point& rDestPt, long nExtraData, OutputDevice* pFirstFrameOutDev ) { ImplTestRefCount(); mpImpGraphic->ImplStartAnimation( pOutDev, rDestPt, nExtraData, pFirstFrameOutDev ); } // ------------------------------------------------------------------------ void Graphic::StartAnimation( OutputDevice* pOutDev, const Point& rDestPt, const Size& rDestSz, long nExtraData, OutputDevice* pFirstFrameOutDev ) { ImplTestRefCount(); mpImpGraphic->ImplStartAnimation( pOutDev, rDestPt, rDestSz, nExtraData, pFirstFrameOutDev ); } // ------------------------------------------------------------------------ void Graphic::StopAnimation( OutputDevice* pOutDev, long nExtraData ) { ImplTestRefCount(); mpImpGraphic->ImplStopAnimation( pOutDev, nExtraData ); } // ------------------------------------------------------------------------ void Graphic::SetAnimationNotifyHdl( const Link& rLink ) { mpImpGraphic->ImplSetAnimationNotifyHdl( rLink ); } // ------------------------------------------------------------------------ Link Graphic::GetAnimationNotifyHdl() const { return mpImpGraphic->ImplGetAnimationNotifyHdl(); } // ------------------------------------------------------------------------ ULONG Graphic::GetAnimationLoopCount() const { return mpImpGraphic->ImplGetAnimationLoopCount(); } // ------------------------------------------------------------------------ void Graphic::ResetAnimationLoopCount() { mpImpGraphic->ImplResetAnimationLoopCount(); } // ------------------------------------------------------------------------ List* Graphic::GetAnimationInfoList() const { return mpImpGraphic->ImplGetAnimationInfoList(); } // ------------------------------------------------------------------------ GraphicReader* Graphic::GetContext() { return mpImpGraphic->ImplGetContext(); } // ------------------------------------------------------------------------ void Graphic::SetContext( GraphicReader* pReader ) { mpImpGraphic->ImplSetContext( pReader ); } // ------------------------------------------------------------------------ USHORT Graphic::GetGraphicsCompressMode( SvStream& rIStm ) { const ULONG nPos = rIStm.Tell(); const USHORT nOldFormat = rIStm.GetNumberFormatInt(); UINT32 nTmp32; UINT16 nTmp16; USHORT nCompressMode = COMPRESSMODE_NONE; rIStm.SetNumberFormatInt( NUMBERFORMAT_INT_LITTLEENDIAN ); rIStm >> nTmp32; // is it a swapped graphic with a bitmap? rIStm.SeekRel( (nTmp32 == (UINT32) GRAPHIC_BITMAP ) ? 40 : -4 ); // try to read bitmap id rIStm >> nTmp16; // check id of BitmapFileHeader if( 0x4D42 == nTmp16 ) { // seek to compress field of BitmapInfoHeader rIStm.SeekRel( 28 ); rIStm >> nTmp32; // Compare with our own compressmode if( ZCOMPRESS == nTmp32 ) nCompressMode = COMPRESSMODE_ZBITMAP; } rIStm.SetNumberFormatInt( nOldFormat ); rIStm.Seek( nPos ); return nCompressMode; } // ------------------------------------------------------------------------ void Graphic::SetDocFileName( const String& rName, ULONG nFilePos ) { mpImpGraphic->ImplSetDocFileName( rName, nFilePos ); } // ------------------------------------------------------------------------ const String& Graphic::GetDocFileName() const { return mpImpGraphic->ImplGetDocFileName(); } // ------------------------------------------------------------------------ ULONG Graphic::GetDocFilePos() const { return mpImpGraphic->ImplGetDocFilePos(); } // ------------------------------------------------------------------------ BOOL Graphic::ReadEmbedded( SvStream& rIStream, BOOL bSwap ) { ImplTestRefCount(); return mpImpGraphic->ImplReadEmbedded( rIStream, bSwap ); } // ------------------------------------------------------------------------ BOOL Graphic::WriteEmbedded( SvStream& rOStream ) { ImplTestRefCount(); return mpImpGraphic->ImplWriteEmbedded( rOStream ); } // ------------------------------------------------------------------------ BOOL Graphic::SwapOut() { ImplTestRefCount(); return mpImpGraphic->ImplSwapOut(); } // ------------------------------------------------------------------------ BOOL Graphic::SwapOut( SvStream* pOStream ) { ImplTestRefCount(); return mpImpGraphic->ImplSwapOut( pOStream ); } // ------------------------------------------------------------------------ BOOL Graphic::SwapIn() { ImplTestRefCount(); return mpImpGraphic->ImplSwapIn(); } // ------------------------------------------------------------------------ BOOL Graphic::SwapIn( SvStream* pStrm ) { ImplTestRefCount(); return mpImpGraphic->ImplSwapIn( pStrm ); } // ------------------------------------------------------------------------ BOOL Graphic::IsSwapOut() const { return mpImpGraphic->ImplIsSwapOut(); } // ------------------------------------------------------------------------ void Graphic::SetLink( const GfxLink& rGfxLink ) { ImplTestRefCount(); mpImpGraphic->ImplSetLink( rGfxLink ); } // ------------------------------------------------------------------------ GfxLink Graphic::GetLink() { return mpImpGraphic->ImplGetLink(); } // ------------------------------------------------------------------------ BOOL Graphic::IsLink() const { return mpImpGraphic->ImplIsLink(); } // ------------------------------------------------------------------------ ULONG Graphic::GetChecksum() const { return mpImpGraphic->ImplGetChecksum(); } // ------------------------------------------------------------------------ BOOL Graphic::ExportNative( SvStream& rOStream ) const { return mpImpGraphic->ImplExportNative( rOStream ); } // ------------------------------------------------------------------------ SvStream& operator>>( SvStream& rIStream, Graphic& rGraphic ) { rGraphic.ImplTestRefCount(); return rIStream >> *rGraphic.mpImpGraphic; } // ------------------------------------------------------------------------ SvStream& operator<<( SvStream& rOStream, const Graphic& rGraphic ) { return rOStream << *rGraphic.mpImpGraphic; }