/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * 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 . */ #include "sal/config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using com::sun::star::uno::Reference; using com::sun::star::uno::XInterface; using com::sun::star::uno::UNO_QUERY; using com::sun::star::uno::Sequence; using com::sun::star::container::XNameContainer; using com::sun::star::beans::XPropertySet; GraphicManager* GraphicObject::mpGlobalMgr = NULL; struct GrfSimpleCacheObj { Graphic maGraphic; GraphicAttr maAttr; GrfSimpleCacheObj( const Graphic& rGraphic, const GraphicAttr& rAttr ) : maGraphic( rGraphic ), maAttr( rAttr ) {} }; TYPEINIT1_AUTOFACTORY( GraphicObject, SvDataCopyStream ); GraphicObject::GraphicObject( const GraphicManager* pMgr ) : maLink (), maUserData () { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr ); } GraphicObject::GraphicObject( const Graphic& rGraphic, const GraphicManager* pMgr ) : maGraphic ( rGraphic ), maLink (), maUserData () { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr ); } GraphicObject::GraphicObject( const GraphicObject& rGraphicObj, const GraphicManager* pMgr ) : SvDataCopyStream(), maGraphic ( rGraphicObj.GetGraphic() ), maAttr ( rGraphicObj.maAttr ), maLink ( rGraphicObj.maLink ), maUserData ( rGraphicObj.maUserData ) { ImplConstruct(); ImplAssignGraphicData(); ImplSetGraphicManager( pMgr, NULL, &rGraphicObj ); } GraphicObject::GraphicObject( const OString& rUniqueID, const GraphicManager* pMgr ) : maLink (), maUserData () { 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 mpSimpleCache; } void GraphicObject::ImplConstruct() { mpMgr = NULL; mpSwapStreamHdl = NULL; mpSwapOutTimer = NULL; mpSimpleCache = NULL; mnAnimationLoopCount = 0; mbAutoSwapped = sal_False; mbIsInSwapIn = sal_False; mbIsInSwapOut = sal_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(); mbEPS = maGraphic.IsEPS(); mnAnimationLoopCount = ( mbAnimated ? maGraphic.GetAnimationLoopCount() : 0 ); } void GraphicObject::ImplSetGraphicManager( const GraphicManager* pMgr, const OString* pID, const GraphicObject* pCopyObj ) { 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 ) { mpGlobalMgr = new GraphicManager( (officecfg::Office::Common::Cache::GraphicManager:: TotalCacheSize::get()), (officecfg::Office::Common::Cache::GraphicManager:: ObjectCacheSize::get())); mpGlobalMgr->SetCacheTimeout( officecfg::Office::Common::Cache::GraphicManager:: ObjectReleaseTime::get()); } mpMgr = mpGlobalMgr; } else mpMgr = (GraphicManager*) pMgr; mpMgr->ImplRegisterObj( *this, maGraphic, pID, pCopyObj ); } } } void GraphicObject::ImplAutoSwapIn() { if( IsSwappedOut() ) { if( mpMgr && mpMgr->ImplFillSwappedGraphicObject( *this, maGraphic ) ) mbAutoSwapped = sal_False; else { mbIsInSwapIn = sal_True; if( maGraphic.SwapIn() ) mbAutoSwapped = sal_False; else { SvStream* pStream = GetSwapStream(); if( GRFMGR_AUTOSWAPSTREAM_NONE != pStream ) { if( GRFMGR_AUTOSWAPSTREAM_LINK == pStream ) { if( HasLink() ) { OUString 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 = sal_False; if( !mbAutoSwapped && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedIn( *this ); } } } sal_Bool GraphicObject::ImplGetCropParams( OutputDevice* pOut, Point& rPt, Size& rSz, const GraphicAttr* pAttr, PolyPolygon& rClipPolyPoly, sal_Bool& bRectClipRegion ) const { sal_Bool bRet = sal_False; if( GetType() != GRAPHIC_NONE ) { Polygon aClipPoly( Rectangle( rPt, rSz ) ); const sal_uInt16 nRot10 = pAttr->GetRotation() % 3600; const Point aOldOrigin( rPt ); const MapMode aMap100( MAP_100TH_MM ); Size aSize100; long nTotalWidth, nTotalHeight; if( nRot10 ) { aClipPoly.Rotate( rPt, nRot10 ); bRectClipRegion = sal_False; } else bRectClipRegion = sal_True; rClipPolyPoly = aClipPoly; if( maGraphic.GetPrefMapMode() == MAP_PIXEL ) aSize100 = Application::GetDefaultDevice()->PixelToLogic( maGraphic.GetPrefSize(), aMap100 ); else { MapMode m(maGraphic.GetPrefMapMode()); aSize100 = pOut->LogicToLogic( maGraphic.GetPrefSize(), &m, &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 ) { double fScale = (double) aSize100.Width() / nTotalWidth; const long nNewLeft = -FRound( ( ( pAttr->GetMirrorFlags() & BMP_MIRROR_HORZ ) ? pAttr->GetRightCrop() : pAttr->GetLeftCrop() ) * fScale ); const long 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; const long nNewTop = -FRound( ( ( pAttr->GetMirrorFlags() & BMP_MIRROR_VERT ) ? pAttr->GetBottomCrop() : pAttr->GetTopCrop() ) * fScale ); const long 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 = sal_True; } } return bRet; } GraphicObject& GraphicObject::operator=( const GraphicObject& rGraphicObj ) { if( &rGraphicObj != this ) { mpMgr->ImplUnregisterObj( *this ); delete mpSwapStreamHdl, mpSwapStreamHdl = NULL; delete mpSimpleCache, mpSimpleCache = NULL; maGraphic = rGraphicObj.GetGraphic(); maAttr = rGraphicObj.maAttr; maLink = rGraphicObj.maLink; maUserData = rGraphicObj.maUserData; ImplAssignGraphicData(); mbAutoSwapped = sal_False; mpMgr = rGraphicObj.mpMgr; mpMgr->ImplRegisterObj( *this, maGraphic, NULL, &rGraphicObj ); } return *this; } sal_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 ) { WriteGraphicObject( rOStm, *this ); } void GraphicObject::Assign( const SvDataCopyStream& rCopyStream ) { *this = (const GraphicObject& ) rCopyStream; } OString GraphicObject::GetUniqueID() const { if ( !IsInSwapIn() && IsEPS() ) const_cast(this)->FireSwapInRequest(); OString aRet; if( mpMgr ) aRet = mpMgr->ImplGetUniqueID( *this ); return aRet; } SvStream* GraphicObject::GetSwapStream() const { return( HasSwapStreamHdl() ? (SvStream*) mpSwapStreamHdl->Call( (void*) this ) : GRFMGR_AUTOSWAPSTREAM_NONE ); } void GraphicObject::SetAttr( const GraphicAttr& rAttr ) { maAttr = rAttr; if( mpSimpleCache && ( mpSimpleCache->maAttr != rAttr ) ) delete mpSimpleCache, mpSimpleCache = NULL; } void GraphicObject::SetLink() { maLink = ""; } void GraphicObject::SetLink( const OUString& rLink ) { maLink = rLink; } void GraphicObject::SetUserData() { maUserData = ""; } void GraphicObject::SetUserData( const OUString& rUserData ) { maUserData = rUserData; } void GraphicObject::SetSwapStreamHdl() { if( mpSwapStreamHdl ) { delete mpSwapOutTimer, mpSwapOutTimer = NULL; delete mpSwapStreamHdl, mpSwapStreamHdl = NULL; } } #define SWAPGRAPHIC_TIMEOUT 5000 // #i122985# it is not correct to set the swap-timeout to a hard-coded 5000ms // as it was before. Added code and experimented what to do as a good // compromise, see description. static sal_uInt32 GetCacheTimeInMs() { static bool bSetAtAll(true); if (bSetAtAll) { static bool bSetToPreferenceTime(true); if (bSetToPreferenceTime) { const sal_uInt32 nSeconds = officecfg::Office::Common::Cache::GraphicManager::ObjectReleaseTime::get( comphelper::getProcessComponentContext()); // The default is 10 minutes. The minimum is one minute, thus 60 // seconds. When the minimum should match to the former hard-coded // 5 seconds, we have a divisor of 12 to use. For the default of 10 // minutes this would mean 50 seconds. Compared to before this is // ten times more (would allow better navigation by switching // through pages) and is controllable by the user by setting the // tools/options/memory/Remove_from_memory_after setting. Seems to // be a good compromise to me. return nSeconds * 1000 / 12; } else { return SWAPGRAPHIC_TIMEOUT; } } return 0; } void GraphicObject::SetSwapStreamHdl(const Link& rHdl) { delete mpSwapStreamHdl, mpSwapStreamHdl = new Link( rHdl ); sal_uInt32 const nSwapOutTimeout(GetCacheTimeInMs()); if( nSwapOutTimeout ) { if( !mpSwapOutTimer ) { mpSwapOutTimer = new Timer; mpSwapOutTimer->SetTimeoutHdl( LINK( this, GraphicObject, ImplAutoSwapOutHdl ) ); } mpSwapOutTimer->SetTimeout( nSwapOutTimeout ); mpSwapOutTimer->Start(); } else delete mpSwapOutTimer, mpSwapOutTimer = NULL; } 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 ); } sal_Bool GraphicObject::IsCached( OutputDevice* pOut, const Point& rPt, const Size& rSz, const GraphicAttr* pAttr, sal_uLong nFlags ) const { sal_Bool bRet; if( nFlags & GRFMGR_DRAW_CACHED ) { Point aPt( rPt ); Size aSz( rSz ); if ( pAttr && pAttr->IsCropped() ) { PolyPolygon aClipPolyPoly; sal_Bool bRectClip; ImplGetCropParams( pOut, aPt, aSz, pAttr, aClipPolyPoly, bRectClip ); } bRet = mpMgr->IsInCache( pOut, aPt, aSz, *this, ( pAttr ? *pAttr : GetAttr() ) ); } else bRet = sal_False; return bRet; } void GraphicObject::ReleaseFromCache() { mpMgr->ReleaseFromCache( *this ); } bool GraphicObject::Draw( OutputDevice* pOut, const Point& rPt, const Size& rSz, const GraphicAttr* pAttr, sal_uLong nFlags ) { GraphicAttr aAttr( pAttr ? *pAttr : GetAttr() ); Point aPt( rPt ); Size aSz( rSz ); const sal_uInt32 nOldDrawMode = pOut->GetDrawMode(); sal_Bool bCropped = aAttr.IsCropped(); sal_Bool bCached = sal_False; bool bRet; // #i29534# Provide output rects for PDF writer Rectangle aCropRect; 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; sal_Bool bRectClip; const sal_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(Region(aClipPolyPoly)); } } } bRet = mpMgr->DrawObj( pOut, aPt, aSz, *this, aAttr, nFlags, bCached ); if( bCropped ) pOut->Pop(); pOut->SetDrawMode( nOldDrawMode ); // #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; } // #i105243# sal_Bool GraphicObject::DrawWithPDFHandling( OutputDevice& rOutDev, const Point& rPt, const Size& rSz, const GraphicAttr* pGrfAttr, const sal_uLong nFlags ) { const GraphicAttr aGrfAttr( pGrfAttr ? *pGrfAttr : GetAttr() ); // Notify PDF writer about linked graphic (if any) sal_Bool bWritingPdfLinkedGraphic( sal_False ); Point aPt( rPt ); Size aSz( rSz ); Rectangle aCropRect; vcl::PDFExtOutDevData* pPDFExtOutDevData = dynamic_cast(rOutDev.GetExtOutDevData()); if( pPDFExtOutDevData ) { // only delegate image handling to PDF, if no special treatment is necessary if( GetGraphic().IsLink() && rSz.Width() > 0L && rSz.Height() > 0L && !aGrfAttr.IsSpecialDrawMode() && !aGrfAttr.IsMirrored() && !aGrfAttr.IsRotated() && !aGrfAttr.IsAdjusted() ) { bWritingPdfLinkedGraphic = true; if( aGrfAttr.IsCropped() ) { PolyPolygon aClipPolyPoly; sal_Bool bRectClip; const sal_Bool bCrop = ImplGetCropParams( &rOutDev, aPt, aSz, &aGrfAttr, aClipPolyPoly, bRectClip ); if ( bCrop && bRectClip ) { aCropRect = aClipPolyPoly.GetBoundRect(); } } pPDFExtOutDevData->BeginGroup(); } } sal_Bool bRet = Draw( &rOutDev, rPt, rSz, &aGrfAttr, nFlags ); // Notify PDF writer about linked graphic (if any) if( bWritingPdfLinkedGraphic ) { pPDFExtOutDevData->EndGroup( const_cast< Graphic& >(GetGraphic()), aGrfAttr.GetTransparency(), Rectangle( aPt, aSz ), aCropRect ); } return bRet; } sal_Bool GraphicObject::DrawTiled( OutputDevice* pOut, const Rectangle& rArea, const Size& rSize, const Size& rOffset, const GraphicAttr* pAttr, sal_uLong nFlags, int nTileCacheSize1D ) { if( pOut == NULL || rSize.Width() == 0 || rSize.Height() == 0 ) return sal_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() ) ); //#i69780 clip final tile size to a sane max size while (((sal_Int64)rSize.Width() * nTileCacheSize1D) > SAL_MAX_UINT16) nTileCacheSize1D /= 2; while (((sal_Int64)rSize.Height() * nTileCacheSize1D) > SAL_MAX_UINT16) nTileCacheSize1D /= 2; return ImplDrawTiled( pOut, rArea, aOutTileSize, rOffset, pAttr, nFlags, nTileCacheSize1D ); } sal_Bool GraphicObject::StartAnimation( OutputDevice* pOut, const Point& rPt, const Size& rSz, long nExtraData, const GraphicAttr* pAttr, sal_uLong /*nFlags*/, OutputDevice* pFirstFrameOutDev ) { sal_Bool bRet = sal_False; GetGraphic(); if( !IsSwappedOut() ) { const GraphicAttr aAttr( pAttr ? *pAttr : GetAttr() ); if( mbAnimated ) { Point aPt( rPt ); Size aSz( rSz ); sal_Bool bCropped = aAttr.IsCropped(); if( bCropped ) { PolyPolygon aClipPolyPoly; sal_Bool bRectClip; const sal_Bool bCrop = ImplGetCropParams( pOut, aPt, aSz, &aAttr, aClipPolyPoly, bRectClip ); pOut->Push( PUSH_CLIPREGION ); if( bCrop ) { if( bRectClip ) pOut->IntersectClipRegion( aClipPolyPoly.GetBoundRect() ); else pOut->IntersectClipRegion(Region(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 = sal_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, const GraphicObject* pCopyObj ) { mpMgr->ImplUnregisterObj( *this ); if( mpSwapOutTimer ) mpSwapOutTimer->Stop(); maGraphic = rGraphic; mbAutoSwapped = sal_False; ImplAssignGraphicData(); maLink = ""; delete mpSimpleCache, mpSimpleCache = NULL; mpMgr->ImplRegisterObj( *this, maGraphic, 0, pCopyObj); if( mpSwapOutTimer ) mpSwapOutTimer->Start(); } void GraphicObject::SetGraphic( const Graphic& rGraphic, const OUString& rLink ) { SetGraphic( rGraphic ); maLink = 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 ) { // crops are in 1/100th mm -> to aMapGraph -> to MAP_PIXEL aCropLeftTop = Application::GetDefaultDevice()->LogicToPixel( Size(rAttr.GetLeftCrop(), rAttr.GetTopCrop()), aMap100); aCropRightBottom = Application::GetDefaultDevice()->LogicToPixel( Size(rAttr.GetRightCrop(), rAttr.GetBottomCrop()), aMap100); } else { // crops are in GraphicObject units -> to aMapGraph 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() ); Rectangle aCropRect; // convert crops to pixel if(rAttr.IsCropped()) { if( aMapGraph == MAP_PIXEL ) { // crops are in 1/100th mm -> to MAP_PIXEL aCropLeftTop = Application::GetDefaultDevice()->LogicToPixel( Size(rAttr.GetLeftCrop(), rAttr.GetTopCrop()), aMap100); aCropRightBottom = Application::GetDefaultDevice()->LogicToPixel( Size(rAttr.GetRightCrop(), rAttr.GetBottomCrop()), aMap100); } else { // crops are in GraphicObject units -> to MAP_PIXEL aCropLeftTop = Application::GetDefaultDevice()->LogicToPixel( Size(rAttr.GetLeftCrop(), rAttr.GetTopCrop()), aMapGraph); aCropRightBottom = Application::GetDefaultDevice()->LogicToPixel( Size(rAttr.GetRightCrop(), rAttr.GetBottomCrop()), aMapGraph); } // convert from prefmapmode to pixel Size aSrcSizePixel( Application::GetDefaultDevice()->LogicToPixel( aSrcSize, aMapGraph)); if(rAttr.IsCropped() && (aSrcSizePixel.Width() != aBitmapEx.GetSizePixel().Width() || aSrcSizePixel.Height() != aBitmapEx.GetSizePixel().Height()) && aSrcSizePixel.Width()) { // the size in pixels calculated from Graphic's internal MapMode (aTransGraphic.GetPrefMapMode()) // and it's internal size (aTransGraphic.GetPrefSize()) is different from it's real pixel size. // This can be interpreted as this values to be set wrong, but needs to be corrected since e.g. // existing cropping is calculated based on this logic values already. // aBitmapEx.Scale(aSrcSizePixel); // another possibility is to adapt the values created so far with a factor; this // will keep the original Bitmap untouched and thus quality will not change // caution: convert to double first, else pretty big errors may occur const double fFactorX((double)aBitmapEx.GetSizePixel().Width() / aSrcSizePixel.Width()); const double fFactorY((double)aBitmapEx.GetSizePixel().Height() / aSrcSizePixel.Height()); aCropLeftTop.Width() = basegfx::fround(aCropLeftTop.Width() * fFactorX); aCropLeftTop.Height() = basegfx::fround(aCropLeftTop.Height() * fFactorY); aCropRightBottom.Width() = basegfx::fround(aCropRightBottom.Width() * fFactorX); aCropRightBottom.Height() = basegfx::fround(aCropRightBottom.Height() * fFactorY); aSrcSizePixel = aBitmapEx.GetSizePixel(); } // setup crop rectangle in pixel aCropRect = Rectangle( aCropLeftTop.Width(), aCropLeftTop.Height(), aSrcSizePixel.Width() - aCropRightBottom.Width(), aSrcSizePixel.Height() - aCropRightBottom.Height() ); } // #105641# Also crop animations if( aTransGraphic.IsAnimated() ) { sal_uInt16 nFrame; Animation aAnim( aTransGraphic.GetAnimation() ); for( nFrame=0; nFrameImplGraphicObjectWasSwappedOut( *this ); return bRet; } sal_Bool GraphicObject::SwapOut( SvStream* pOStm ) { sal_Bool bRet = ( !mbAutoSwapped ? maGraphic.SwapOut( pOStm ) : sal_False ); if( bRet && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedOut( *this ); return bRet; } sal_Bool GraphicObject::SwapIn() { sal_Bool bRet; if( mbAutoSwapped ) { ImplAutoSwapIn(); bRet = sal_True; } else if( mpMgr && mpMgr->ImplFillSwappedGraphicObject( *this, maGraphic ) ) bRet = sal_True; else { bRet = maGraphic.SwapIn(); if( bRet && mpMgr ) mpMgr->ImplGraphicObjectWasSwappedIn( *this ); } if( bRet ) ImplAssignGraphicData(); return bRet; } void GraphicObject::SetSwapState() { if( !IsSwappedOut() ) { mbAutoSwapped = sal_True; if( mpMgr ) mpMgr->ImplGraphicObjectWasSwappedOut( *this ); } } IMPL_LINK_NOARG(GraphicObject, ImplAutoSwapOutHdl) { if( !IsSwappedOut() ) { mbIsInSwapOut = sal_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 = sal_False; } if( mpSwapOutTimer ) mpSwapOutTimer->Start(); return 0L; } SvStream& operator>>( SvStream& rIStm, GraphicObject& rGraphicObj ) { VersionCompat aCompat( rIStm, STREAM_READ ); Graphic aGraphic; GraphicAttr aAttr; sal_Bool bLink; rIStm >> aGraphic >> aAttr >> bLink; rGraphicObj.SetGraphic( aGraphic ); rGraphicObj.SetAttr( aAttr ); if( bLink ) { OUString aLink = read_lenPrefixed_uInt8s_ToOUString(rIStm, RTL_TEXTENCODING_UTF8); rGraphicObj.SetLink(aLink); } else rGraphicObj.SetLink(); rGraphicObj.SetSwapStreamHdl(); return rIStm; } SvStream& WriteGraphicObject( SvStream& rOStm, const GraphicObject& rGraphicObj ) { VersionCompat aCompat( rOStm, STREAM_WRITE, 1 ); const sal_Bool bLink = rGraphicObj.HasLink(); WriteGraphic( rOStm, rGraphicObj.GetGraphic() ); WriteGraphicAttr( rOStm, rGraphicObj.GetAttr() ); rOStm.WriteUChar( bLink ); if( bLink ) write_uInt16_lenPrefixed_uInt8s_FromOUString(rOStm, rGraphicObj.GetLink(), RTL_TEXTENCODING_UTF8); return rOStm; } #define UNO_NAME_GRAPHOBJ_URLPREFIX "vnd.sun.star.GraphicObject:" GraphicObject GraphicObject::CreateGraphicObjectFromURL( const OUString &rURL ) { const OUString aURL( rURL ), aPrefix( UNO_NAME_GRAPHOBJ_URLPREFIX ); if( aURL.startsWith( aPrefix ) ) { // graphic manager url OString aUniqueID(OUStringToOString(rURL.copy(sizeof(UNO_NAME_GRAPHOBJ_URLPREFIX) - 1), RTL_TEXTENCODING_UTF8)); return GraphicObject( aUniqueID ); } else { Graphic aGraphic; if ( !aURL.isEmpty() ) { SvStream* pStream = utl::UcbStreamHelper::CreateStream( aURL, STREAM_READ ); if( pStream ) { GraphicConverter::Import( *pStream, aGraphic ); delete pStream; } } return GraphicObject( aGraphic ); } } void GraphicObject::InspectForGraphicObjectImageURL( const Reference< XInterface >& xIf, std::vector< OUString >& rvEmbedImgUrls ) { static OUString sImageURL( "ImageURL" ); Reference< XPropertySet > xProps( xIf, UNO_QUERY ); if ( xProps.is() ) { if ( xProps->getPropertySetInfo()->hasPropertyByName( sImageURL ) ) { OUString sURL; xProps->getPropertyValue( sImageURL ) >>= sURL; if ( !sURL.isEmpty() && sURL.startsWith( UNO_NAME_GRAPHOBJ_URLPREFIX ) ) rvEmbedImgUrls.push_back( sURL ); } } Reference< XNameContainer > xContainer( xIf, UNO_QUERY ); if ( xContainer.is() ) { Sequence< OUString > sNames = xContainer->getElementNames(); sal_Int32 nContainees = sNames.getLength(); for ( sal_Int32 index = 0; index < nContainees; ++index ) { Reference< XInterface > xCtrl; xContainer->getByName( sNames[ index ] ) >>= xCtrl; InspectForGraphicObjectImageURL( xCtrl, rvEmbedImgUrls ); } } } // calculate scalings between real image size and logic object size. This // is necessary since the crop values are relative to original bitmap size basegfx::B2DVector GraphicObject::calculateCropScaling( double fWidth, double fHeight, double fLeftCrop, double fTopCrop, double fRightCrop, double fBottomCrop) const { const MapMode aMapMode100thmm(MAP_100TH_MM); Size aBitmapSize(GetPrefSize()); double fFactorX(1.0); double fFactorY(1.0); if(MAP_PIXEL == GetPrefMapMode().GetMapUnit()) { aBitmapSize = Application::GetDefaultDevice()->PixelToLogic(aBitmapSize, aMapMode100thmm); } else { aBitmapSize = Application::GetDefaultDevice()->LogicToLogic(aBitmapSize, GetPrefMapMode(), aMapMode100thmm); } const double fDivX(aBitmapSize.Width() - fLeftCrop - fRightCrop); const double fDivY(aBitmapSize.Height() - fTopCrop - fBottomCrop); if(!basegfx::fTools::equalZero(fDivX)) { fFactorX = fabs(fWidth) / fDivX; } if(!basegfx::fTools::equalZero(fDivY)) { fFactorY = fabs(fHeight) / fDivY; } return basegfx::B2DVector(fFactorX,fFactorY); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */