/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include // #i99665# using namespace com::sun::star; extern void ClrContourCache( const SdrObject *pObj ); // TxtFly.Cxx inline sal_Bool GetRealURL( const SwGrfNode& rNd, String& rTxt ) { sal_Bool bRet = rNd.GetFileFilterNms( &rTxt, 0 ); if( bRet ) rTxt = URIHelper::removePassword( rTxt, INetURLObject::WAS_ENCODED, INetURLObject::DECODE_UNAMBIGUOUS); return bRet; } static void lcl_PaintReplacement( const SwRect &rRect, const String &rText, const ViewShell &rSh, const SwNoTxtFrm *pFrm, sal_Bool bDefect ) { static Font *pFont = 0; if ( !pFont ) { pFont = new Font(); pFont->SetWeight( WEIGHT_BOLD ); pFont->SetStyleName( aEmptyStr ); pFont->SetName(rtl::OUString("Arial Unicode")); pFont->SetFamily( FAMILY_SWISS ); pFont->SetTransparent( sal_True ); } Color aCol( COL_RED ); FontUnderline eUnderline = UNDERLINE_NONE; const SwFmtURL &rURL = pFrm->FindFlyFrm()->GetFmt()->GetURL(); if( rURL.GetURL().Len() || rURL.GetMap() ) { sal_Bool bVisited = sal_False; if ( rURL.GetMap() ) { ImageMap *pMap = (ImageMap*)rURL.GetMap(); for( sal_uInt16 i = 0; i < pMap->GetIMapObjectCount(); i++ ) { IMapObject *pObj = pMap->GetIMapObject( i ); if( rSh.GetDoc()->IsVisitedURL( pObj->GetURL() ) ) { bVisited = sal_True; break; } } } else if ( rURL.GetURL().Len() ) bVisited = rSh.GetDoc()->IsVisitedURL( rURL.GetURL() ); SwFmt *pFmt = rSh.GetDoc()->GetFmtFromPool( static_cast (bVisited ? RES_POOLCHR_INET_VISIT : RES_POOLCHR_INET_NORMAL ) ); aCol = pFmt->GetColor().GetValue(); eUnderline = pFmt->GetUnderline().GetLineStyle(); } pFont->SetUnderline( eUnderline ); pFont->SetColor( aCol ); const BitmapEx& rBmp = ViewShell::GetReplacementBitmap( bDefect != sal_False ); Graphic::DrawEx( rSh.GetOut(), rText, *pFont, rBmp, rRect.Pos(), rRect.SSize() ); } /************************************************************************* |* |* SwGrfFrm::SwGrfFrm(ViewShell * const,SwGrfNode *) |* *************************************************************************/ SwNoTxtFrm::SwNoTxtFrm(SwNoTxtNode * const pNode, SwFrm* pSib ) : SwCntntFrm( pNode, pSib ) { InitCtor(); } // Initialization: Currently add the Frame to the Cache void SwNoTxtFrm::InitCtor() { nType = FRMC_NOTXT; // The graphic's weight is 0 if it has not been read, // < 0 if we had a read error and we needed to use the replacement and // > 0 if it is available nWeight = 0; } /************************************************************************* |* |* SwNoTxtNode::MakeFrm() |* *************************************************************************/ SwCntntFrm *SwNoTxtNode::MakeFrm( SwFrm* pSib ) { return new SwNoTxtFrm(this, pSib); } /************************************************************************* |* |* SwNoTxtFrm::~SwNoTxtFrm() |* *************************************************************************/ SwNoTxtFrm::~SwNoTxtFrm() { StopAnimation(); } /************************************************************************* |* |* void SwNoTxtFrm::Modify( SwHint * pOld, SwHint * pNew ) |* *************************************************************************/ void SetOutDev( ViewShell *pSh, OutputDevice *pOut ) { pSh->pOut = pOut; } static void lcl_ClearArea( const SwFrm &rFrm, OutputDevice &rOut, const SwRect& rPtArea, const SwRect &rGrfArea ) { SwRegionRects aRegion( rPtArea, 4 ); aRegion -= rGrfArea; if ( !aRegion.empty() ) { const SvxBrushItem *pItem; const Color *pCol; SwRect aOrigRect; if ( rFrm.GetBackgroundBrush( pItem, pCol, aOrigRect, sal_False ) ) for( sal_uInt16 i = 0; i < aRegion.size(); ++i ) ::DrawGraphic( pItem, &rOut, aOrigRect, aRegion[i] ); else { rOut.Push( PUSH_FILLCOLOR|PUSH_LINECOLOR ); rOut.SetFillColor( rFrm.getRootFrm()->GetCurrShell()->Imp()->GetRetoucheColor()); rOut.SetLineColor(); for( sal_uInt16 i = 0; i < aRegion.size(); ++i ) rOut.DrawRect( aRegion[i].SVRect() ); rOut.Pop(); } } } /************************************************************************* |* |* void SwNoTxtFrm::Paint() |* *************************************************************************/ void SwNoTxtFrm::Paint(SwRect const& rRect, SwPrintData const*const) const { if ( Frm().IsEmpty() ) return; const ViewShell* pSh = getRootFrm()->GetCurrShell(); if( !pSh->GetViewOptions()->IsGraphic() ) { StopAnimation(); // #i6467# - no paint of placeholder for page preview if ( pSh->GetWin() && !pSh->IsPreView() ) { const SwNoTxtNode* pNd = GetNode()->GetNoTxtNode(); String aTxt( pNd->GetTitle() ); if ( !aTxt.Len() && pNd->IsGrfNode() ) GetRealURL( *(SwGrfNode*)pNd, aTxt ); if( !aTxt.Len() ) aTxt = FindFlyFrm()->GetFmt()->GetName(); lcl_PaintReplacement( Frm(), aTxt, *pSh, this, sal_False ); } return; } if( pSh->GetAccessibilityOptions()->IsStopAnimatedGraphics() || // #i9684# Stop animation during printing/pdf export !pSh->GetWin() ) StopAnimation(); SfxProgress::EnterLock(); // No progress reschedules in paint (SwapIn) OutputDevice *pOut = pSh->GetOut(); pOut->Push(); bool bClip = true; PolyPolygon aPoly; SwNoTxtNode& rNoTNd = *(SwNoTxtNode*)GetNode(); SwGrfNode* pGrfNd = rNoTNd.GetGrfNode(); if( pGrfNd ) pGrfNd->SetFrameInPaint( sal_True ); // #i13147# - add 2nd parameter with value to // method call to indicate that it is called // for paint in order to avoid load of the intrinsic graphic. if ( ( !pOut->GetConnectMetaFile() || !pSh->GetWin() ) && FindFlyFrm()->GetContour( aPoly, sal_True ) ) { pOut->SetClipRegion( aPoly ); bClip = false; } SwRect aOrigPaint( rRect ); if ( HasAnimation() && pSh->GetWin() ) { aOrigPaint = Frm(); aOrigPaint += Prt().Pos(); } SwRect aGrfArea( Frm() ); SwRect aPaintArea( aGrfArea ); // In case the picture fly frm was clipped, render it with the origin // size instead of scaling it if ( rNoTNd.getIDocumentSettingAccess()->get( IDocumentSettingAccess::CLIPPED_PICTURES ) ) { const SwFlyFreeFrm *pFly = dynamic_cast< const SwFlyFreeFrm* >( FindFlyFrm() ); if( pFly ) aGrfArea = SwRect( Frm().Pos( ), pFly->GetUnclippedFrm( ).SSize( ) ); } aPaintArea._Intersection( aOrigPaint ); SwRect aNormal( Frm().Pos() + Prt().Pos(), Prt().SSize() ); aNormal.Justify(); // Normalized rectangle for the comparisons if( aPaintArea.IsOver( aNormal ) ) { // Calculate the four to-be-deleted rectangles if( pSh->GetWin() ) ::lcl_ClearArea( *this, *pSh->GetOut(), aPaintArea, aNormal ); // The intersection of the PaintArea and the Bitmap contains the absolutely visible area of the Frame aPaintArea._Intersection( aNormal ); if ( bClip ) pOut->IntersectClipRegion( aPaintArea.SVRect() ); /// delete unused 3rd parameter PaintPicture( pOut, aGrfArea ); } else // If it's not visible, simply delete the given Area lcl_ClearArea( *this, *pSh->GetOut(), aPaintArea, SwRect() ); if( pGrfNd ) pGrfNd->SetFrameInPaint( sal_False ); pOut->Pop(); SfxProgress::LeaveLock(); } /************************************************************************* |* |* void lcl_CalcRect( Point & aPt, Size & aDim, |* sal_uInt16 nMirror ) |* |* Calculate the position and the size of the graphic in the Frame, |* corresponding to the current graphic attributes |* |* Point& the position in the Frame (also returned) |* Size& the graphic's size (also returned) |* nMirror the current mirror attribute |* *************************************************************************/ static void lcl_CalcRect( Point& rPt, Size& rDim, sal_uInt16 nMirror ) { if( nMirror == RES_MIRROR_GRAPH_VERT || nMirror == RES_MIRROR_GRAPH_BOTH ) { rPt.X() += rDim.Width() -1; rDim.Width() = -rDim.Width(); } if( nMirror == RES_MIRROR_GRAPH_HOR || nMirror == RES_MIRROR_GRAPH_BOTH ) { rPt.Y() += rDim.Height() -1; rDim.Height() = -rDim.Height(); } } /************************************************************************* |* |* void SwNoTxtFrm::GetGrfArea() |* |* Calculate the Bitmap's position and the size within the passed rectangle |* *************************************************************************/ void SwNoTxtFrm::GetGrfArea( SwRect &rRect, SwRect* pOrigRect, sal_Bool ) const { // Currently only used for scaling, cropping and mirroring the contour of graphics! // Everything else is handled by GraphicObject // We put the graphic's visible rectangle into rRect. // pOrigRect contains position and size of the whole graphic. const SwAttrSet& rAttrSet = GetNode()->GetSwAttrSet(); const SwCropGrf& rCrop = rAttrSet.GetCropGrf(); sal_uInt16 nMirror = rAttrSet.GetMirrorGrf().GetValue(); if( rAttrSet.GetMirrorGrf().IsGrfToggle() ) { if( !(FindPageFrm()->GetVirtPageNum() % 2) ) { switch ( nMirror ) { case RES_MIRROR_GRAPH_DONT: nMirror = RES_MIRROR_GRAPH_VERT; break; case RES_MIRROR_GRAPH_VERT: nMirror = RES_MIRROR_GRAPH_DONT; break; case RES_MIRROR_GRAPH_HOR: nMirror = RES_MIRROR_GRAPH_BOTH; break; default: nMirror = RES_MIRROR_GRAPH_HOR; break; } } } // We read graphic from the Node, if needed. // It may fail, however. long nLeftCrop, nRightCrop, nTopCrop, nBottomCrop; Size aOrigSz( ((SwNoTxtNode*)GetNode())->GetTwipSize() ); if ( !aOrigSz.Width() ) { aOrigSz.Width() = Prt().Width(); nLeftCrop = -rCrop.GetLeft(); nRightCrop = -rCrop.GetRight(); } else { nLeftCrop = Max( aOrigSz.Width() - (rCrop.GetRight() + rCrop.GetLeft()), long(1) ); const double nScale = double(Prt().Width()) / double(nLeftCrop); nLeftCrop = long(nScale * -rCrop.GetLeft() ); nRightCrop = long(nScale * -rCrop.GetRight() ); } // crop values have to be mirrored too if( nMirror == RES_MIRROR_GRAPH_VERT || nMirror == RES_MIRROR_GRAPH_BOTH ) { long nTmpCrop = nLeftCrop; nLeftCrop = nRightCrop; nRightCrop= nTmpCrop; } if( !aOrigSz.Height() ) { aOrigSz.Height() = Prt().Height(); nTopCrop = -rCrop.GetTop(); nBottomCrop= -rCrop.GetBottom(); } else { nTopCrop = Max( aOrigSz.Height() - (rCrop.GetTop() + rCrop.GetBottom()), long(1) ); const double nScale = double(Prt().Height()) / double(nTopCrop); nTopCrop = long(nScale * -rCrop.GetTop() ); nBottomCrop= long(nScale * -rCrop.GetBottom() ); } // crop values have to be mirrored too if( nMirror == RES_MIRROR_GRAPH_HOR || nMirror == RES_MIRROR_GRAPH_BOTH ) { long nTmpCrop = nTopCrop; nTopCrop = nBottomCrop; nBottomCrop= nTmpCrop; } Size aVisSz( Prt().SSize() ); Size aGrfSz( aVisSz ); Point aVisPt( Frm().Pos() + Prt().Pos() ); Point aGrfPt( aVisPt ); // Set the "visible" rectangle first if ( nLeftCrop > 0 ) { aVisPt.X() += nLeftCrop; aVisSz.Width() -= nLeftCrop; } if ( nTopCrop > 0 ) { aVisPt.Y() += nTopCrop; aVisSz.Height() -= nTopCrop; } if ( nRightCrop > 0 ) aVisSz.Width() -= nRightCrop; if ( nBottomCrop > 0 ) aVisSz.Height() -= nBottomCrop; rRect.Pos ( aVisPt ); rRect.SSize( aVisSz ); // Calculate the whole graphic if needed if ( pOrigRect ) { Size aTmpSz( aGrfSz ); aGrfPt.X() += nLeftCrop; aTmpSz.Width() -= nLeftCrop + nRightCrop; aGrfPt.Y() += nTopCrop; aTmpSz.Height()-= nTopCrop + nBottomCrop; if( RES_MIRROR_GRAPH_DONT != nMirror ) lcl_CalcRect( aGrfPt, aTmpSz, nMirror ); pOrigRect->Pos ( aGrfPt ); pOrigRect->SSize( aTmpSz ); } } /************************************************************************* |* |* Size SwNoTxtFrm::GetSize() |* |* By returning the surrounding Fly's size which equals the graphic's size |* *************************************************************************/ const Size& SwNoTxtFrm::GetSize() const { // Return the Frame's size const SwFrm *pFly = FindFlyFrm(); if( !pFly ) pFly = this; return pFly->Prt().SSize(); } /************************************************************************* |* |* SwNoTxtFrm::MakeAll() |* *************************************************************************/ void SwNoTxtFrm::MakeAll() { SwCntntNotify aNotify( this ); SwBorderAttrAccess aAccess( SwFrm::GetCache(), this ); const SwBorderAttrs &rAttrs = *aAccess.Get(); while ( !bValidPos || !bValidSize || !bValidPrtArea ) { MakePos(); if ( !bValidSize ) Frm().Width( GetUpper()->Prt().Width() ); MakePrtArea( rAttrs ); if ( !bValidSize ) { bValidSize = sal_True; Format(); } } } /************************************************************************* |* |* SwNoTxtFrm::Format() |* |* Calculate the Bitmap's site, if needed |* *************************************************************************/ void SwNoTxtFrm::Format( const SwBorderAttrs * ) { const Size aNewSize( GetSize() ); // Did the height change? SwTwips nChgHght = IsVertical() ? (SwTwips)(aNewSize.Width() - Prt().Width()) : (SwTwips)(aNewSize.Height() - Prt().Height()); if( nChgHght > 0) Grow( nChgHght ); else if( nChgHght < 0) Shrink( Min(Prt().Height(), -nChgHght) ); } /************************************************************************* |* |* SwNoTxtFrm::GetCharRect() |* |*************************************************************************/ sal_Bool SwNoTxtFrm::GetCharRect( SwRect &rRect, const SwPosition& rPos, SwCrsrMoveState *pCMS ) const { if ( &rPos.nNode.GetNode() != (SwNode*)GetNode() ) return sal_False; Calc(); SwRect aFrameRect( Frm() ); rRect = aFrameRect; rRect.Pos( Frm().Pos() + Prt().Pos() ); rRect.SSize( Prt().SSize() ); rRect.Justify(); // Is the Bitmap in the visible area at all? if( !aFrameRect.IsOver( rRect ) ) { // If not, then the Cursor is on the Frame rRect = aFrameRect; rRect.Width( 1 ); } else rRect._Intersection( aFrameRect ); if ( pCMS ) { if ( pCMS->bRealHeight ) { pCMS->aRealHeight.Y() = rRect.Height(); pCMS->aRealHeight.X() = 0; } } return sal_True; } sal_Bool SwNoTxtFrm::GetCrsrOfst(SwPosition* pPos, Point& , SwCrsrMoveState* ) const { SwCntntNode* pCNd = (SwCntntNode*)GetNode(); pPos->nNode = *pCNd; pPos->nContent.Assign( pCNd, 0 ); return sal_True; } #define CLEARCACHE( pNd ) {\ (pNd)->GetGrfObj().ReleaseFromCache();\ SwFlyFrm* pFly = FindFlyFrm();\ if( pFly && pFly->GetFmt()->GetSurround().IsContour() )\ {\ ClrContourCache( pFly->GetVirtDrawObj() );\ pFly->NotifyBackground( FindPageFrm(), Prt(), PREP_FLY_ATTR_CHG );\ }\ } void SwNoTxtFrm::Modify( const SfxPoolItem* pOld, const SfxPoolItem* pNew ) { sal_uInt16 nWhich = pNew ? pNew->Which() : pOld ? pOld->Which() : 0; // #i73788# // no for RES_LINKED_GRAPHIC_STREAM_ARRIVED if ( RES_GRAPHIC_PIECE_ARRIVED != nWhich && RES_GRAPHIC_ARRIVED != nWhich && RES_GRF_REREAD_AND_INCACHE != nWhich && RES_LINKED_GRAPHIC_STREAM_ARRIVED != nWhich ) { SwCntntFrm::Modify( pOld, pNew ); } bool bComplete = true; switch( nWhich ) { case RES_OBJECTDYING: break; case RES_GRF_REREAD_AND_INCACHE: if( ND_GRFNODE == GetNode()->GetNodeType() ) { bComplete = false; SwGrfNode* pNd = (SwGrfNode*) GetNode(); ViewShell *pVSh = 0; pNd->GetDoc()->GetEditShell( &pVSh ); if( pVSh ) { GraphicAttr aAttr; if( pNd->GetGrfObj().IsCached( pVSh->GetOut(), Point(), Prt().SSize(), &pNd->GetGraphicAttr( aAttr, this ) )) { ViewShell *pSh = pVSh; do { SET_CURR_SHELL( pSh ); if( pSh->GetWin() ) { if( pSh->IsPreView() ) ::RepaintPagePreview( pSh, Frm().SVRect() ); else pSh->GetWin()->Invalidate( Frm().SVRect() ); } } while( pVSh != (pSh = (ViewShell*)pSh->GetNext() )); } else pNd->SwapIn(); } } break; case RES_UPDATE_ATTR: case RES_FMT_CHG: CLEARCACHE( (SwGrfNode*) GetNode() ) break; case RES_ATTRSET_CHG: { sal_uInt16 n; for( n = RES_GRFATR_BEGIN; n < RES_GRFATR_END; ++n ) if( SFX_ITEM_SET == ((SwAttrSetChg*)pOld)->GetChgSet()-> GetItemState( n, sal_False )) { CLEARCACHE( (SwGrfNode*) GetNode() ) break; } if( RES_GRFATR_END == n ) // not found return ; } break; case RES_GRAPHIC_PIECE_ARRIVED: case RES_GRAPHIC_ARRIVED: // i73788# - handle RES_LINKED_GRAPHIC_STREAM_ARRIVED as RES_GRAPHIC_ARRIVED case RES_LINKED_GRAPHIC_STREAM_ARRIVED: if ( GetNode()->GetNodeType() == ND_GRFNODE ) { bComplete = false; SwGrfNode* pNd = (SwGrfNode*) GetNode(); CLEARCACHE( pNd ) SwRect aRect( Frm() ); ViewShell *pVSh = 0; pNd->GetDoc()->GetEditShell( &pVSh ); if( !pVSh ) break; ViewShell *pSh = pVSh; do { SET_CURR_SHELL( pSh ); if( pSh->IsPreView() ) { if( pSh->GetWin() ) ::RepaintPagePreview( pSh, aRect ); } else if ( pSh->VisArea().IsOver( aRect ) && OUTDEV_WINDOW == pSh->GetOut()->GetOutDevType() ) { // invalidate instead of painting pSh->GetWin()->Invalidate( aRect.SVRect() ); } pSh = (ViewShell *)pSh->GetNext(); } while( pSh != pVSh ); } break; default: if ( !pNew || !isGRFATR(nWhich) ) return; } if( bComplete ) { InvalidatePrt(); SetCompletePaint(); } } static void lcl_correctlyAlignRect( SwRect& rAlignedGrfArea, const SwRect& rInArea, OutputDevice* pOut ) { if(!pOut) return; Rectangle aPxRect = pOut->LogicToPixel( rInArea.SVRect() ); Rectangle aNewPxRect( aPxRect ); while( aNewPxRect.Left() < aPxRect.Left() ) { rAlignedGrfArea.Left( rAlignedGrfArea.Left()+1 ); aNewPxRect = pOut->LogicToPixel( rAlignedGrfArea.SVRect() ); } while( aNewPxRect.Top() < aPxRect.Top() ) { rAlignedGrfArea.Top( rAlignedGrfArea.Top()+1 ); aNewPxRect = pOut->LogicToPixel( rAlignedGrfArea.SVRect() ); } while( aNewPxRect.Bottom() > aPxRect.Bottom() ) { rAlignedGrfArea.Bottom( rAlignedGrfArea.Bottom()-1 ); aNewPxRect = pOut->LogicToPixel( rAlignedGrfArea.SVRect() ); } while( aNewPxRect.Right() > aPxRect.Right() ) { rAlignedGrfArea.Right( rAlignedGrfArea.Right()-1 ); aNewPxRect = pOut->LogicToPixel( rAlignedGrfArea.SVRect() ); } } // Paint the graphic. // We require either a QuickDraw-Bitmap or a graphic here. If we do not have // either, we return a replacement. // delete unused 3rd parameter. // use aligned rectangle for drawing graphic. // pixel-align coordinations for drawing graphic. void SwNoTxtFrm::PaintPicture( OutputDevice* pOut, const SwRect &rGrfArea ) const { ViewShell* pShell = getRootFrm()->GetCurrShell(); SwNoTxtNode& rNoTNd = *(SwNoTxtNode*)GetNode(); SwGrfNode* pGrfNd = rNoTNd.GetGrfNode(); SwOLENode* pOLENd = rNoTNd.GetOLENode(); const sal_Bool bPrn = pOut == rNoTNd.getIDocumentDeviceAccess()->getPrinter( false ) || pOut->GetConnectMetaFile(); const bool bIsChart = pOLENd && ChartPrettyPainter::IsChart( pOLENd->GetOLEObj().GetObject() ); // calculate aligned rectangle from parameter . // Use aligned rectangle instead of in // the following code. SwRect aAlignedGrfArea = rGrfArea; ::SwAlignRect( aAlignedGrfArea, pShell ); if( !bIsChart ) { // Because for drawing a graphic left-top-corner and size coordinations are // used, these coordinations have to be determined on pixel level. ::SwAlignGrfRect( &aAlignedGrfArea, *pOut ); } else //if( bIsChart ) { // #i78025# charts own borders are not completely visible // the above pixel correction is not correct - at least not for charts // so a different pixel correction is chosen here // this might be a good idea for all other OLE objects also, // but as I cannot oversee the consequences I fix it only for charts for now lcl_correctlyAlignRect( aAlignedGrfArea, rGrfArea, pOut ); } if( pGrfNd ) { // Fix for bug fdo#33781 const sal_uInt16 nFormerAntialiasingAtOutput( pOut->GetAntialiasing() ); if (pShell->Imp()->GetDrawView()->IsAntiAliasing()) { pOut->SetAntialiasing( nFormerAntialiasingAtOutput | ANTIALIASING_ENABLE_B2DDRAW ); } bool bForceSwap = false, bContinue = true; GraphicObject& rGrfObj = pGrfNd->GetGrfObj(); GraphicAttr aGrfAttr; pGrfNd->GetGraphicAttr( aGrfAttr, this ); if( !bPrn ) { // #i73788# if ( pGrfNd->IsLinkedInputStreamReady() ) { pGrfNd->UpdateLinkWithInputStream(); } // #i85717#, #i90395# - check, if asynchronous retrieval // if input stream for the graphic is possible else if ( ( rGrfObj.GetType() == GRAPHIC_DEFAULT || rGrfObj.GetType() == GRAPHIC_NONE ) && pGrfNd->IsLinkedFile() && pGrfNd->IsAsyncRetrieveInputStreamPossible() ) { Size aTmpSz; ::sfx2::SvLinkSource* pGrfObj = pGrfNd->GetLink()->GetObj(); if( !pGrfObj || !pGrfObj->IsDataComplete() || !(aTmpSz = pGrfNd->GetTwipSize()).Width() || !aTmpSz.Height() || !pGrfNd->GetAutoFmtLvl() ) { pGrfNd->TriggerAsyncRetrieveInputStream(); // #i73788# } String aTxt( pGrfNd->GetTitle() ); if ( !aTxt.Len() ) GetRealURL( *pGrfNd, aTxt ); ::lcl_PaintReplacement( aAlignedGrfArea, aTxt, *pShell, this, sal_False ); bContinue = false; } else if( rGrfObj.IsCached( pOut, aAlignedGrfArea.Pos(), aAlignedGrfArea.SSize(), &aGrfAttr )) { rGrfObj.DrawWithPDFHandling( *pOut, aAlignedGrfArea.Pos(), aAlignedGrfArea.SSize(), &aGrfAttr ); bContinue = false; } } if( bContinue ) { const sal_Bool bSwapped = rGrfObj.IsSwappedOut(); const bool bSwappedIn = 0 != pGrfNd->SwapIn( bPrn ); if( bSwappedIn && rGrfObj.GetGraphic().IsSupportedGraphic()) { const bool bAnimate = rGrfObj.IsAnimated() && !pShell->IsPreView() && !pShell->GetAccessibilityOptions()->IsStopAnimatedGraphics() && // #i9684# Stop animation during printing/pdf export pShell->GetWin(); if( bAnimate && FindFlyFrm() != ::GetFlyFromMarked( 0, pShell )) { OutputDevice* pVout; if( pOut == pShell->GetOut() && SwRootFrm::FlushVout() ) pVout = pOut, pOut = pShell->GetOut(); else if( pShell->GetWin() && OUTDEV_VIRDEV == pOut->GetOutDevType() ) pVout = pOut, pOut = pShell->GetWin(); else pVout = 0; OSL_ENSURE( OUTDEV_VIRDEV != pOut->GetOutDevType() || pShell->GetViewOptions()->IsPDFExport(), "pOut should not be a virtual device" ); rGrfObj.StartAnimation( pOut, aAlignedGrfArea.Pos(), aAlignedGrfArea.SSize(), long(this), 0, GRFMGR_DRAW_STANDARD, pVout ); } else rGrfObj.DrawWithPDFHandling( *pOut, aAlignedGrfArea.Pos(), aAlignedGrfArea.SSize(), &aGrfAttr ); } else { sal_uInt16 nResId = 0; if( bSwappedIn ) { if( GRAPHIC_NONE == rGrfObj.GetType() ) nResId = STR_COMCORE_READERROR; else if ( !rGrfObj.GetGraphic().IsSupportedGraphic() ) nResId = STR_COMCORE_CANT_SHOW; } ((SwNoTxtFrm*)this)->nWeight = -1; String aText; if ( !nResId && !(aText = pGrfNd->GetTitle()).Len() && (!GetRealURL( *pGrfNd, aText ) || !aText.Len())) { nResId = STR_COMCORE_READERROR; } if ( nResId ) aText = SW_RESSTR( nResId ); ::lcl_PaintReplacement( aAlignedGrfArea, aText, *pShell, this, sal_True ); } // When printing, we must not collect the graphics if( bSwapped && bPrn ) bForceSwap = true; } if( bForceSwap ) pGrfNd->SwapOut(); if ( pShell->Imp()->GetDrawView()->IsAntiAliasing() ) pOut->SetAntialiasing( nFormerAntialiasingAtOutput ); } else if( bIsChart // Charts must be painted resolution dependent!! #i82893#, #i75867# && ChartPrettyPainter::ShouldPrettyPaintChartOnThisDevice( pOut ) && svt::EmbeddedObjectRef::TryRunningState( pOLENd->GetOLEObj().GetOleRef() ) && ChartPrettyPainter::DoPrettyPaintChart( uno::Reference< frame::XModel >( pOLENd->GetOLEObj().GetOleRef()->getComponent(), uno::UNO_QUERY), pOut, aAlignedGrfArea.SVRect() ) ) { (void)(0);// all was done in if statement } else if( pOLENd ) { // Fix for bug fdo#33781 const sal_uInt16 nFormerAntialiasingAtOutput( pOut->GetAntialiasing() ); if (pShell->Imp()->GetDrawView()->IsAntiAliasing()) { sal_uInt16 nNewAntialiasingAtOutput = nFormerAntialiasingAtOutput | ANTIALIASING_ENABLE_B2DDRAW; // #i99665# // Adjust AntiAliasing mode at output device for chart OLE if ( pOLENd->IsChart() ) nNewAntialiasingAtOutput |= ANTIALIASING_PIXELSNAPHAIRLINE; pOut->SetAntialiasing( nNewAntialiasingAtOutput ); } Point aPosition(aAlignedGrfArea.Pos()); Size aSize(aAlignedGrfArea.SSize()); // We do not have a printer in the BrowseMode and thus no JobSetup, so we create one ... const JobSetup* pJobSetup = pOLENd->getIDocumentDeviceAccess()->getJobsetup(); bool bDummyJobSetup = 0 == pJobSetup; if( bDummyJobSetup ) pJobSetup = new JobSetup(); // #i42323# //TODO/LATER: is it a problem that the JopSetup isn't used? //xRef->DoDraw( pOut, aAlignedGrfArea.Pos(), aAlignedGrfArea.SSize(), *pJobSetup ); Graphic* pGraphic = pOLENd->GetGraphic(); if ( pGraphic && pGraphic->GetType() != GRAPHIC_NONE ) { pGraphic->Draw( pOut, aPosition, aSize ); // shade the representation if the object is activated outplace uno::Reference < embed::XEmbeddedObject > xObj = pOLENd->GetOLEObj().GetOleRef(); if ( xObj.is() && xObj->getCurrentState() == embed::EmbedStates::ACTIVE ) { ::svt::EmbeddedObjectRef::DrawShading( Rectangle( aPosition, aSize ), pOut ); } } else ::svt::EmbeddedObjectRef::DrawPaintReplacement( Rectangle( aPosition, aSize ), pOLENd->GetOLEObj().GetCurrentPersistName(), pOut ); if( bDummyJobSetup ) delete pJobSetup; // ... and clean it up again sal_Int64 nMiscStatus = pOLENd->GetOLEObj().GetOleRef()->getStatus( pOLENd->GetAspect() ); if ( !bPrn && pShell->ISA( SwCrsrShell ) && nMiscStatus & embed::EmbedMisc::MS_EMBED_ACTIVATEWHENVISIBLE ) { const SwFlyFrm *pFly = FindFlyFrm(); OSL_ENSURE( pFly, "OLE not in FlyFrm" ); ((SwFEShell*)pShell)->ConnectObj( pOLENd->GetOLEObj().GetObject(), pFly->Prt(), pFly->Frm()); } if ( pShell->Imp()->GetDrawView()->IsAntiAliasing() ) pOut->SetAntialiasing( nFormerAntialiasingAtOutput ); } } sal_Bool SwNoTxtFrm::IsTransparent() const { const ViewShell* pSh = getRootFrm()->GetCurrShell(); if ( !pSh || !pSh->GetViewOptions()->IsGraphic() ) return sal_True; const SwGrfNode *pNd; if( 0 != (pNd = GetNode()->GetGrfNode()) ) return pNd->IsTransparent(); //#29381# OLE are always transparent return sal_True; } void SwNoTxtFrm::StopAnimation( OutputDevice* pOut ) const { // Stop animated graphics SwGrfNode* pGrfNd = (SwGrfNode*)GetNode()->GetGrfNode(); if( pGrfNd && pGrfNd->IsAnimated() ) pGrfNd->GetGrfObj().StopAnimation( pOut, long(this) ); } sal_Bool SwNoTxtFrm::HasAnimation() const { const SwGrfNode* pGrfNd = GetNode()->GetGrfNode(); return pGrfNd && pGrfNd->IsAnimated(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */