/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include "svx/svditer.hxx" #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; // interface to SdrPageWindow SdrPageWindow* SdrPageView::FindPageWindow(SdrPaintWindow& rPaintWindow) const { for(SdrPageWindowVector::const_iterator a = maPageWindows.begin(); a != maPageWindows.end(); ++a) { if(&((*a)->GetPaintWindow()) == &rPaintWindow) { return *a; } } return 0L; } const SdrPageWindow* SdrPageView::FindPatchedPageWindow( const OutputDevice& _rOutDev ) const { for ( SdrPageWindowVector::const_iterator loop = maPageWindows.begin(); loop != maPageWindows.end(); ++loop ) { const SdrPageWindow& rPageWindow( *(*loop) ); const SdrPaintWindow& rPaintWindow( rPageWindow.GetOriginalPaintWindow() ? *rPageWindow.GetOriginalPaintWindow() : rPageWindow.GetPaintWindow() ); if ( &rPaintWindow.GetOutputDevice() == &_rOutDev ) { return &rPageWindow; } } return NULL; } SdrPageWindow* SdrPageView::FindPageWindow(const OutputDevice& rOutDev) const { for(SdrPageWindowVector::const_iterator a = maPageWindows.begin(); a != maPageWindows.end(); ++a) { if(&((*a)->GetPaintWindow().GetOutputDevice()) == &rOutDev) { return *a; } } return 0L; } SdrPageWindow* SdrPageView::GetPageWindow(sal_uInt32 nIndex) const { if(nIndex < maPageWindows.size()) { return maPageWindows[nIndex]; } return 0L; } void SdrPageView::ClearPageWindows() { for(SdrPageWindowVector::const_iterator a = maPageWindows.begin(); a != maPageWindows.end(); ++a) { delete *a; } maPageWindows.clear(); } void SdrPageView::AppendPageWindow(SdrPageWindow& rNew) { maPageWindows.push_back(&rNew); } SdrPageWindow* SdrPageView::RemovePageWindow(SdrPageWindow& rOld) { const SdrPageWindowVector::iterator aFindResult = ::std::find(maPageWindows.begin(), maPageWindows.end(), &rOld); if(aFindResult != maPageWindows.end()) { // remember return value SdrPageWindow* pSdrPageWindow = *aFindResult; maPageWindows.erase(aFindResult); return pSdrPageWindow; } return 0L; } SdrPageView::SdrPageView(SdrPage* pPage1, SdrView& rNewView) : mrView(rNewView), // col_auto color lets the view takes the default SvxColorConfig entry maDocumentColor( COL_AUTO ), maBackgroundColor(COL_AUTO ), // #i48367# also react on autocolor mpPreparedPageWindow(0) // #i72752# { mpPage = pPage1; if(mpPage) { aPgOrg.X()=mpPage->GetLftBorder(); aPgOrg.Y()=mpPage->GetUppBorder(); } mbHasMarked = false; aLayerVisi.SetAll(); aLayerPrn.SetAll(); mbVisible = false; pAktList = NULL; pAktGroup = NULL; SetAktGroupAndList(NULL, mpPage); for(sal_uInt32 a(0L); a < rNewView.PaintWindowCount(); a++) { AddPaintWindowToPageView(*rNewView.GetPaintWindow(a)); } } SdrPageView::~SdrPageView() { // cleanup window vector ClearPageWindows(); } SdrPageWindow& SdrPageView::CreateNewPageWindowEntry(SdrPaintWindow& rPaintWindow) { SdrPageWindow& rWindow = *(new SdrPageWindow(*this, rPaintWindow)); AppendPageWindow(rWindow); return rWindow; } void SdrPageView::AddPaintWindowToPageView(SdrPaintWindow& rPaintWindow) { if(!FindPageWindow(rPaintWindow)) { CreateNewPageWindowEntry(rPaintWindow); } } void SdrPageView::RemovePaintWindowFromPageView(SdrPaintWindow& rPaintWindow) { SdrPageWindow* pCandidate = FindPageWindow(rPaintWindow); if(pCandidate) { pCandidate = RemovePageWindow(*pCandidate); if(pCandidate) { delete pCandidate; } } } ::com::sun::star::uno::Reference< ::com::sun::star::awt::XControlContainer > SdrPageView::GetControlContainer( const OutputDevice& _rDevice ) const { ::com::sun::star::uno::Reference< ::com::sun::star::awt::XControlContainer > xReturn; const SdrPageWindow* pCandidate = FindPatchedPageWindow( _rDevice ); if ( pCandidate ) xReturn = pCandidate->GetControlContainer( true ); return xReturn; } void SdrPageView::ModelHasChanged() { if (GetAktGroup()!=NULL) CheckAktGroup(); } bool SdrPageView::IsReadOnly() const { return (0L == GetPage() || GetView().GetModel()->IsReadOnly() || GetPage()->IsReadOnly() || GetObjList()->IsReadOnly()); } void SdrPageView::Show() { if(!IsVisible()) { mbVisible = true; InvalidateAllWin(); for(sal_uInt32 a(0L); a < GetView().PaintWindowCount(); a++) { AddPaintWindowToPageView(*GetView().GetPaintWindow(a)); } } } void SdrPageView::Hide() { if(IsVisible()) { InvalidateAllWin(); mbVisible = false; ClearPageWindows(); } } Rectangle SdrPageView::GetPageRect() const { if (GetPage()==NULL) return Rectangle(); return Rectangle(Point(),Size(GetPage()->GetWdt()+1,GetPage()->GetHgt()+1)); } void SdrPageView::InvalidateAllWin() { if(IsVisible() && GetPage()) { Rectangle aRect(Point(0,0),Size(GetPage()->GetWdt()+1,GetPage()->GetHgt()+1)); aRect.Union(GetPage()->GetAllObjBoundRect()); GetView().InvalidateAllWin(aRect); } } void SdrPageView::PrePaint() { const sal_uInt32 nCount(PageWindowCount()); for(sal_uInt32 a(0); a < nCount; a++) { SdrPageWindow* pCandidate = GetPageWindow(a); if(pCandidate) { pCandidate->PrePaint(); } } } void SdrPageView::CompleteRedraw( SdrPaintWindow& rPaintWindow, const vcl::Region& rReg, sdr::contact::ViewObjectContactRedirector* pRedirector ) { if(GetPage()) { SdrPageWindow* pPageWindow = FindPageWindow(rPaintWindow); bool bIsTempTarget(false); if(!pPageWindow) { // create temp PageWindow pPageWindow = new SdrPageWindow(*((SdrPageView*)this), rPaintWindow); bIsTempTarget = true; } // do the redraw pPageWindow->PrepareRedraw(rReg); pPageWindow->RedrawAll(pRedirector); // get rid of temp PageWindow if(bIsTempTarget) { delete pPageWindow; pPageWindow = 0L; } } } // #i74769# use SdrPaintWindow directly void SdrPageView::setPreparedPageWindow(SdrPageWindow* pKnownTarget) { // #i72752# remember prepared SdrPageWindow mpPreparedPageWindow = pKnownTarget; } void SdrPageView::DrawLayer( SdrLayerID nID, OutputDevice* pGivenTarget, sdr::contact::ViewObjectContactRedirector* pRedirector, const Rectangle& rRect ) { if(GetPage()) { if(pGivenTarget) { SdrPageWindow* pKnownTarget = FindPageWindow(*pGivenTarget); if(pKnownTarget) { // paint known target pKnownTarget->RedrawLayer(&nID, pRedirector); } else { // #i72752# DrawLayer() uses a OutputDevice different from BeginDrawLayer. This happens // e.g. when SW paints a single text line in text edit mode. Try to use it SdrPageWindow* pPreparedTarget = mpPreparedPageWindow; if(pPreparedTarget) { // if we have a prepared target, do not use a new SdrPageWindow since this // works but is expensive. Just use a temporary PaintWindow SdrPaintWindow aTemporaryPaintWindow(mrView, *pGivenTarget); // Copy existing paint region to use the same as prepared in BeginDrawLayer SdrPaintWindow& rExistingPaintWindow = pPreparedTarget->GetPaintWindow(); const vcl::Region& rExistingRegion = rExistingPaintWindow.GetRedrawRegion(); bool bUseRect(false); if (!rRect.IsEmpty()) { vcl::Region r(rExistingRegion); r.Intersect(rRect); // fdo#74435: FIXME: visibility check broken if empty if (!r.IsEmpty()) bUseRect = true; } if (!bUseRect) aTemporaryPaintWindow.SetRedrawRegion(rExistingRegion); else aTemporaryPaintWindow.SetRedrawRegion(vcl::Region(rRect)); // patch the ExistingPageWindow pPreparedTarget->patchPaintWindow(aTemporaryPaintWindow); // redraw the layer pPreparedTarget->RedrawLayer(&nID, pRedirector); // restore the ExistingPageWindow pPreparedTarget->unpatchPaintWindow(); } else { OSL_FAIL("SdrPageView::DrawLayer: Creating temporary SdrPageWindow (ObjectContact), this should never be needed (!)"); // None of the known OutputDevices is the target of this paint, use // a temporary SdrPageWindow for this Redraw. SdrPaintWindow aTemporaryPaintWindow(mrView, *pGivenTarget); SdrPageWindow aTemporaryPageWindow(*((SdrPageView*)this), aTemporaryPaintWindow); // #i72752# // Copy existing paint region if other PageWindows exist, this was created by // PrepareRedraw() from BeginDrawLayer(). Needs to be used e.g. when suddenly SW // paints into an unknown device other than the view was created for (e.g. VirtualDevice) if(PageWindowCount()) { SdrPageWindow* pExistingPageWindow = GetPageWindow(0L); SdrPaintWindow& rExistingPaintWindow = pExistingPageWindow->GetPaintWindow(); const vcl::Region& rExistingRegion = rExistingPaintWindow.GetRedrawRegion(); aTemporaryPaintWindow.SetRedrawRegion(rExistingRegion); } aTemporaryPageWindow.RedrawLayer(&nID, pRedirector); } } } else { // paint in all known windows for(sal_uInt32 a(0L); a < PageWindowCount(); a++) { SdrPageWindow* pTarget = GetPageWindow(a); pTarget->RedrawLayer(&nID, pRedirector); } } } } void SdrPageView::SetDesignMode( bool _bDesignMode ) const { for ( sal_uInt32 i = 0L; i < PageWindowCount(); ++i ) { const SdrPageWindow& rPageViewWindow = *GetPageWindow(i); rPageViewWindow.SetDesignMode( _bDesignMode ); } } void SdrPageView::DrawPageViewGrid(OutputDevice& rOut, const Rectangle& rRect, Color aColor) { if (GetPage()==NULL) return; long nx1=GetView().aGridBig.Width(); long nx2=GetView().aGridFin.Width(); long ny1=GetView().aGridBig.Height(); long ny2=GetView().aGridFin.Height(); if (nx1==0) nx1=nx2; if (nx2==0) nx2=nx1; if (ny1==0) ny1=ny2; if (ny2==0) ny2=ny1; if (nx1==0) { nx1=ny1; nx2=ny2; } if (ny1==0) { ny1=nx1; ny2=nx2; } if (nx1<0) nx1=-nx1; if (nx2<0) nx2=-nx2; if (ny1<0) ny1=-ny1; if (ny2<0) ny2=-ny2; if (nx1!=0) { // no more global output size, use window size instead to decide grid sizes long nScreenWdt = rOut.GetOutputSizePixel().Width(); long nMinDotPix=2; long nMinLinPix=4; if (nScreenWdt>=1600) { nMinDotPix=4; nMinLinPix=8; } else if (nScreenWdt>=1024) { nMinDotPix=3; nMinLinPix=6; } else { // e. g. 640x480 nMinDotPix=2; nMinLinPix=4; } Size aMinDotDist(rOut.PixelToLogic(Size(nMinDotPix,nMinDotPix))); Size aMinLinDist(rOut.PixelToLogic(Size(nMinLinPix,nMinLinPix))); bool bHoriSolid=nx2 nx1*=2.5 if (nTgl==2) nx1*=2; nVal0=a; nTgl++; if (nTgl>=3) nTgl=0; } nTgl=0; nVal0=ny1; while (ny1 ny1*=2.5 if (nTgl==2) ny1*=2; nVal0=a; nTgl++; if (nTgl>=3) nTgl=0; } bool bHoriFine=nx2GetLftBorder()+1+nWrX; long x2=GetPage()->GetWdt()-GetPage()->GetRgtBorder()-1+nWrY; long y1=GetPage()->GetUppBorder()+1+nWrX; long y2=GetPage()->GetHgt()-GetPage()->GetLwrBorder()-1+nWrY; const SdrPageGridFrameList* pFrames=GetPage()->GetGridFrameList(this,NULL); sal_uInt16 nGridPaintAnz=1; if (pFrames!=NULL) nGridPaintAnz=pFrames->GetCount(); for (sal_uInt16 nGridPaintNum=0; nGridPaintNumrRect.Right() +nX1Pix) x2=rRect.Right() +nX1Pix; if (y1rRect.Bottom()+nY1Pix) y2=rRect.Bottom()+nY1Pix; } long xBigOrg=aOrg.X()+nWrX; while (xBigOrg>=x1) xBigOrg-=nx1; while (xBigOrg=x1) xFinOrg-=nx2; while (xFinOrg=y1) yBigOrg-=ny1; while (yBigOrg=y1) yFinOrg-=ny2; while (yFinOrg= 1000) { nStepOffset -= 1000; nPointOffset++; } } } if( bVertLines ) { sal_uIntPtr nGridFlags = ( bVertSolid ? GRID_VERTLINES : GRID_DOTS ); sal_uInt16 nSteps = sal_uInt16(ny1 / ny2); sal_uInt32 nRestPerStepMul1000 = nSteps ? ( ((ny1 * 1000L)/ nSteps) - (ny2 * 1000L) ) : 0; sal_uInt32 nStepOffset = 0; sal_uInt16 nPointOffset = 0; for(sal_uInt16 a=0;a= 1000) { nStepOffset -= 1000; nPointOffset++; } } } } } rOut.EnableMapMode(bMap0); rOut.SetLineColor(aColorMerk); } } void SdrPageView::AdjHdl() { GetView().AdjustMarkHdl(); } void SdrPageView::SetLayer(const OUString& rName, SetOfByte& rBS, bool bJa) { if(!GetPage()) return; SdrLayerID nID = GetPage()->GetLayerAdmin().GetLayerID(rName, true); if(SDRLAYER_NOTFOUND != nID) rBS.Set(nID, bJa); } bool SdrPageView::IsLayer(const OUString& rName, const SetOfByte& rBS) const { if(!GetPage()) return false; bool bRet(false); if (!rName.isEmpty()) { SdrLayerID nId = GetPage()->GetLayerAdmin().GetLayerID(rName, true); if(SDRLAYER_NOTFOUND != nId) { bRet = rBS.IsSet(nId); } } return bRet; } bool SdrPageView::IsObjMarkable(SdrObject* pObj) const { if(pObj) { // excluded from selection? if(pObj->IsMarkProtect()) { return false; } // only visible are selectable if( !pObj->IsVisible() ) { return false; } if(pObj->ISA(SdrObjGroup)) { // If object is a Group object, visibility may depend on // multiple layers. If one object is markable, Group is markable. SdrObjList* pObjList = static_cast(pObj)->GetSubList(); if(pObjList && pObjList->GetObjCount()) { bool bGroupIsMarkable(false); for(size_t a = 0; !bGroupIsMarkable && a < pObjList->GetObjCount(); ++a) { SdrObject* pCandidate = pObjList->GetObj(a); // call recursively if(IsObjMarkable(pCandidate)) { bGroupIsMarkable = true; } } return bGroupIsMarkable; } else { // #i43302# // Allow empty groups to be selected to be able to delete them return true; } } else { // the layer has to be visible and must not be locked SdrLayerID nL = pObj->GetLayer(); return (aLayerVisi.IsSet(sal_uInt8(nL)) && !aLayerLock.IsSet(sal_uInt8(nL))); } } return false; } void SdrPageView::SetPageOrigin(const Point& rOrg) { if (rOrg!=aPgOrg) { aPgOrg=rOrg; if (GetView().IsGridVisible()) { InvalidateAllWin(); } } } void SdrPageView::ImpInvalidateHelpLineArea(sal_uInt16 nNum) const { if (GetView().IsHlplVisible() && nNumOutputToWindow()) { OutputDevice& rOutDev = pCandidate->GetOutputDevice(); Rectangle aR(rHL.GetBoundRect(rOutDev)); Size aSiz(rOutDev.PixelToLogic(Size(1,1))); aR.Left() -= aSiz.Width(); aR.Right() += aSiz.Width(); aR.Top() -= aSiz.Height(); aR.Bottom() += aSiz.Height(); ((SdrView&)GetView()).InvalidateOneWin(static_cast(rOutDev), aR); } } } } void SdrPageView::SetHelpLines(const SdrHelpLineList& rHLL) { aHelpLines=rHLL; InvalidateAllWin(); } void SdrPageView::SetHelpLine(sal_uInt16 nNum, const SdrHelpLine& rNewHelpLine) { if (nNum aHelpLines.GetCount()) nNum = aHelpLines.GetCount(); aHelpLines.Insert(rHL,nNum); if (GetView().IsHlplVisible()) ImpInvalidateHelpLineArea(nNum); } // set current group and list void SdrPageView::SetAktGroupAndList(SdrObject* pNewGroup, SdrObjList* pNewList) { if(pAktGroup != pNewGroup) { pAktGroup = pNewGroup; } if(pAktList != pNewList) { pAktList = pNewList; } } bool SdrPageView::EnterGroup(SdrObject* pObj) { bool bRet(false); if(pObj && pObj->IsGroupObject()) { bool bGlueInvalidate(GetView().ImpIsGlueVisible()); if(bGlueInvalidate) { GetView().GlueInvalidate(); } // deselect all GetView().UnmarkAll(); // set current group and list SdrObjList* pNewObjList = pObj->GetSubList(); SetAktGroupAndList(pObj, pNewObjList); // select contained object if only one object is contained, // else select nothing and let the user decide what to do next if(pNewObjList && pNewObjList->GetObjCount() == 1) { SdrObject* pFirstObject = pNewObjList->GetObj(0); if(GetView().GetSdrPageView()) { GetView().MarkObj(pFirstObject, GetView().GetSdrPageView()); } } // build new handles GetView().AdjustMarkHdl(); // invalidate only when view wants to visualize group entering if(GetView().DoVisualizeEnteredGroup()) { InvalidateAllWin(); } if (bGlueInvalidate) { GetView().GlueInvalidate(); } bRet = true; } return bRet; } void SdrPageView::LeaveOneGroup() { if(GetAktGroup()) { bool bGlueInvalidate = (GetView().ImpIsGlueVisible()); if(bGlueInvalidate) GetView().GlueInvalidate(); SdrObject* pLastGroup = GetAktGroup(); SdrObject* pParentGroup = GetAktGroup()->GetUpGroup(); SdrObjList* pParentList = GetPage(); if(pParentGroup) pParentList = pParentGroup->GetSubList(); // deselect everything GetView().UnmarkAll(); // allocations, pAktGroup and pAktList need to be set SetAktGroupAndList(pParentGroup, pParentList); // select the group we just left if(pLastGroup) if(GetView().GetSdrPageView()) GetView().MarkObj(pLastGroup, GetView().GetSdrPageView()); GetView().AdjustMarkHdl(); // invalidate only if view wants to visualize group entering if(GetView().DoVisualizeEnteredGroup()) InvalidateAllWin(); if(bGlueInvalidate) GetView().GlueInvalidate(); } } void SdrPageView::LeaveAllGroup() { if(GetAktGroup()) { bool bGlueInvalidate = (GetView().ImpIsGlueVisible()); if(bGlueInvalidate) GetView().GlueInvalidate(); SdrObject* pLastGroup = GetAktGroup(); // deselect everything GetView().UnmarkAll(); // allocations, pAktGroup and pAktList always need to be set SetAktGroupAndList(NULL, GetPage()); // find and select uppermost group if(pLastGroup) { while(pLastGroup->GetUpGroup()) pLastGroup = pLastGroup->GetUpGroup(); if(GetView().GetSdrPageView()) GetView().MarkObj(pLastGroup, GetView().GetSdrPageView()); } GetView().AdjustMarkHdl(); // invalidate only when view wants to visualize group entering if(GetView().DoVisualizeEnteredGroup()) InvalidateAllWin(); if(bGlueInvalidate) GetView().GlueInvalidate(); } } sal_uInt16 SdrPageView::GetEnteredLevel() const { sal_uInt16 nCount=0; SdrObject* pGrp=GetAktGroup(); while (pGrp!=NULL) { nCount++; pGrp=pGrp->GetUpGroup(); } return nCount; } void SdrPageView::CheckAktGroup() { SdrObject* pGrp=GetAktGroup(); while (pGrp!=NULL && (!pGrp->IsInserted() || pGrp->GetObjList()==NULL || pGrp->GetPage()==NULL || pGrp->GetModel()==NULL)) { // anything outside of the borders? pGrp=pGrp->GetUpGroup(); } if (pGrp!=GetAktGroup()) { if (pGrp!=NULL) EnterGroup(pGrp); else LeaveAllGroup(); } } // Set background color for svx at SdrPageViews void SdrPageView::SetApplicationBackgroundColor(Color aBackgroundColor) { maBackgroundColor = aBackgroundColor; } // Set document color for svx at SdrPageViews void SdrPageView::SetApplicationDocumentColor(Color aDocumentColor) { maDocumentColor = aDocumentColor; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */