/************************************************************************* * * OpenOffice.org - a multi-platform office productivity suite * * $RCSfile: grfmgr.cxx,v $ * * $Revision: 1.32 $ * * last change: $Author: obo $ $Date: 2006-09-17 15:54:18 $ * * 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_goodies.hxx" #define ENABLE_BYTESTRING_STREAM_OPERATORS #include #include #include #include #include #include #include #include #include #include #include #include #include "grfmgr.hxx" // ----------- // - Defines - // ----------- #define WATERMARK_LUM_OFFSET 50 #define WATERMARK_CON_OFFSET -70 // ----------- // - statics - // ----------- GraphicManager* GraphicObject::mpGlobalMgr = NULL; // --------------------- // - GrfDirectCacheObj - // --------------------- struct GrfSimpleCacheObj { Graphic maGraphic; GraphicAttr maAttr; GrfSimpleCacheObj( const Graphic& rGraphic, const GraphicAttr& rAttr ) : maGraphic( rGraphic ), maAttr( rAttr ) {} }; // ----------------- // - GraphicObject - // ----------------- TYPEINIT1_AUTOFACTORY( GraphicObject, SvDataCopyStream ); // ----------------------------------------------------------------------------- GraphicObject::GraphicObject( const GraphicManager* pMgr ) : mpLink ( NULL ), mpUserData ( NULL ) { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr ); } // ----------------------------------------------------------------------------- GraphicObject::GraphicObject( const Graphic& rGraphic, const GraphicManager* pMgr ) : maGraphic ( rGraphic ), mpLink ( NULL ), mpUserData ( NULL ) { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr ); } // ----------------------------------------------------------------------------- GraphicObject::GraphicObject( const Graphic& rGraphic, const String& rLink, const GraphicManager* pMgr ) : maGraphic ( rGraphic ), mpLink ( rLink.Len() ? ( new String( rLink ) ) : NULL ), mpUserData ( NULL ) { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr ); } // ----------------------------------------------------------------------------- GraphicObject::GraphicObject( const GraphicObject& rGraphicObj, const GraphicManager* pMgr ) : SvDataCopyStream(), maGraphic ( rGraphicObj.GetGraphic() ), maAttr ( rGraphicObj.maAttr ), mpLink ( rGraphicObj.mpLink ? ( new String( *rGraphicObj.mpLink ) ) : NULL ), mpUserData ( rGraphicObj.mpUserData ? ( new String( *rGraphicObj.mpUserData ) ) : NULL ) { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr ); } // ----------------------------------------------------------------------------- GraphicObject::GraphicObject( const ByteString& rUniqueID, const GraphicManager* pMgr ) : mpLink ( NULL ), mpUserData ( NULL ) { ImplConstruct(); // assign default properties ImplAssignGraphicData(); ImplSetGraphicManager( pMgr, &rUniqueID ); // update properties ImplAssignGraphicData(); } // ----------------------------------------------------------------------------- GraphicObject::~GraphicObject() { if( mpMgr ) { mpMgr->ImplUnregisterObj( *this ); if( ( mpMgr == mpGlobalMgr ) && !mpGlobalMgr->ImplHasObjects() ) delete mpGlobalMgr, mpGlobalMgr = NULL; } delete mpSwapOutTimer; delete mpSwapStreamHdl; delete mpLink; delete mpUserData; delete mpSimpleCache; } // ----------------------------------------------------------------------------- void GraphicObject::ImplConstruct() { mpMgr = NULL; mpSwapStreamHdl = NULL; mpSwapOutTimer = NULL; mpSimpleCache = NULL; mnAnimationLoopCount = 0; mbAutoSwapped = FALSE; mbIsInSwapIn = FALSE; mbIsInSwapOut = FALSE; } // ----------------------------------------------------------------------------- void GraphicObject::ImplAssignGraphicData() { maPrefSize = maGraphic.GetPrefSize(); maPrefMapMode = maGraphic.GetPrefMapMode(); mnSizeBytes = maGraphic.GetSizeBytes(); meType = maGraphic.GetType(); mbTransparent = maGraphic.IsTransparent(); mbAlpha = maGraphic.IsAlpha(); mbAnimated = maGraphic.IsAnimated(); mnAnimationLoopCount = ( mbAnimated ? maGraphic.GetAnimationLoopCount() : 0 ); if( maGraphic.GetType() == GRAPHIC_GDIMETAFILE ) { const GDIMetaFile& rMtf = GetGraphic().GetGDIMetaFile(); mbEPS = ( rMtf.GetActionCount() == 1 ) && ( META_EPS_ACTION == rMtf.GetAction( 0 )->GetType() ); } else mbEPS = FALSE; } // ----------------------------------------------------------------------------- void GraphicObject::ImplSetGraphicManager( const GraphicManager* pMgr, const ByteString* pID ) { if( !mpMgr || ( pMgr != mpMgr ) ) { if( !pMgr && mpMgr && ( mpMgr == mpGlobalMgr ) ) return; else { if( mpMgr ) { mpMgr->ImplUnregisterObj( *this ); if( ( mpMgr == mpGlobalMgr ) && !mpGlobalMgr->ImplHasObjects() ) delete mpGlobalMgr, mpGlobalMgr = NULL; } if( !pMgr ) { if( !mpGlobalMgr ) { SvtCacheOptions aCacheOptions; mpGlobalMgr = new GraphicManager( aCacheOptions.GetGraphicManagerTotalCacheSize(), aCacheOptions.GetGraphicManagerObjectCacheSize() ); mpGlobalMgr->SetCacheTimeout( aCacheOptions.GetGraphicManagerObjectReleaseTime() ); } mpMgr = mpGlobalMgr; } else mpMgr = (GraphicManager*) pMgr; mpMgr->ImplRegisterObj( *this, maGraphic, pID ); } } } // ----------------------------------------------------------------------------- void GraphicObject::ImplAutoSwapIn() { if( IsSwappedOut() ) { if( mpMgr && mpMgr->ImplFillSwappedGraphicObject( *this, maGraphic ) ) mbAutoSwapped = FALSE; else { mbIsInSwapIn = TRUE; if( maGraphic.SwapIn() ) mbAutoSwapped = FALSE; else { SvStream* pStream = GetSwapStream(); if( GRFMGR_AUTOSWAPSTREAM_NONE != pStream ) { if( GRFMGR_AUTOSWAPSTREAM_LINK == pStream ) { if( HasLink() ) { String aURLStr; if( ::utl::LocalFileHelper::ConvertPhysicalNameToURL( GetLink(), aURLStr ) ) { SvStream* pIStm = ::utl::UcbStreamHelper::CreateStream( aURLStr, STREAM_READ ); if( pIStm ) { (*pIStm) >> maGraphic; mbAutoSwapped = ( maGraphic.GetType() != GRAPHIC_NONE ); delete pIStm; } } } } else if( GRFMGR_AUTOSWAPSTREAM_TEMP == pStream ) mbAutoSwapped = !maGraphic.SwapIn(); else if( GRFMGR_AUTOSWAPSTREAM_LOADED == pStream ) mbAutoSwapped = maGraphic.IsSwapOut(); else { mbAutoSwapped = !maGraphic.SwapIn( pStream ); delete pStream; } } else { DBG_ASSERT( ( GRAPHIC_NONE == meType ) || ( GRAPHIC_DEFAULT == meType ), "GraphicObject::ImplAutoSwapIn: could not get stream to swap in graphic! (=>KA)" ); } } mbIsInSwapIn = FALSE; if( !mbAutoSwapped && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedIn( *this ); } } } // ----------------------------------------------------------------------------- BOOL GraphicObject::ImplGetCropParams( OutputDevice* pOut, Point& rPt, Size& rSz, const GraphicAttr* pAttr, PolyPolygon& rClipPolyPoly, BOOL& bRectClipRegion ) const { BOOL bRet = FALSE; if( GetType() != GRAPHIC_NONE ) { Polygon aClipPoly( Rectangle( rPt, rSz ) ); const USHORT nRot10 = pAttr->GetRotation() % 3600; const Point aOldOrigin( rPt ); // --> OD 2005-09-30 #i54875# - It's not needed to get the graphic again. // const Graphic& rGraphic = GetGraphic(); // <-- const MapMode aMap100( MAP_100TH_MM ); Size aSize100; long nTotalWidth, nTotalHeight; long nNewLeft, nNewTop, nNewRight, nNewBottom; double fScale; if( nRot10 ) { aClipPoly.Rotate( rPt, nRot10 ); bRectClipRegion = FALSE; } else bRectClipRegion = TRUE; rClipPolyPoly = aClipPoly; // --> OD 2005-09-30 #i54875# - directly access member to // get and . // if( rGraphic.GetPrefMapMode() == MAP_PIXEL ) // aSize100 = Application::GetDefaultDevice()->PixelToLogic( rGraphic.GetPrefSize(), aMap100 ); // else // aSize100 = pOut->LogicToLogic( rGraphic.GetPrefSize(), rGraphic.GetPrefMapMode(), aMap100 ); if( maGraphic.GetPrefMapMode() == MAP_PIXEL ) aSize100 = Application::GetDefaultDevice()->PixelToLogic( maGraphic.GetPrefSize(), aMap100 ); else aSize100 = pOut->LogicToLogic( maGraphic.GetPrefSize(), maGraphic.GetPrefMapMode(), aMap100 ); // <-- nTotalWidth = aSize100.Width() - pAttr->GetLeftCrop() - pAttr->GetRightCrop(); nTotalHeight = aSize100.Height() - pAttr->GetTopCrop() - pAttr->GetBottomCrop(); if( aSize100.Width() > 0 && aSize100.Height() > 0 && nTotalWidth > 0 && nTotalHeight > 0 ) { fScale = (double) aSize100.Width() / nTotalWidth; nNewLeft = -FRound( ( ( pAttr->GetMirrorFlags() & BMP_MIRROR_HORZ ) ? pAttr->GetRightCrop() : pAttr->GetLeftCrop() ) * fScale ); nNewRight = nNewLeft + FRound( aSize100.Width() * fScale ) - 1; fScale = (double) rSz.Width() / aSize100.Width(); rPt.X() += FRound( nNewLeft * fScale ); rSz.Width() = FRound( ( nNewRight - nNewLeft + 1 ) * fScale ); fScale = (double) aSize100.Height() / nTotalHeight; nNewTop = -FRound( ( ( pAttr->GetMirrorFlags() & BMP_MIRROR_VERT ) ? pAttr->GetBottomCrop() : pAttr->GetTopCrop() ) * fScale ); nNewBottom = nNewTop + FRound( aSize100.Height() * fScale ) - 1; fScale = (double) rSz.Height() / aSize100.Height(); rPt.Y() += FRound( nNewTop * fScale ); rSz.Height() = FRound( ( nNewBottom - nNewTop + 1 ) * fScale ); if( nRot10 ) { Polygon aOriginPoly( 1 ); aOriginPoly[ 0 ] = rPt; aOriginPoly.Rotate( aOldOrigin, nRot10 ); rPt = aOriginPoly[ 0 ]; } bRet = TRUE; } } return bRet; } // ----------------------------------------------------------------------------- GraphicObject& GraphicObject::operator=( const GraphicObject& rGraphicObj ) { if( &rGraphicObj != this ) { mpMgr->ImplUnregisterObj( *this ); delete mpSwapStreamHdl, mpSwapStreamHdl = NULL; delete mpSimpleCache, mpSimpleCache = NULL; delete mpLink; delete mpUserData; maGraphic = rGraphicObj.GetGraphic(); maAttr = rGraphicObj.maAttr; mpLink = rGraphicObj.mpLink ? new String( *rGraphicObj.mpLink ) : NULL; mpUserData = rGraphicObj.mpUserData ? new String( *rGraphicObj.mpUserData ) : NULL; ImplAssignGraphicData(); mbAutoSwapped = FALSE; mpMgr = rGraphicObj.mpMgr; mpMgr->ImplRegisterObj( *this, maGraphic, NULL ); } return *this; } // ----------------------------------------------------------------------------- BOOL GraphicObject::operator==( const GraphicObject& rGraphicObj ) const { return( ( rGraphicObj.maGraphic == maGraphic ) && ( rGraphicObj.maAttr == maAttr ) && ( rGraphicObj.GetLink() == GetLink() ) ); } // ------------------------------------------------------------------------ void GraphicObject::Load( SvStream& rIStm ) { rIStm >> *this; } // ------------------------------------------------------------------------ void GraphicObject::Save( SvStream& rOStm ) { rOStm << *this; } // ------------------------------------------------------------------------ void GraphicObject::Assign( const SvDataCopyStream& rCopyStream ) { *this = (const GraphicObject& ) rCopyStream; } // ----------------------------------------------------------------------------- ByteString GraphicObject::GetUniqueID() const { ByteString aRet; if( mpMgr ) aRet = mpMgr->ImplGetUniqueID( *this ); return aRet; } // ----------------------------------------------------------------------------- ULONG GraphicObject::GetChecksum() const { return( ( maGraphic.IsSupportedGraphic() && !maGraphic.IsSwapOut() ) ? maGraphic.GetChecksum() : 0 ); } // ----------------------------------------------------------------------------- SvStream* GraphicObject::GetSwapStream() const { return( HasSwapStreamHdl() ? (SvStream*) mpSwapStreamHdl->Call( (void*) this ) : GRFMGR_AUTOSWAPSTREAM_NONE ); } // ----------------------------------------------------------------------------- // !!! to be removed ULONG GraphicObject::GetReleaseFromCache() const { return 0; } // ----------------------------------------------------------------------------- void GraphicObject::SetAttr( const GraphicAttr& rAttr ) { maAttr = rAttr; if( mpSimpleCache && ( mpSimpleCache->maAttr != rAttr ) ) delete mpSimpleCache, mpSimpleCache = NULL; } // ----------------------------------------------------------------------------- void GraphicObject::SetLink() { if( mpLink ) delete mpLink, mpLink = NULL; } // ----------------------------------------------------------------------------- void GraphicObject::SetLink( const String& rLink ) { delete mpLink, mpLink = new String( rLink ); } // ----------------------------------------------------------------------------- String GraphicObject::GetLink() const { if( mpLink ) return *mpLink; else return String(); } // ----------------------------------------------------------------------------- void GraphicObject::SetUserData() { if( mpUserData ) delete mpUserData, mpUserData = NULL; } // ----------------------------------------------------------------------------- void GraphicObject::SetUserData( const String& rUserData ) { delete mpUserData, mpUserData = new String( rUserData ); } // ----------------------------------------------------------------------------- String GraphicObject::GetUserData() const { if( mpUserData ) return *mpUserData; else return String(); } // ----------------------------------------------------------------------------- void GraphicObject::SetSwapStreamHdl() { if( mpSwapStreamHdl ) { delete mpSwapOutTimer, mpSwapOutTimer = NULL; delete mpSwapStreamHdl, mpSwapStreamHdl = NULL; } } // ----------------------------------------------------------------------------- void GraphicObject::SetSwapStreamHdl( const Link& rHdl, const ULONG nSwapOutTimeout ) { delete mpSwapStreamHdl, mpSwapStreamHdl = new Link( rHdl ); if( nSwapOutTimeout ) { if( !mpSwapOutTimer ) { mpSwapOutTimer = new Timer; mpSwapOutTimer->SetTimeoutHdl( LINK( this, GraphicObject, ImplAutoSwapOutHdl ) ); } mpSwapOutTimer->SetTimeout( nSwapOutTimeout ); mpSwapOutTimer->Start(); } else delete mpSwapOutTimer, mpSwapOutTimer = NULL; } // ----------------------------------------------------------------------------- Link GraphicObject::GetSwapStreamHdl() const { if( mpSwapStreamHdl ) return *mpSwapStreamHdl; else return Link(); } // ----------------------------------------------------------------------------- void GraphicObject::FireSwapInRequest() { ImplAutoSwapIn(); } // ----------------------------------------------------------------------------- void GraphicObject::FireSwapOutRequest() { ImplAutoSwapOutHdl( NULL ); } // ----------------------------------------------------------------------------- void GraphicObject::GraphicManagerDestroyed() { // we're alive, but our manager doesn't live anymore ==> connect to default manager mpMgr = NULL; ImplSetGraphicManager( NULL ); } // ----------------------------------------------------------------------------- void GraphicObject::SetGraphicManager( const GraphicManager& rMgr ) { ImplSetGraphicManager( &rMgr ); } // ----------------------------------------------------------------------------- BOOL GraphicObject::IsCached( OutputDevice* pOut, const Point& rPt, const Size& rSz, const GraphicAttr* pAttr, ULONG nFlags ) const { BOOL bRet; if( nFlags & GRFMGR_DRAW_CACHED ) { // --> OD 2005-10-11 #i54875# - Consider cropped graphics. // Note: The graphic manager caches a cropped graphic with its // uncropped position and size. // bRet = mpMgr->IsInCache( pOut, rPt, rSz, *this, ( pAttr ? *pAttr : GetAttr() ) ); Point aPt( rPt ); Size aSz( rSz ); if ( pAttr->IsCropped() ) { PolyPolygon aClipPolyPoly; BOOL bRectClip; ImplGetCropParams( pOut, aPt, aSz, pAttr, aClipPolyPoly, bRectClip ); } bRet = mpMgr->IsInCache( pOut, aPt, aSz, *this, ( pAttr ? *pAttr : GetAttr() ) ); } else bRet = FALSE; return bRet; } // ----------------------------------------------------------------------------- void GraphicObject::ReleaseFromCache() { mpMgr->ReleaseFromCache( *this ); } // ----------------------------------------------------------------------------- void GraphicObject::SetAnimationNotifyHdl( const Link& rLink ) { maGraphic.SetAnimationNotifyHdl( rLink ); } // ----------------------------------------------------------------------------- List* GraphicObject::GetAnimationInfoList() const { return maGraphic.GetAnimationInfoList(); } // ----------------------------------------------------------------------------- BOOL GraphicObject::Draw( OutputDevice* pOut, const Point& rPt, const Size& rSz, const GraphicAttr* pAttr, ULONG nFlags ) { GraphicAttr aAttr( pAttr ? *pAttr : GetAttr() ); Point aPt( rPt ); Size aSz( rSz ); const sal_uInt32 nOldDrawMode = pOut->GetDrawMode(); BOOL bCropped = aAttr.IsCropped(); BOOL bCached = FALSE; BOOL bWritingPdfLinkedGraphic = FALSE; BOOL bRet; // #i29534# Provide output rects for PDF writer Rectangle aCropRect; // #i29534# Notify PDF writer about linked graphic (if any) vcl::ExtOutDevData* pExtOutDevData; if( (pExtOutDevData=pOut->GetExtOutDevData()) && pExtOutDevData->ISA(vcl::PDFExtOutDevData) ) { // #i29534# Only delegate image handling to PDF, if no special // treatment is necessary if( GetGraphic().IsLink() && aSz.Width() > 0L && aSz.Height() > 0L && !aAttr.IsSpecialDrawMode() && !aAttr.IsMirrored() && !aAttr.IsRotated() && !aAttr.IsAdjusted() ) { bWritingPdfLinkedGraphic = TRUE; static_cast< vcl::PDFExtOutDevData* >( pExtOutDevData )->BeginGroup(); } } if( !( GRFMGR_DRAW_USE_DRAWMODE_SETTINGS & nFlags ) ) pOut->SetDrawMode( nOldDrawMode & ( ~( DRAWMODE_SETTINGSLINE | DRAWMODE_SETTINGSFILL | DRAWMODE_SETTINGSTEXT | DRAWMODE_SETTINGSGRADIENT ) ) ); // mirrored horizontically if( aSz.Width() < 0L ) { aPt.X() += aSz.Width() + 1; aSz.Width() = -aSz.Width(); aAttr.SetMirrorFlags( aAttr.GetMirrorFlags() ^ BMP_MIRROR_HORZ ); } // mirrored vertically if( aSz.Height() < 0L ) { aPt.Y() += aSz.Height() + 1; aSz.Height() = -aSz.Height(); aAttr.SetMirrorFlags( aAttr.GetMirrorFlags() ^ BMP_MIRROR_VERT ); } if( bCropped ) { PolyPolygon aClipPolyPoly; BOOL bRectClip; const BOOL bCrop = ImplGetCropParams( pOut, aPt, aSz, &aAttr, aClipPolyPoly, bRectClip ); pOut->Push( PUSH_CLIPREGION ); if( bCrop ) { if( bRectClip ) { // #i29534# Store crop rect for later forwarding to // PDF writer aCropRect = aClipPolyPoly.GetBoundRect(); pOut->IntersectClipRegion( aCropRect ); } else { pOut->IntersectClipRegion( aClipPolyPoly ); } } } bRet = mpMgr->DrawObj( pOut, aPt, aSz, *this, aAttr, nFlags, bCached ); if( bCropped ) pOut->Pop(); pOut->SetDrawMode( nOldDrawMode ); // #i29534# Notify PDF writer about linked graphic (if any) if( bWritingPdfLinkedGraphic ) { static_cast< vcl::PDFExtOutDevData* >( pExtOutDevData )->EndGroup( const_cast< Graphic& >(GetGraphic()), aAttr.GetTransparency(), Rectangle( aPt, aSz ), aCropRect ); } // #i29534# Moved below OutDev restoration, to avoid multiple swap-ins // (code above needs to call GetGraphic twice) if( bCached ) { if( mpSwapOutTimer ) mpSwapOutTimer->Start(); else FireSwapOutRequest(); } return bRet; } // ----------------------------------------------------------------------------- BOOL GraphicObject::DrawTiled( OutputDevice* pOut, const Rectangle& rArea, const Size& rSize, const Size& rOffset, const GraphicAttr* pAttr, ULONG nFlags, int nTileCacheSize1D ) { if( pOut == NULL || rSize.Width() == 0 || rSize.Height() == 0 ) return FALSE; const MapMode aOutMapMode( pOut->GetMapMode() ); const MapMode aMapMode( aOutMapMode.GetMapUnit(), Point(), aOutMapMode.GetScaleX(), aOutMapMode.GetScaleY() ); // #106258# Clamp size to 1 for zero values. This is okay, since // logical size of zero is handled above already const Size aOutTileSize( ::std::max( 1L, pOut->LogicToPixel( rSize, aOutMapMode ).Width() ), ::std::max( 1L, pOut->LogicToPixel( rSize, aOutMapMode ).Height() ) ); return ImplDrawTiled( pOut, rArea, aOutTileSize, rOffset, pAttr, nFlags, nTileCacheSize1D ); } // ----------------------------------------------------------------------------- BOOL GraphicObject::StartAnimation( OutputDevice* pOut, const Point& rPt, const Size& rSz, long nExtraData, const GraphicAttr* pAttr, ULONG /*nFlags*/, OutputDevice* pFirstFrameOutDev ) { BOOL bRet = FALSE; GetGraphic(); if( !IsSwappedOut() ) { const GraphicAttr aAttr( pAttr ? *pAttr : GetAttr() ); if( mbAnimated ) { Point aPt( rPt ); Size aSz( rSz ); BOOL bCropped = aAttr.IsCropped(); if( bCropped ) { PolyPolygon aClipPolyPoly; BOOL bRectClip; const BOOL bCrop = ImplGetCropParams( pOut, aPt, aSz, &aAttr, aClipPolyPoly, bRectClip ); pOut->Push( PUSH_CLIPREGION ); if( bCrop ) { if( bRectClip ) pOut->IntersectClipRegion( aClipPolyPoly.GetBoundRect() ); else pOut->IntersectClipRegion( aClipPolyPoly ); } } if( !mpSimpleCache || ( mpSimpleCache->maAttr != aAttr ) || pFirstFrameOutDev ) { if( mpSimpleCache ) delete mpSimpleCache; mpSimpleCache = new GrfSimpleCacheObj( GetTransformedGraphic( &aAttr ), aAttr ); mpSimpleCache->maGraphic.SetAnimationNotifyHdl( GetAnimationNotifyHdl() ); } mpSimpleCache->maGraphic.StartAnimation( pOut, aPt, aSz, nExtraData, pFirstFrameOutDev ); if( bCropped ) pOut->Pop(); bRet = TRUE; } else bRet = Draw( pOut, rPt, rSz, &aAttr, GRFMGR_DRAW_STANDARD ); } return bRet; } // ----------------------------------------------------------------------------- void GraphicObject::StopAnimation( OutputDevice* pOut, long nExtraData ) { if( mpSimpleCache ) mpSimpleCache->maGraphic.StopAnimation( pOut, nExtraData ); } // ----------------------------------------------------------------------------- const Graphic& GraphicObject::GetGraphic() const { if( mbAutoSwapped ) ( (GraphicObject*) this )->ImplAutoSwapIn(); return maGraphic; } // ----------------------------------------------------------------------------- void GraphicObject::SetGraphic( const Graphic& rGraphic ) { mpMgr->ImplUnregisterObj( *this ); if( mpSwapOutTimer ) mpSwapOutTimer->Stop(); maGraphic = rGraphic; mbAutoSwapped = FALSE; ImplAssignGraphicData(); delete mpLink, mpLink = NULL; delete mpSimpleCache, mpSimpleCache = NULL; mpMgr->ImplRegisterObj( *this, maGraphic, NULL ); if( mpSwapOutTimer ) mpSwapOutTimer->Start(); } // ----------------------------------------------------------------------------- void GraphicObject::SetGraphic( const Graphic& rGraphic, const String& rLink ) { SetGraphic( rGraphic ); mpLink = new String( rLink ); } // ----------------------------------------------------------------------------- Graphic GraphicObject::GetTransformedGraphic( const Size& rDestSize, const MapMode& rDestMap, const GraphicAttr& rAttr ) const { // #104550# Extracted from svx/source/svdraw/svdograf.cxx Graphic aTransGraphic( maGraphic ); const GraphicType eType = GetType(); const Size aSrcSize( aTransGraphic.GetPrefSize() ); // #104115# Convert the crop margins to graphic object mapmode const MapMode aMapGraph( aTransGraphic.GetPrefMapMode() ); const MapMode aMap100( MAP_100TH_MM ); Size aCropLeftTop; Size aCropRightBottom; if( GRAPHIC_GDIMETAFILE == eType ) { GDIMetaFile aMtf( aTransGraphic.GetGDIMetaFile() ); if( aMapGraph == MAP_PIXEL ) { aCropLeftTop = Application::GetDefaultDevice()->LogicToPixel( Size( rAttr.GetLeftCrop(), rAttr.GetTopCrop() ), aMap100 ); aCropRightBottom = Application::GetDefaultDevice()->LogicToPixel( Size( rAttr.GetRightCrop(), rAttr.GetBottomCrop() ), aMap100 ); } else { aCropLeftTop = OutputDevice::LogicToLogic( Size( rAttr.GetLeftCrop(), rAttr.GetTopCrop() ), aMap100, aMapGraph ); aCropRightBottom = OutputDevice::LogicToLogic( Size( rAttr.GetRightCrop(), rAttr.GetBottomCrop() ), aMap100, aMapGraph ); } // #104115# If the metafile is cropped, give it a special // treatment: clip against the remaining area, scale up such // that this area later fills the desired size, and move the // origin to the upper left edge of that area. if( rAttr.IsCropped() ) { const MapMode aMtfMapMode( aMtf.GetPrefMapMode() ); Rectangle aClipRect( aMtfMapMode.GetOrigin().X() + aCropLeftTop.Width(), aMtfMapMode.GetOrigin().Y() + aCropLeftTop.Height(), aMtfMapMode.GetOrigin().X() + aSrcSize.Width() - aCropRightBottom.Width(), aMtfMapMode.GetOrigin().Y() + aSrcSize.Height() - aCropRightBottom.Height() ); // #104115# To correctly crop rotated metafiles, clip by view rectangle aMtf.AddAction( new MetaISectRectClipRegionAction( aClipRect ), 0 ); // #104115# To crop the metafile, scale larger than the output rectangle aMtf.Scale( (double)rDestSize.Width() / (aSrcSize.Width() - aCropLeftTop.Width() - aCropRightBottom.Width()), (double)rDestSize.Height() / (aSrcSize.Height() - aCropLeftTop.Height() - aCropRightBottom.Height()) ); // #104115# Adapt the pref size by hand (scale changes it // proportionally, but we want it to be smaller than the // former size, to crop the excess out) aMtf.SetPrefSize( Size( (long)((double)rDestSize.Width() * (1.0 + (aCropLeftTop.Width() + aCropRightBottom.Width()) / aSrcSize.Width()) + .5), (long)((double)rDestSize.Height() * (1.0 + (aCropLeftTop.Height() + aCropRightBottom.Height()) / aSrcSize.Height()) + .5) ) ); // #104115# Adapt the origin of the new mapmode, such that it // is shifted to the place where the cropped output starts Point aNewOrigin( (long)((double)aMtfMapMode.GetOrigin().X() + rDestSize.Width() * aCropLeftTop.Width() / (aSrcSize.Width() - aCropLeftTop.Width() - aCropRightBottom.Width()) + .5), (long)((double)aMtfMapMode.GetOrigin().Y() + rDestSize.Height() * aCropLeftTop.Height() / (aSrcSize.Height() - aCropLeftTop.Height() - aCropRightBottom.Height()) + .5) ); MapMode aNewMap( rDestMap ); aNewMap.SetOrigin( OutputDevice::LogicToLogic(aNewOrigin, aMtfMapMode, rDestMap) ); aMtf.SetPrefMapMode( aNewMap ); } else { aMtf.Scale( Fraction( rDestSize.Width(), aSrcSize.Width() ), Fraction( rDestSize.Height(), aSrcSize.Height() ) ); aMtf.SetPrefMapMode( rDestMap ); } aTransGraphic = aMtf; } else if( GRAPHIC_BITMAP == eType ) { BitmapEx aBitmapEx( aTransGraphic.GetBitmapEx() ); // convert crops to pixel aCropLeftTop = Application::GetDefaultDevice()->LogicToPixel( Size( rAttr.GetLeftCrop(), rAttr.GetTopCrop() ), aMap100 ); aCropRightBottom = Application::GetDefaultDevice()->LogicToPixel( Size( rAttr.GetRightCrop(), rAttr.GetBottomCrop() ), aMap100 ); // convert from prefmapmode to pixel const Size aSrcSizePixel( Application::GetDefaultDevice()->LogicToPixel( aSrcSize, aMapGraph ) ); // setup crop rectangle in pixel Rectangle aCropRect( aCropLeftTop.Width(), aCropLeftTop.Height(), aSrcSizePixel.Width() - aCropRightBottom.Width(), aSrcSizePixel.Height() - aCropRightBottom.Height() ); // #105641# Also crop animations if( aTransGraphic.IsAnimated() ) { int nFrame; Animation aAnim( aTransGraphic.GetAnimation() ); for( nFrame=0; nFramemaGraphic.ResetAnimationLoopCount(); } } // ----------------------------------------------------------------------------- BOOL GraphicObject::SwapOut() { BOOL bRet = ( !mbAutoSwapped ? maGraphic.SwapOut() : FALSE ); if( bRet && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedOut( *this ); return bRet; } // ----------------------------------------------------------------------------- BOOL GraphicObject::SwapOut( SvStream* pOStm ) { BOOL bRet = ( !mbAutoSwapped ? maGraphic.SwapOut( pOStm ) : FALSE ); if( bRet && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedOut( *this ); return bRet; } // ----------------------------------------------------------------------------- BOOL GraphicObject::SwapIn() { BOOL bRet; if( mbAutoSwapped ) { ImplAutoSwapIn(); bRet = TRUE; } else if( mpMgr && mpMgr->ImplFillSwappedGraphicObject( *this, maGraphic ) ) bRet = TRUE; else { bRet = maGraphic.SwapIn(); if( bRet && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedIn( *this ); } if( bRet ) ImplAssignGraphicData(); return bRet; } // ----------------------------------------------------------------------------- BOOL GraphicObject::SwapIn( SvStream* pIStm ) { BOOL bRet; if( mbAutoSwapped ) { ImplAutoSwapIn(); bRet = TRUE; } else if( mpMgr && mpMgr->ImplFillSwappedGraphicObject( *this, maGraphic ) ) bRet = TRUE; else { bRet = maGraphic.SwapIn( pIStm ); if( bRet && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedIn( *this ); } if( bRet ) ImplAssignGraphicData(); return bRet; } // ----------------------------------------------------------------------------- void GraphicObject::SetSwapState() { if( !IsSwappedOut() ) { mbAutoSwapped = TRUE; if( mpMgr ) mpMgr->ImplGraphicObjectWasSwappedOut( *this ); } } // ----------------------------------------------------------------------------- IMPL_LINK( GraphicObject, ImplAutoSwapOutHdl, void*, EMPTYARG ) { if( !IsSwappedOut() ) { mbIsInSwapOut = TRUE; SvStream* pStream = GetSwapStream(); if( GRFMGR_AUTOSWAPSTREAM_NONE != pStream ) { if( GRFMGR_AUTOSWAPSTREAM_LINK == pStream ) mbAutoSwapped = SwapOut( NULL ); else { if( GRFMGR_AUTOSWAPSTREAM_TEMP == pStream ) mbAutoSwapped = SwapOut(); else { mbAutoSwapped = SwapOut( pStream ); delete pStream; } } } mbIsInSwapOut = FALSE; } if( mpSwapOutTimer ) mpSwapOutTimer->Start(); return 0L; } // ------------------------------------------------------------------------ SvStream& operator>>( SvStream& rIStm, GraphicObject& rGraphicObj ) { VersionCompat aCompat( rIStm, STREAM_READ ); Graphic aGraphic; GraphicAttr aAttr; ByteString aLink; BOOL bLink; rIStm >> aGraphic >> aAttr >> bLink; rGraphicObj.SetGraphic( aGraphic ); rGraphicObj.SetAttr( aAttr ); if( bLink ) { rIStm >> aLink; rGraphicObj.SetLink( UniString( aLink, RTL_TEXTENCODING_UTF8 ) ); } else rGraphicObj.SetLink(); rGraphicObj.SetSwapStreamHdl(); return rIStm; } // ------------------------------------------------------------------------ SvStream& operator<<( SvStream& rOStm, const GraphicObject& rGraphicObj ) { VersionCompat aCompat( rOStm, STREAM_WRITE, 1 ); const BOOL bLink = rGraphicObj.HasLink(); rOStm << rGraphicObj.GetGraphic() << rGraphicObj.GetAttr() << bLink; if( bLink ) rOStm << ByteString( rGraphicObj.GetLink(), RTL_TEXTENCODING_UTF8 ); return rOStm; }