/* -*- 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 "svddrgm1.hxx" #include #include #include #include #include #include "svx/xattr.hxx" #include #include #include #include #include #include #include #include "svx/svdstr.hrc" #include "svdglob.hxx" #include #include #include #include #include #include #include #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 #include #include #include #include #include #include #include #include SdrDragEntry::SdrDragEntry() : mbAddToTransparent(false) { } SdrDragEntry::~SdrDragEntry() { } SdrDragEntryPolyPolygon::SdrDragEntryPolyPolygon(const basegfx::B2DPolyPolygon& rOriginalPolyPolygon) : SdrDragEntry(), maOriginalPolyPolygon(rOriginalPolyPolygon) { } SdrDragEntryPolyPolygon::~SdrDragEntryPolyPolygon() { } drawinglayer::primitive2d::Primitive2DContainer SdrDragEntryPolyPolygon::createPrimitive2DSequenceInCurrentState(SdrDragMethod& rDragMethod) { drawinglayer::primitive2d::Primitive2DContainer aRetval; if(maOriginalPolyPolygon.count()) { basegfx::B2DPolyPolygon aCopy(maOriginalPolyPolygon); const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer; rDragMethod.applyCurrentTransformationToPolyPolygon(aCopy); basegfx::BColor aColA(aSvtOptionsDrawinglayer.GetStripeColorA().getBColor()); basegfx::BColor aColB(aSvtOptionsDrawinglayer.GetStripeColorB().getBColor()); const double fStripeLength(aSvtOptionsDrawinglayer.GetStripeLength()); if(Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { aColA = aColB = Application::GetSettings().GetStyleSettings().GetHighlightColor().getBColor(); aColB.invert(); } aRetval.resize(2); aRetval[0] = new drawinglayer::primitive2d::PolyPolygonMarkerPrimitive2D( aCopy, aColA, aColB, fStripeLength); const basegfx::BColor aHilightColor(aSvtOptionsDrawinglayer.getHilightColor().getBColor()); const double fTransparence(aSvtOptionsDrawinglayer.GetTransparentSelectionPercent() * 0.01); aRetval[1] = new drawinglayer::primitive2d::PolyPolygonSelectionPrimitive2D( aCopy, aHilightColor, fTransparence, 3.0, false); } return aRetval; } SdrDragEntrySdrObject::SdrDragEntrySdrObject(const SdrObject& rOriginal, sdr::contact::ObjectContact& rObjectContact, bool bModify) : SdrDragEntry(), maOriginal(rOriginal), mpClone(nullptr), mrObjectContact(rObjectContact), mbModify(bModify) { // add SdrObject parts to transparent overlay stuff setAddToTransparent(true); } SdrDragEntrySdrObject::~SdrDragEntrySdrObject() { if(mpClone) { SdrObject::Free(mpClone); } } void SdrDragEntrySdrObject::prepareCurrentState(SdrDragMethod& rDragMethod) { // for the moment, i need to re-create the clone in all cases. I need to figure // out when clone and original have the same class, so that i can use operator= // in those cases if(mpClone) { SdrObject::Free(mpClone); mpClone = nullptr; } if(mbModify) { if(!mpClone) { mpClone = maOriginal.getFullDragClone(); } // apply original transformation, implemented at the DragMethods rDragMethod.applyCurrentTransformationToSdrObject(*mpClone); } } drawinglayer::primitive2d::Primitive2DContainer SdrDragEntrySdrObject::createPrimitive2DSequenceInCurrentState(SdrDragMethod&) { const SdrObject* pSource = &maOriginal; if(mbModify && mpClone) { // choose source for geometry data pSource = mpClone; } // get VOC and Primitive2DContainer sdr::contact::ViewContact& rVC = pSource->GetViewContact(); sdr::contact::ViewObjectContact& rVOC = rVC.GetViewObjectContact(mrObjectContact); sdr::contact::DisplayInfo aDisplayInfo; // Do not use the last ViewPort set at the OC from the last ProcessDisplay(), // here we want the complete primitive sequence without visibility clippings mrObjectContact.resetViewPort(); return rVOC.getPrimitive2DSequenceHierarchy(aDisplayInfo); } SdrDragEntryPrimitive2DSequence::SdrDragEntryPrimitive2DSequence( const drawinglayer::primitive2d::Primitive2DContainer& rSequence) : SdrDragEntry(), maPrimitive2DSequence(rSequence) { // add parts to transparent overlay stuff if necessary setAddToTransparent(true); } SdrDragEntryPrimitive2DSequence::~SdrDragEntryPrimitive2DSequence() { } drawinglayer::primitive2d::Primitive2DContainer SdrDragEntryPrimitive2DSequence::createPrimitive2DSequenceInCurrentState(SdrDragMethod& rDragMethod) { drawinglayer::primitive2d::Primitive2DReference aTransformPrimitive2D( new drawinglayer::primitive2d::TransformPrimitive2D( rDragMethod.getCurrentTransformation(), maPrimitive2DSequence)); return drawinglayer::primitive2d::Primitive2DContainer { aTransformPrimitive2D }; } SdrDragEntryPointGlueDrag::SdrDragEntryPointGlueDrag(const std::vector< basegfx::B2DPoint >& rPositions, bool bIsPointDrag) : maPositions(rPositions), mbIsPointDrag(bIsPointDrag) { // add SdrObject parts to transparent overlay stuff setAddToTransparent(true); } SdrDragEntryPointGlueDrag::~SdrDragEntryPointGlueDrag() { } drawinglayer::primitive2d::Primitive2DContainer SdrDragEntryPointGlueDrag::createPrimitive2DSequenceInCurrentState(SdrDragMethod& rDragMethod) { drawinglayer::primitive2d::Primitive2DContainer aRetval; if(!maPositions.empty()) { basegfx::B2DPolygon aPolygon; sal_uInt32 a(0); for(a = 0; a < maPositions.size(); a++) { aPolygon.append(maPositions[a]); } basegfx::B2DPolyPolygon aPolyPolygon(aPolygon); rDragMethod.applyCurrentTransformationToPolyPolygon(aPolyPolygon); const basegfx::B2DPolygon aTransformed(aPolyPolygon.getB2DPolygon(0)); std::vector< basegfx::B2DPoint > aTransformedPositions; aTransformedPositions.reserve(aTransformed.count()); for(a = 0; a < aTransformed.count(); a++) { aTransformedPositions.push_back(aTransformed.getB2DPoint(a)); } if(mbIsPointDrag) { const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer; basegfx::BColor aColor(aSvtOptionsDrawinglayer.GetStripeColorA().getBColor()); if(Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { aColor = Application::GetSettings().GetStyleSettings().GetHighlightColor().getBColor(); } drawinglayer::primitive2d::Primitive2DReference aMarkerArrayPrimitive2D( new drawinglayer::primitive2d::MarkerArrayPrimitive2D(aTransformedPositions, drawinglayer::primitive2d::createDefaultCross_3x3(aColor))); aRetval = drawinglayer::primitive2d::Primitive2DContainer { aMarkerArrayPrimitive2D }; } else { drawinglayer::primitive2d::Primitive2DReference aMarkerArrayPrimitive2D( new drawinglayer::primitive2d::MarkerArrayPrimitive2D(aTransformedPositions, SdrHdl::createGluePointBitmap())); aRetval = drawinglayer::primitive2d::Primitive2DContainer { aMarkerArrayPrimitive2D }; } } return aRetval; } void SdrDragMethod::resetSdrDragEntries() { // clear entries; creation is on demand clearSdrDragEntries(); } basegfx::B2DRange SdrDragMethod::getCurrentRange() const { return maOverlayObjectList.getBaseRange(); } void SdrDragMethod::clearSdrDragEntries() { for(SdrDragEntry* p : maSdrDragEntries) { delete p; } maSdrDragEntries.clear(); } void SdrDragMethod::addSdrDragEntry(SdrDragEntry* pNew) { if(pNew) { maSdrDragEntries.push_back(pNew); } } void SdrDragMethod::createSdrDragEntries() { if(getSdrDragView().GetSdrPageView() && getSdrDragView().GetSdrPageView()->HasMarkedObjPageView()) { if(getSdrDragView().IsDraggingPoints()) { createSdrDragEntries_PointDrag(); } else if(getSdrDragView().IsDraggingGluePoints()) { createSdrDragEntries_GlueDrag(); } else { if(getSolidDraggingActive()) { createSdrDragEntries_SolidDrag(); } else { createSdrDragEntries_PolygonDrag(); } } } } void SdrDragMethod::createSdrDragEntryForSdrObject(const SdrObject& rOriginal, sdr::contact::ObjectContact& rObjectContact) { // add full object drag; Clone() at the object has to work // for this addSdrDragEntry(new SdrDragEntrySdrObject(rOriginal, rObjectContact, true/*bModify*/)); } void SdrDragMethod::createSdrDragEntries_SolidDrag() { const size_t nMarkCount(getSdrDragView().GetMarkedObjectCount()); SdrPageView* pPV = getSdrDragView().GetSdrPageView(); if(pPV) { for(size_t a = 0; a < nMarkCount; ++a) { SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(a); if(pM->GetPageView() == pPV) { const SdrObject* pObject = pM->GetMarkedSdrObj(); if(pObject) { if(pPV->PageWindowCount()) { sdr::contact::ObjectContact& rOC = pPV->GetPageWindow(0)->GetObjectContact(); SdrObjListIter aIter(*pObject); while(aIter.IsMore()) { SdrObject* pCandidate = aIter.Next(); if(pCandidate) { const bool bSuppressFullDrag(!pCandidate->supportsFullDrag()); bool bAddWireframe(bSuppressFullDrag); if(!bAddWireframe && !pCandidate->HasLineStyle()) { // add wireframe for objects without outline bAddWireframe = true; } if(!bSuppressFullDrag) { // add full object drag; Clone() at the object has to work // for this createSdrDragEntryForSdrObject(*pCandidate, rOC); } if(bAddWireframe) { // when dragging a 50% transparent copy of a filled or not filled object without // outline, this is normally hard to see. Add extra wireframe in that case. This // works nice e.g. with text frames etc. addSdrDragEntry(new SdrDragEntryPolyPolygon(pCandidate->TakeXorPoly())); } } } } } } } } } void SdrDragMethod::createSdrDragEntries_PolygonDrag() { const size_t nMarkCount(getSdrDragView().GetMarkedObjectCount()); bool bNoPolygons(getSdrDragView().IsNoDragXorPolys() || nMarkCount > getSdrDragView().GetDragXorPolyLimit()); basegfx::B2DPolyPolygon aResult; sal_uInt32 nPointCount(0); for(size_t a = 0; !bNoPolygons && a < nMarkCount; ++a) { SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(a); if(pM->GetPageView() == getSdrDragView().GetSdrPageView()) { const basegfx::B2DPolyPolygon aNewPolyPolygon(pM->GetMarkedSdrObj()->TakeXorPoly()); for(sal_uInt32 b(0); b < aNewPolyPolygon.count(); b++) { nPointCount += aNewPolyPolygon.getB2DPolygon(b).count(); } if(nPointCount > getSdrDragView().GetDragXorPointLimit()) { bNoPolygons = true; } if(!bNoPolygons) { aResult.append(aNewPolyPolygon); } } } if(bNoPolygons) { const Rectangle aR(getSdrDragView().GetSdrPageView()->MarkSnap()); const basegfx::B2DRange aNewRectangle(aR.Left(), aR.Top(), aR.Right(), aR.Bottom()); basegfx::B2DPolygon aNewPolygon(basegfx::tools::createPolygonFromRect(aNewRectangle)); aResult = basegfx::B2DPolyPolygon(basegfx::tools::expandToCurve(aNewPolygon)); } if(aResult.count()) { addSdrDragEntry(new SdrDragEntryPolyPolygon(aResult)); } } void SdrDragMethod::createSdrDragEntries_PointDrag() { const size_t nMarkCount(getSdrDragView().GetMarkedObjectCount()); std::vector< basegfx::B2DPoint > aPositions; for(size_t nm = 0; nm < nMarkCount; ++nm) { SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(nm); if(pM->GetPageView() == getSdrDragView().GetSdrPageView()) { const SdrUShortCont& rPts = pM->GetMarkedPoints(); if (!rPts.empty()) { const SdrObject* pObj = pM->GetMarkedSdrObj(); const SdrPathObj* pPath = dynamic_cast< const SdrPathObj* >(pObj); if(pPath) { const basegfx::B2DPolyPolygon aPathXPP = pPath->GetPathPoly(); if(aPathXPP.count()) { for(SdrUShortCont::const_iterator it = rPts.begin(); it != rPts.end(); ++it) { sal_uInt32 nPolyNum, nPointNum; const sal_uInt16 nObjPt = *it; if(sdr::PolyPolygonEditor::GetRelativePolyPoint(aPathXPP, nObjPt, nPolyNum, nPointNum)) { aPositions.push_back(aPathXPP.getB2DPolygon(nPolyNum).getB2DPoint(nPointNum)); } } } } } } } if(!aPositions.empty()) { addSdrDragEntry(new SdrDragEntryPointGlueDrag(aPositions, true)); } } void SdrDragMethod::createSdrDragEntries_GlueDrag() { const size_t nMarkCount(getSdrDragView().GetMarkedObjectCount()); std::vector< basegfx::B2DPoint > aPositions; for(size_t nm = 0; nm < nMarkCount; ++nm) { SdrMark* pM = getSdrDragView().GetSdrMarkByIndex(nm); if(pM->GetPageView() == getSdrDragView().GetSdrPageView()) { const SdrUShortCont& rPts = pM->GetMarkedGluePoints(); if (!rPts.empty()) { const SdrObject* pObj = pM->GetMarkedSdrObj(); const SdrGluePointList* pGPL = pObj->GetGluePointList(); if (pGPL) { for(SdrUShortCont::const_iterator it = rPts.begin(); it != rPts.end(); ++it) { const sal_uInt16 nObjPt = *it; const sal_uInt16 nGlueNum(pGPL->FindGluePoint(nObjPt)); if(SDRGLUEPOINT_NOTFOUND != nGlueNum) { const Point aPoint((*pGPL)[nGlueNum].GetAbsolutePos(*pObj)); aPositions.push_back(basegfx::B2DPoint(aPoint.X(), aPoint.Y())); } } } } } } if(!aPositions.empty()) { addSdrDragEntry(new SdrDragEntryPointGlueDrag(aPositions, false)); } } void SdrDragMethod::ImpTakeDescriptionStr(sal_uInt16 nStrCacheID, OUString& rStr) const { ImpTakeDescriptionOptions nOpt=ImpTakeDescriptionOptions::NONE; if (IsDraggingPoints()) { nOpt=ImpTakeDescriptionOptions::POINTS; } else if (IsDraggingGluePoints()) { nOpt=ImpTakeDescriptionOptions::GLUEPOINTS; } getSdrDragView().ImpTakeDescriptionStr(nStrCacheID,rStr,nOpt); } SdrObject* SdrDragMethod::GetDragObj() const { SdrObject* pObj=nullptr; if (getSdrDragView().mpDragHdl!=nullptr) pObj=getSdrDragView().mpDragHdl->GetObj(); if (pObj==nullptr) pObj=getSdrDragView().mpMarkedObj; return pObj; } SdrPageView* SdrDragMethod::GetDragPV() const { SdrPageView* pPV=nullptr; if (getSdrDragView().mpDragHdl!=nullptr) pPV=getSdrDragView().mpDragHdl->GetPageView(); if (pPV==nullptr) pPV=getSdrDragView().mpMarkedPV; return pPV; } void SdrDragMethod::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { // the original applies the transformation using TRGetBaseGeometry/TRSetBaseGeometry. // Later this should be the only needed one for linear transforms (not for SdrDragCrook and // SdrDragDistort, those are NOT linear). Currently, this can not yet be used since the // special handling of rotate/mirror due to the not-being-able to handle it in the old // drawinglayer stuff. Text would currently not correctly be mirrored in the preview. basegfx::B2DHomMatrix aObjectTransform; basegfx::B2DPolyPolygon aObjectPolyPolygon; bool bPolyUsed(rTarget.TRGetBaseGeometry(aObjectTransform, aObjectPolyPolygon)); // apply transform to object transform aObjectTransform *= getCurrentTransformation(); if(bPolyUsed) { // do something special since the object size is in the polygon // break up matrix to get the scale basegfx::B2DTuple aScale; basegfx::B2DTuple aTranslate; double fRotate, fShearX; aObjectTransform.decompose(aScale, aTranslate, fRotate, fShearX); // get polygon's position and size const basegfx::B2DRange aPolyRange(aObjectPolyPolygon.getB2DRange()); // get the scaling factors (do not mirror, this is in the object transformation) const double fScaleX(fabs(aScale.getX()) / (basegfx::fTools::equalZero(aPolyRange.getWidth()) ? 1.0 : aPolyRange.getWidth())); const double fScaleY(fabs(aScale.getY()) / (basegfx::fTools::equalZero(aPolyRange.getHeight()) ? 1.0 : aPolyRange.getHeight())); // prepare transform matrix for polygon basegfx::B2DHomMatrix aPolyTransform(basegfx::tools::createTranslateB2DHomMatrix( -aPolyRange.getMinX(), -aPolyRange.getMinY())); aPolyTransform.scale(fScaleX, fScaleY); // transform the polygon aObjectPolyPolygon.transform(aPolyTransform); } rTarget.TRSetBaseGeometry(getCurrentTransformation() * aObjectTransform, aObjectPolyPolygon); } void SdrDragMethod::applyCurrentTransformationToPolyPolygon(basegfx::B2DPolyPolygon& rTarget) { // original uses CurrentTransformation rTarget.transform(getCurrentTransformation()); } SdrDragMethod::SdrDragMethod(SdrDragView& rNewView) : maSdrDragEntries(), maOverlayObjectList(), mrSdrDragView(rNewView), mbMoveOnly(false), mbSolidDraggingActive(getSdrDragView().IsSolidDragging()), mbShiftPressed(false) { if(mbSolidDraggingActive && Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { // fallback to wireframe when high contrast is used mbSolidDraggingActive = false; } } SdrDragMethod::~SdrDragMethod() { clearSdrDragEntries(); } void SdrDragMethod::Show() { getSdrDragView().ShowDragObj(); } void SdrDragMethod::Hide() { getSdrDragView().HideDragObj(); } basegfx::B2DHomMatrix SdrDragMethod::getCurrentTransformation() { return basegfx::B2DHomMatrix(); } void SdrDragMethod::CancelSdrDrag() { Hide(); } typedef std::map< const SdrObject*, SdrObject* > SdrObjectAndCloneMap; void SdrDragMethod::CreateOverlayGeometry(sdr::overlay::OverlayManager& rOverlayManager) { // create SdrDragEntries on demand if(maSdrDragEntries.empty()) { createSdrDragEntries(); } // if there are entries, derive OverlayObjects from the entries, including // modification from current interactive state if(!maSdrDragEntries.empty()) { // #i54102# SdrDragEntrySdrObject creates clones of SdrObjects as base for creating the needed // primitives, holding the original and the clone. If connectors (Edges) are involved, // the cloned connectors need to be connected to the cloned SdrObjects (after cloning // they are connected to the original SdrObjects). To do so, trigger the preparation // steps for SdrDragEntrySdrObject, save an association of (orig, clone) in a helper // and evtl. remember if it was an edge SdrObjectAndCloneMap aOriginalAndClones; std::vector< SdrEdgeObj* > aEdges; sal_uInt32 a; // #i54102# execute prepareCurrentState for all SdrDragEntrySdrObject, register pair of original and // clone, remember edges for(a = 0; a < maSdrDragEntries.size(); a++) { SdrDragEntrySdrObject* pSdrDragEntrySdrObject = dynamic_cast< SdrDragEntrySdrObject*>(maSdrDragEntries[a]); if(pSdrDragEntrySdrObject) { pSdrDragEntrySdrObject->prepareCurrentState(*this); SdrEdgeObj* pSdrEdgeObj = dynamic_cast< SdrEdgeObj* >(pSdrDragEntrySdrObject->getClone()); if(pSdrEdgeObj) { aEdges.push_back(pSdrEdgeObj); } if(pSdrDragEntrySdrObject->getClone()) { aOriginalAndClones[&pSdrDragEntrySdrObject->getOriginal()] = pSdrDragEntrySdrObject->getClone(); } } } // #i54102# if there are edges, reconnect their ends to the corresponding clones (if found) if(aEdges.size()) { for(a = 0; a < aEdges.size(); a++) { SdrEdgeObj* pSdrEdgeObj = aEdges[a]; SdrObject* pConnectedTo = pSdrEdgeObj->GetConnectedNode(true); if(pConnectedTo) { SdrObjectAndCloneMap::iterator aEntry = aOriginalAndClones.find(pConnectedTo); if(aEntry != aOriginalAndClones.end()) { pSdrEdgeObj->ConnectToNode(true, aEntry->second); } } pConnectedTo = pSdrEdgeObj->GetConnectedNode(false); if(pConnectedTo) { SdrObjectAndCloneMap::iterator aEntry = aOriginalAndClones.find(pConnectedTo); if(aEntry != aOriginalAndClones.end()) { pSdrEdgeObj->ConnectToNode(false, aEntry->second); } } } } // collect primitives for visualisation drawinglayer::primitive2d::Primitive2DContainer aResult; drawinglayer::primitive2d::Primitive2DContainer aResultTransparent; for(a = 0; a < maSdrDragEntries.size(); a++) { SdrDragEntry* pCandidate = maSdrDragEntries[a]; if(pCandidate) { const drawinglayer::primitive2d::Primitive2DContainer aCandidateResult(pCandidate->createPrimitive2DSequenceInCurrentState(*this)); if(!aCandidateResult.empty()) { if(pCandidate->getAddToTransparent()) { aResultTransparent.append(aCandidateResult); } else { aResult.append(aCandidateResult); } } } } if(DoAddConnectorOverlays()) { const drawinglayer::primitive2d::Primitive2DContainer aConnectorOverlays(AddConnectorOverlays()); if(!aConnectorOverlays.empty()) { // add connector overlays to transparent part aResultTransparent.append(aConnectorOverlays); } } if(!aResult.empty()) { sdr::overlay::OverlayObject* pNewOverlayObject = new sdr::overlay::OverlayPrimitive2DSequenceObject(aResult); rOverlayManager.add(*pNewOverlayObject); addToOverlayObjectList(*pNewOverlayObject); } if(!aResultTransparent.empty()) { drawinglayer::primitive2d::Primitive2DReference aUnifiedTransparencePrimitive2D(new drawinglayer::primitive2d::UnifiedTransparencePrimitive2D(aResultTransparent, 0.5)); aResultTransparent = drawinglayer::primitive2d::Primitive2DContainer { aUnifiedTransparencePrimitive2D }; sdr::overlay::OverlayObject* pNewOverlayObject = new sdr::overlay::OverlayPrimitive2DSequenceObject(aResultTransparent); rOverlayManager.add(*pNewOverlayObject); addToOverlayObjectList(*pNewOverlayObject); } } // add DragStripes if necessary (help lines cross the page when dragging) if(getSdrDragView().IsDragStripes()) { Rectangle aActionRectangle; getSdrDragView().TakeActionRect(aActionRectangle); const basegfx::B2DPoint aTopLeft(aActionRectangle.Left(), aActionRectangle.Top()); const basegfx::B2DPoint aBottomRight(aActionRectangle.Right(), aActionRectangle.Bottom()); sdr::overlay::OverlayRollingRectangleStriped* pNew = new sdr::overlay::OverlayRollingRectangleStriped( aTopLeft, aBottomRight, true, false); rOverlayManager.add(*pNew); addToOverlayObjectList(*pNew); } } void SdrDragMethod::destroyOverlayGeometry() { maOverlayObjectList.clear(); } bool SdrDragMethod::DoAddConnectorOverlays() { // these conditions are translated from SdrDragView::ImpDrawEdgeXor const SdrMarkList& rMarkedNodes = getSdrDragView().GetEdgesOfMarkedNodes(); if(!rMarkedNodes.GetMarkCount()) { return false; } if(getSdrDragView().IsDraggingPoints() || getSdrDragView().IsDraggingGluePoints()) { return false; } if(!getMoveOnly() && !( dynamic_cast(this) != nullptr || dynamic_cast(this) != nullptr || dynamic_cast(this) != nullptr || dynamic_cast(this) != nullptr )) { return false; } // one more migrated from SdrEdgeObj::NspToggleEdgeXor if( dynamic_cast< const SdrDragObjOwn* >(this) != nullptr || dynamic_cast< const SdrDragMovHdl* >(this) != nullptr ) { return false; } return true; } drawinglayer::primitive2d::Primitive2DContainer SdrDragMethod::AddConnectorOverlays() { drawinglayer::primitive2d::Primitive2DContainer aRetval; const bool bDetail(getMoveOnly()); const SdrMarkList& rMarkedNodes = getSdrDragView().GetEdgesOfMarkedNodes(); for(size_t a = 0; a < rMarkedNodes.GetMarkCount(); ++a) { SdrMark* pEM = rMarkedNodes.GetMark(a); if(pEM && pEM->GetMarkedSdrObj()) { SdrEdgeObj* pEdge = dynamic_cast< SdrEdgeObj* >(pEM->GetMarkedSdrObj()); if(pEdge) { const basegfx::B2DPolygon aEdgePolygon(pEdge->ImplAddConnectorOverlay(*this, pEM->IsCon1(), pEM->IsCon2(), bDetail)); if(aEdgePolygon.count()) { // this polygon is a temporary calculated connector path, so it is not possible to fetch // the needed primitives directly from the pEdge object which does not get changed. If full // drag is on, use the SdrObjects ItemSet to create a adequate representation bool bUseSolidDragging(getSolidDraggingActive()); if(bUseSolidDragging) { // switch off solid dragging if connector is not visible if(!pEdge->HasLineStyle()) { bUseSolidDragging = false; } } if(bUseSolidDragging) { const SfxItemSet& rItemSet = pEdge->GetMergedItemSet(); const drawinglayer::attribute::SdrLineAttribute aLine( drawinglayer::primitive2d::createNewSdrLineAttribute(rItemSet)); if(!aLine.isDefault()) { const drawinglayer::attribute::SdrLineStartEndAttribute aLineStartEnd( drawinglayer::primitive2d::createNewSdrLineStartEndAttribute( rItemSet, aLine.getWidth())); aRetval.push_back(drawinglayer::primitive2d::createPolygonLinePrimitive( aEdgePolygon, aLine, aLineStartEnd)); } } else { const SvtOptionsDrawinglayer aSvtOptionsDrawinglayer; basegfx::BColor aColA(aSvtOptionsDrawinglayer.GetStripeColorA().getBColor()); basegfx::BColor aColB(aSvtOptionsDrawinglayer.GetStripeColorB().getBColor()); const double fStripeLength(aSvtOptionsDrawinglayer.GetStripeLength()); if(Application::GetSettings().GetStyleSettings().GetHighContrastMode()) { aColA = aColB = Application::GetSettings().GetStyleSettings().GetHighlightColor().getBColor(); aColB.invert(); } drawinglayer::primitive2d::Primitive2DReference aPolyPolygonMarkerPrimitive2D( new drawinglayer::primitive2d::PolygonMarkerPrimitive2D( aEdgePolygon, aColA, aColB, fStripeLength)); aRetval.push_back(aPolyPolygonMarkerPrimitive2D); } } } } } return aRetval; } SdrDragMovHdl::SdrDragMovHdl(SdrDragView& rNewView) : SdrDragMethod(rNewView) { } void SdrDragMovHdl::createSdrDragEntries() { // SdrDragMovHdl does not use the default drags, // but creates nothing } void SdrDragMovHdl::TakeSdrDragComment(OUString& rStr) const { rStr=ImpGetResStr(STR_DragMethMovHdl); if (getSdrDragView().IsDragWithCopy()) rStr+=ImpGetResStr(STR_EditWithCopy); } bool SdrDragMovHdl::BeginSdrDrag() { if( !GetDragHdl() ) return false; DragStat().Ref1()=GetDragHdl()->GetPos(); DragStat().SetShown(!DragStat().IsShown()); SdrHdlKind eKind=GetDragHdl()->GetKind(); SdrHdl* pH1=GetHdlList().GetHdl(SdrHdlKind::Ref1); SdrHdl* pH2=GetHdlList().GetHdl(SdrHdlKind::Ref2); if (eKind==SdrHdlKind::MirrorAxis) { if (pH1==nullptr || pH2==nullptr) { OSL_FAIL("SdrDragMovHdl::BeginSdrDrag(): Moving the axis of reflection: reference handles not found."); return false; } DragStat().SetActionRect(Rectangle(pH1->GetPos(),pH2->GetPos())); } else { Point aPt(GetDragHdl()->GetPos()); DragStat().SetActionRect(Rectangle(aPt,aPt)); } return true; } void SdrDragMovHdl::MoveSdrDrag(const Point& rNoSnapPnt) { Point aPnt(rNoSnapPnt); if ( GetDragHdl() && DragStat().CheckMinMoved(rNoSnapPnt)) { if (GetDragHdl()->GetKind()==SdrHdlKind::MirrorAxis) { SdrHdl* pH1=GetHdlList().GetHdl(SdrHdlKind::Ref1); SdrHdl* pH2=GetHdlList().GetHdl(SdrHdlKind::Ref2); if (pH1==nullptr || pH2==nullptr) return; if (!DragStat().IsNoSnap()) { long nBestXSnap=0; long nBestYSnap=0; bool bXSnapped=false; bool bYSnapped=false; Point aDif(aPnt-DragStat().GetStart()); getSdrDragView().CheckSnap(Ref1()+aDif,nBestXSnap,nBestYSnap,bXSnapped,bYSnapped); getSdrDragView().CheckSnap(Ref2()+aDif,nBestXSnap,nBestYSnap,bXSnapped,bYSnapped); aPnt.X()+=nBestXSnap; aPnt.Y()+=nBestYSnap; } if (aPnt!=DragStat().GetNow()) { Hide(); DragStat().NextMove(aPnt); Point aDif(DragStat().GetNow()-DragStat().GetStart()); pH1->SetPos(Ref1()+aDif); pH2->SetPos(Ref2()+aDif); SdrHdl* pHM = GetHdlList().GetHdl(SdrHdlKind::MirrorAxis); if(pHM) pHM->Touch(); Show(); DragStat().SetActionRect(Rectangle(pH1->GetPos(),pH2->GetPos())); } } else { if (!DragStat().IsNoSnap()) SnapPos(aPnt); long nSA=0; if (getSdrDragView().IsAngleSnapEnabled()) nSA=getSdrDragView().GetSnapAngle(); if (getSdrDragView().IsMirrorAllowed(true,true)) { // limited if (!getSdrDragView().IsMirrorAllowed()) nSA=4500; if (!getSdrDragView().IsMirrorAllowed(true)) nSA=9000; } if (getSdrDragView().IsOrtho() && nSA!=9000) nSA=4500; if (nSA!=0) { // angle snapping SdrHdlKind eRef=SdrHdlKind::Ref1; if (GetDragHdl()->GetKind()==SdrHdlKind::Ref1) eRef=SdrHdlKind::Ref2; SdrHdl* pH=GetHdlList().GetHdl(eRef); if (pH!=nullptr) { Point aRef(pH->GetPos()); long nAngle=NormAngle360(GetAngle(aPnt-aRef)); long nNewAngle=nAngle; nNewAngle+=nSA/2; nNewAngle/=nSA; nNewAngle*=nSA; nNewAngle=NormAngle360(nNewAngle); double a=(nNewAngle-nAngle)*nPi180; double nSin=sin(a); double nCos=cos(a); RotatePoint(aPnt,aRef,nSin,nCos); // eliminate rounding errors for certain values if (nSA==9000) { if (nNewAngle==0 || nNewAngle==18000) aPnt.Y()=aRef.Y(); if (nNewAngle==9000 || nNewAngle==27000) aPnt.X()=aRef.X(); } if (nSA==4500) OrthoDistance8(aRef,aPnt,true); } } if (aPnt!=DragStat().GetNow()) { Hide(); DragStat().NextMove(aPnt); GetDragHdl()->SetPos(DragStat().GetNow()); SdrHdl* pHM = GetHdlList().GetHdl(SdrHdlKind::MirrorAxis); if(pHM) pHM->Touch(); Show(); DragStat().SetActionRect(Rectangle(aPnt,aPnt)); } } } } bool SdrDragMovHdl::EndSdrDrag(bool /*bCopy*/) { if( GetDragHdl() ) { switch (GetDragHdl()->GetKind()) { case SdrHdlKind::Ref1: Ref1()=DragStat().GetNow(); break; case SdrHdlKind::Ref2: Ref2()=DragStat().GetNow(); break; case SdrHdlKind::MirrorAxis: Ref1()+=DragStat().GetNow()-DragStat().GetStart(); Ref2()+=DragStat().GetNow()-DragStat().GetStart(); break; default: break; } } return true; } void SdrDragMovHdl::CancelSdrDrag() { Hide(); SdrHdl* pHdl = GetDragHdl(); if( pHdl ) pHdl->SetPos(DragStat().GetRef1()); SdrHdl* pHM = GetHdlList().GetHdl(SdrHdlKind::MirrorAxis); if(pHM) pHM->Touch(); } Pointer SdrDragMovHdl::GetSdrDragPointer() const { const SdrHdl* pHdl = GetDragHdl(); if (pHdl!=nullptr) { return pHdl->GetPointer(); } return Pointer(PointerStyle::RefHand); } SdrDragObjOwn::SdrDragObjOwn(SdrDragView& rNewView) : SdrDragMethod(rNewView), mpClone(nullptr) { const SdrObject* pObj = GetDragObj(); if(pObj) { // suppress full drag for some object types setSolidDraggingActive(pObj->supportsFullDrag()); } } SdrDragObjOwn::~SdrDragObjOwn() { if(mpClone) { SdrObject::Free(mpClone); } } void SdrDragObjOwn::createSdrDragEntries() { if(mpClone) { basegfx::B2DPolyPolygon aDragPolyPolygon; bool bAddWireframe(true); if(getSolidDraggingActive()) { SdrPageView* pPV = getSdrDragView().GetSdrPageView(); if(pPV && pPV->PageWindowCount()) { sdr::contact::ObjectContact& rOC = pPV->GetPageWindow(0)->GetObjectContact(); addSdrDragEntry(new SdrDragEntrySdrObject(*mpClone, rOC, false)); // potentially no wireframe needed, full drag works bAddWireframe = false; } } if(!bAddWireframe) { // check for extra conditions for wireframe, e.g. no border at // objects if(!mpClone->HasLineStyle()) { bAddWireframe = true; } } if(bAddWireframe) { // use wireframe poly when full drag is off or did not work aDragPolyPolygon = mpClone->TakeXorPoly(); } // add evtl. extra DragPolyPolygon const basegfx::B2DPolyPolygon aSpecialDragPolyPolygon(mpClone->getSpecialDragPoly(DragStat())); if(aSpecialDragPolyPolygon.count()) { aDragPolyPolygon.append(aSpecialDragPolyPolygon); } if(aDragPolyPolygon.count()) { addSdrDragEntry(new SdrDragEntryPolyPolygon(aDragPolyPolygon)); } } } void SdrDragObjOwn::TakeSdrDragComment(OUString& rStr) const { // #i103058# get info string from the clone preferred, the original will // not be changed. For security, use original as fallback if(mpClone) { rStr = mpClone->getSpecialDragComment(DragStat()); } else { const SdrObject* pObj = GetDragObj(); if(pObj) { rStr = pObj->getSpecialDragComment(DragStat()); } } } bool SdrDragObjOwn::BeginSdrDrag() { if(!mpClone) { const SdrObject* pObj = GetDragObj(); if(pObj && !pObj->IsResizeProtect()) { if(pObj->beginSpecialDrag(DragStat())) { // create initial clone to have a start visualization mpClone = pObj->getFullDragClone(); mpClone->applySpecialDrag(DragStat()); return true; } } } return false; } void SdrDragObjOwn::MoveSdrDrag(const Point& rNoSnapPnt) { const SdrObject* pObj = GetDragObj(); if (!pObj) // No object to drag. Bail out. return; Point aPnt(rNoSnapPnt); SdrPageView* pPV = GetDragPV(); if (!pPV) // No page view available. Bail out. return; if(!DragStat().IsNoSnap()) { SnapPos(aPnt); } if(getSdrDragView().IsOrtho()) { if (DragStat().IsOrtho8Possible()) { OrthoDistance8(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); } else if (DragStat().IsOrtho4Possible()) { OrthoDistance4(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); } } if (!DragStat().CheckMinMoved(rNoSnapPnt)) // Not moved by the minimum threshold. Nothing to do. return; Hide(); DragStat().NextMove(aPnt); // since SdrDragObjOwn currently supports no transformation of // existing SdrDragEntries but only their recreation, a recreation // after every move is needed in this mode. Delete existing // SdrDragEntries here to force their recreation in the following Show(). clearSdrDragEntries(); // delete current clone (after the last reference to it is deleted above) if(mpClone) { SdrObject::Free(mpClone); mpClone = nullptr; } // create a new clone and modify to current drag state mpClone = pObj->getFullDragClone(); mpClone->applySpecialDrag(DragStat()); // #120999# AutoGrowWidth may change for SdrTextObj due to the automatism used // with bDisableAutoWidthOnDragging, so not only geometry changes but // also this (pretty indirect) property change is possible. If it gets // changed, it needs to be copied to the original since nothing will // happen when it only changes in the drag clone const bool bOldAutoGrowWidth(static_cast(pObj->GetMergedItem(SDRATTR_TEXT_AUTOGROWWIDTH)).GetValue()); const bool bNewAutoGrowWidth(static_cast(mpClone->GetMergedItem(SDRATTR_TEXT_AUTOGROWWIDTH)).GetValue()); if (bOldAutoGrowWidth != bNewAutoGrowWidth) { GetDragObj()->SetMergedItem(makeSdrTextAutoGrowWidthItem(bNewAutoGrowWidth)); } Show(); } bool SdrDragObjOwn::EndSdrDrag(bool /*bCopy*/) { Hide(); std::vector< SdrUndoAction* > vConnectorUndoActions; bool bRet = false; SdrObject* pObj = GetDragObj(); if(pObj) { SdrUndoAction* pUndo = nullptr; SdrUndoAction* pUndo2 = nullptr; const bool bUndo = getSdrDragView().IsUndoEnabled(); if( bUndo ) { if(!getSdrDragView().IsInsObjPoint() && pObj->IsInserted() ) { if (DragStat().IsEndDragChangesAttributes()) { pUndo=getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoAttrObject(*pObj); if (DragStat().IsEndDragChangesGeoAndAttributes()) { vConnectorUndoActions = getSdrDragView().CreateConnectorUndo( *pObj ); pUndo2 = getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pObj); } } else { vConnectorUndoActions = getSdrDragView().CreateConnectorUndo( *pObj ); pUndo= getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pObj); } } if( pUndo ) { getSdrDragView().BegUndo( pUndo->GetComment() ); } else { getSdrDragView().BegUndo(); } } // Maybe use operator = for setting changed object data (do not change selection in // view, this will destroy the interactor). This is possible since a clone is now // directly modified by the modifiers. Only SdrTableObj is adding own UNDOs // in its SdrTableObj::endSpecialDrag, so currently not possible. OTOH it uses // a CreateUndoGeoObject(), so maybe setting SetEndDragChangesAttributes is okay. I // will test this now Rectangle aBoundRect0; if(pObj->GetUserCall()) { aBoundRect0 = pObj->GetLastBoundRect(); } bRet = pObj->applySpecialDrag(DragStat()); if (DragStat().IsEndDragChangesLayout()) { auto pGeoUndo = dynamic_cast(pUndo); if (pGeoUndo) pGeoUndo->SetSkipChangeLayout(true); } if(bRet) { pObj->SetChanged(); pObj->BroadcastObjectChange(); pObj->SendUserCall( SdrUserCallType::Resize, aBoundRect0 ); } if(bRet) { if( bUndo ) { getSdrDragView().AddUndoActions( vConnectorUndoActions ); if ( pUndo ) { getSdrDragView().AddUndo(pUndo); } if ( pUndo2 ) { getSdrDragView().AddUndo(pUndo2); } } } else { if( bUndo ) { std::vector< SdrUndoAction* >::iterator vConnectorUndoIter( vConnectorUndoActions.begin() ); while( vConnectorUndoIter != vConnectorUndoActions.end() ) { delete *vConnectorUndoIter++; } delete pUndo; delete pUndo2; } } if( bUndo ) getSdrDragView().EndUndo(); } return bRet; } Pointer SdrDragObjOwn::GetSdrDragPointer() const { const SdrHdl* pHdl=GetDragHdl(); if (pHdl) { return pHdl->GetPointer(); } return Pointer(PointerStyle::Move); } void SdrDragMove::createSdrDragEntryForSdrObject(const SdrObject& rOriginal, sdr::contact::ObjectContact& rObjectContact) { // for SdrDragMove, use current Primitive2DContainer of SdrObject visualization // in given ObjectContact directly sdr::contact::ViewContact& rVC = rOriginal.GetViewContact(); sdr::contact::ViewObjectContact& rVOC = rVC.GetViewObjectContact(rObjectContact); sdr::contact::DisplayInfo aDisplayInfo; // Do not use the last ViewPort set at the OC from the last ProcessDisplay(), // here we want the complete primitive sequence without visible clippings rObjectContact.resetViewPort(); addSdrDragEntry(new SdrDragEntryPrimitive2DSequence(rVOC.getPrimitive2DSequenceHierarchy(aDisplayInfo))); } void SdrDragMove::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { rTarget.Move(Size(DragStat().GetDX(), DragStat().GetDY())); } SdrDragMove::SdrDragMove(SdrDragView& rNewView) : SdrDragMethod(rNewView) , nBestXSnap(0) , nBestYSnap(0) , bXSnapped(false) , bYSnapped(false) { setMoveOnly(true); } void SdrDragMove::TakeSdrDragComment(OUString& rStr) const { OUString aStr; ImpTakeDescriptionStr(STR_DragMethMove, rStr); rStr += " (x="; getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDX(), aStr); rStr += aStr + " y="; getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDY(), aStr); rStr += aStr + ")"; if(getSdrDragView().IsDragWithCopy()) { if(!getSdrDragView().IsInsObjPoint() && !getSdrDragView().IsInsGluePoint()) { rStr += ImpGetResStr(STR_EditWithCopy); } } } bool SdrDragMove::BeginSdrDrag() { DragStat().SetActionRect(GetMarkedRect()); Show(); return true; } basegfx::B2DHomMatrix SdrDragMove::getCurrentTransformation() { return basegfx::tools::createTranslateB2DHomMatrix(DragStat().GetDX(), DragStat().GetDY()); } void SdrDragMove::ImpCheckSnap(const Point& rPt) { Point aPt(rPt); SdrSnap nRet=SnapPos(aPt); aPt-=rPt; if (nRet & SdrSnap::XSNAPPED) { if (bXSnapped) { if (std::abs(aPt.X())aLR.Left() || aSR2.Right()aLR.Right()) { aPt1.X()-=aSR2.Right()-aLR.Right(); } } else aPt1.X()=DragStat().GetStart().X(); // no space to move to if (aSR2.Top()>aLR.Top() || aSR2.Bottom()aLR.Bottom()) { aPt1.Y()-=aSR2.Bottom()-aLR.Bottom(); } } else aPt1.Y()=DragStat().GetStart().Y(); // no space to move to } if (getSdrDragView().IsDraggingGluePoints()) { // restrict glue points to the BoundRect of the Obj aPt1-=DragStat().GetStart(); const SdrMarkList& rML=GetMarkedObjectList(); const size_t nMarkCount=rML.GetMarkCount(); for (size_t nMarkNum=0; nMarkNumGetMarkedGluePoints(); if (!rPts.empty()) { const SdrObject* pObj=pM->GetMarkedSdrObj(); const SdrGluePointList* pGPL=pObj->GetGluePointList(); Rectangle aBound(pObj->GetCurrentBoundRect()); for (SdrUShortCont::const_iterator it = rPts.begin(); it != rPts.end(); ++it) { sal_uInt16 nId = *it; sal_uInt16 nGlueNum=pGPL->FindGluePoint(nId); if (nGlueNum!=SDRGLUEPOINT_NOTFOUND) { Point aPt((*pGPL)[nGlueNum].GetAbsolutePos(*pObj)); aPt+=aPt1; // move by this much if (aPt.X()aBound.Right() ) aPt1.X()-=aPt.X()-aBound.Right() ; if (aPt.Y()aBound.Bottom()) aPt1.Y()-=aPt.Y()-aBound.Bottom(); } } } } aPt1+=DragStat().GetStart(); } if (bOrtho) OrthoDistance8(DragStat().GetStart(),aPt1,false); if (aPt1!=DragStat().GetNow()) { Hide(); DragStat().NextMove(aPt1); Rectangle aAction(GetMarkedRect()); aAction.Move(DragStat().GetDX(),DragStat().GetDY()); DragStat().SetActionRect(aAction); Show(); } } } bool SdrDragMove::EndSdrDrag(bool bCopy) { Hide(); if (getSdrDragView().IsInsObjPoint() || getSdrDragView().IsInsGluePoint()) bCopy=false; if (IsDraggingPoints()) { getSdrDragView().MoveMarkedPoints(Size(DragStat().GetDX(),DragStat().GetDY())); } else if (IsDraggingGluePoints()) { getSdrDragView().MoveMarkedGluePoints(Size(DragStat().GetDX(),DragStat().GetDY()),bCopy); } else { getSdrDragView().MoveMarkedObj(Size(DragStat().GetDX(),DragStat().GetDY()),bCopy); } return true; } Pointer SdrDragMove::GetSdrDragPointer() const { if (IsDraggingPoints() || IsDraggingGluePoints()) { return Pointer(PointerStyle::MovePoint); } else { return Pointer(PointerStyle::Move); } } SdrDragResize::SdrDragResize(SdrDragView& rNewView) : SdrDragMethod(rNewView), aXFact(1,1), aYFact(1,1) { } void SdrDragResize::TakeSdrDragComment(OUString& rStr) const { ImpTakeDescriptionStr(STR_DragMethResize, rStr); Fraction aFact1(1,1); Point aStart(DragStat().GetStart()); Point aRef(DragStat().GetRef1()); sal_Int32 nXDiv(aStart.X() - aRef.X()); if(!nXDiv) nXDiv = 1; sal_Int32 nYDiv(aStart.Y() - aRef.Y()); if(!nYDiv) nYDiv = 1; bool bX(aXFact != aFact1 && std::abs(nXDiv) > 1); bool bY(aYFact != aFact1 && std::abs(nYDiv) > 1); if(bX || bY) { OUString aStr; rStr += " ("; bool bEqual(aXFact == aYFact); if(bX) { if(!bEqual) rStr += "x="; SdrModel::TakePercentStr(aXFact, aStr); rStr += aStr; } if(bY && !bEqual) { if(bX) rStr += " "; rStr += "y="; SdrModel::TakePercentStr(aYFact, aStr); rStr += aStr; } rStr += ")"; } if(getSdrDragView().IsDragWithCopy()) rStr += ImpGetResStr(STR_EditWithCopy); } bool SdrDragResize::BeginSdrDrag() { SdrHdlKind eRefHdl=SdrHdlKind::Move; SdrHdl* pRefHdl=nullptr; switch (GetDragHdlKind()) { case SdrHdlKind::UpperLeft: eRefHdl=SdrHdlKind::LowerRight; break; case SdrHdlKind::Upper: eRefHdl=SdrHdlKind::Lower; DragStat().SetHorFixed(true); break; case SdrHdlKind::UpperRight: eRefHdl=SdrHdlKind::LowerLeft; break; case SdrHdlKind::Left : eRefHdl=SdrHdlKind::Right; DragStat().SetVerFixed(true); break; case SdrHdlKind::Right: eRefHdl=SdrHdlKind::Left ; DragStat().SetVerFixed(true); break; case SdrHdlKind::LowerLeft: eRefHdl=SdrHdlKind::UpperRight; break; case SdrHdlKind::Lower: eRefHdl=SdrHdlKind::Upper; DragStat().SetHorFixed(true); break; case SdrHdlKind::LowerRight: eRefHdl=SdrHdlKind::UpperLeft; break; default: break; } if (eRefHdl!=SdrHdlKind::Move) pRefHdl=GetHdlList().GetHdl(eRefHdl); if (pRefHdl!=nullptr && !getSdrDragView().IsResizeAtCenter()) { // Calc hack to adjust for calc grid DragStat().Ref1()=pRefHdl->GetPos() - getSdrDragView().GetGridOffset(); } else { SdrHdl* pRef1=GetHdlList().GetHdl(SdrHdlKind::UpperLeft); SdrHdl* pRef2=GetHdlList().GetHdl(SdrHdlKind::LowerRight); if (pRef1!=nullptr && pRef2!=nullptr) { DragStat().Ref1()=Rectangle(pRef1->GetPos(),pRef2->GetPos()).Center(); } else { DragStat().Ref1()=GetMarkedRect().Center(); } } Show(); return true; } basegfx::B2DHomMatrix SdrDragResize::getCurrentTransformation() { basegfx::B2DHomMatrix aRetval(basegfx::tools::createTranslateB2DHomMatrix( -DragStat().Ref1().X(), -DragStat().Ref1().Y())); aRetval.scale(aXFact, aYFact); aRetval.translate(DragStat().Ref1().X(), DragStat().Ref1().Y()); return aRetval; } void SdrDragResize::MoveSdrDrag(const Point& rNoSnapPnt) { Point aPnt(GetSnapPos(rNoSnapPnt)); Point aStart(DragStat().GetStart()); Point aRef(DragStat().GetRef1()); Fraction aMaxFact(0x7FFFFFFF,1); Rectangle aLR(getSdrDragView().GetWorkArea()); bool bWorkArea=!aLR.IsEmpty(); bool bDragLimit=IsDragLimit(); if (bDragLimit || bWorkArea) { Rectangle aSR(GetMarkedRect()); if (bDragLimit) { Rectangle aR2(GetDragLimitRect()); if (bWorkArea) aLR.Intersection(aR2); else aLR=aR2; } if (aPnt.X()aLR.Right()) aPnt.X()=aLR.Right(); if (aPnt.Y()aLR.Bottom()) aPnt.Y()=aLR.Bottom(); if (aRef.X()>aSR.Left()) { Fraction aMax(aRef.X()-aLR.Left(),aRef.X()-aSR.Left()); if (aMaxaSR.Top()) { Fraction aMax(aRef.Y()-aLR.Top(),aRef.Y()-aSR.Top()); if (aMaxFraction(nYMul,nYDiv)) !=getSdrDragView().IsBigOrtho()) { nXMul=nYMul; nXDiv=nYDiv; } else { nYMul=nXMul; nYDiv=nXDiv; } } } else { if (bOrtho) { if (DragStat().IsHorFixed()) { bXNeg=false; nXMul=nYMul; nXDiv=nYDiv; } if (DragStat().IsVerFixed()) { bYNeg=false; nYMul=nXMul; nYDiv=nXDiv; } } else { if (DragStat().IsHorFixed()) { bXNeg=false; nXMul=1; nXDiv=1; } if (DragStat().IsVerFixed()) { bYNeg=false; nYMul=1; nYDiv=1; } } } Fraction aNeuXFact(nXMul,nXDiv); Fraction aNeuYFact(nYMul,nYDiv); if (bOrtho) { if (aNeuXFact>aMaxFact) { aNeuXFact=aMaxFact; aNeuYFact=aMaxFact; } if (aNeuYFact>aMaxFact) { aNeuXFact=aMaxFact; aNeuYFact=aMaxFact; } } if (bXNeg) aNeuXFact=Fraction(-aNeuXFact.GetNumerator(),aNeuXFact.GetDenominator()); if (bYNeg) aNeuYFact=Fraction(-aNeuYFact.GetNumerator(),aNeuYFact.GetDenominator()); if (DragStat().CheckMinMoved(aPnt)) { if ((!DragStat().IsHorFixed() && aPnt.X()!=DragStat().GetNow().X()) || (!DragStat().IsVerFixed() && aPnt.Y()!=DragStat().GetNow().Y())) { Hide(); DragStat().NextMove(aPnt); aXFact=aNeuXFact; aYFact=aNeuYFact; Show(); } } } void SdrDragResize::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { rTarget.Resize(DragStat().Ref1(),aXFact,aYFact); } bool SdrDragResize::EndSdrDrag(bool bCopy) { Hide(); if (IsDraggingPoints()) { getSdrDragView().ResizeMarkedPoints(DragStat().Ref1(),aXFact,aYFact); } else if (IsDraggingGluePoints()) { getSdrDragView().ResizeMarkedGluePoints(DragStat().Ref1(),aXFact,aYFact,bCopy); } else { getSdrDragView().ResizeMarkedObj(DragStat().Ref1(),aXFact,aYFact,bCopy); } return true; } Pointer SdrDragResize::GetSdrDragPointer() const { const SdrHdl* pHdl=GetDragHdl(); if (pHdl!=nullptr) { return pHdl->GetPointer(); } return Pointer(PointerStyle::Move); } void SdrDragRotate::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { rTarget.Rotate(DragStat().GetRef1(), nAngle, sin(nAngle*nPi180), cos(nAngle*nPi180)); } SdrDragRotate::SdrDragRotate(SdrDragView& rNewView) : SdrDragMethod(rNewView), nSin(0.0), nCos(1.0), nAngle0(0), nAngle(0), bRight(false) { } void SdrDragRotate::TakeSdrDragComment(OUString& rStr) const { ImpTakeDescriptionStr(STR_DragMethRotate, rStr); rStr += " ("; sal_Int32 nTmpAngle(NormAngle360(nAngle)); if(bRight && nAngle) { nTmpAngle -= 36000; } OUString aStr; SdrModel::TakeAngleStr(nTmpAngle, aStr); rStr += aStr + ")"; if(getSdrDragView().IsDragWithCopy()) rStr += ImpGetResStr(STR_EditWithCopy); } bool SdrDragRotate::BeginSdrDrag() { SdrHdl* pH=GetHdlList().GetHdl(SdrHdlKind::Ref1); if (pH!=nullptr) { Show(); DragStat().Ref1()=pH->GetPos(); nAngle0=GetAngle(DragStat().GetStart()-DragStat().GetRef1()); return true; } else { OSL_FAIL("SdrDragRotate::BeginSdrDrag(): No reference point handle found."); return false; } } basegfx::B2DHomMatrix SdrDragRotate::getCurrentTransformation() { return basegfx::tools::createRotateAroundPoint( DragStat().GetRef1().X(), DragStat().GetRef1().Y(), -atan2(nSin, nCos)); } void SdrDragRotate::MoveSdrDrag(const Point& rPnt_) { Point aPnt(rPnt_); if (DragStat().CheckMinMoved(aPnt)) { long nNewAngle=NormAngle360(GetAngle(aPnt-DragStat().GetRef1())-nAngle0); long nSA=0; if (getSdrDragView().IsAngleSnapEnabled()) nSA=getSdrDragView().GetSnapAngle(); if (!getSdrDragView().IsRotateAllowed()) nSA=9000; if (nSA!=0) { // angle snapping nNewAngle+=nSA/2; nNewAngle/=nSA; nNewAngle*=nSA; } nNewAngle=NormAngle180(nNewAngle); if (nAngle!=nNewAngle) { sal_uInt16 nSekt0=GetAngleSector(nAngle); sal_uInt16 nSekt1=GetAngleSector(nNewAngle); if (nSekt0==0 && nSekt1==3) bRight=true; if (nSekt0==3 && nSekt1==0) bRight=false; nAngle=nNewAngle; double a=nAngle*nPi180; double nSin1=sin(a); // calculate now, so as little time as possible double nCos1=cos(a); // passes between Hide() and Show() Hide(); nSin=nSin1; nCos=nCos1; DragStat().NextMove(aPnt); Show(); } } } bool SdrDragRotate::EndSdrDrag(bool bCopy) { Hide(); if (nAngle!=0) { if (IsDraggingPoints()) { getSdrDragView().RotateMarkedPoints(DragStat().GetRef1(),nAngle); } else if (IsDraggingGluePoints()) { getSdrDragView().RotateMarkedGluePoints(DragStat().GetRef1(),nAngle,bCopy); } else { getSdrDragView().RotateMarkedObj(DragStat().GetRef1(),nAngle,bCopy); } } return true; } Pointer SdrDragRotate::GetSdrDragPointer() const { return Pointer(PointerStyle::Rotate); } SdrDragShear::SdrDragShear(SdrDragView& rNewView, bool bSlant1) : SdrDragMethod(rNewView), aFact(1,1), nAngle0(0), nAngle(0), nTan(0.0), bVertical(false), bResize(false), bUpSideDown(false), bSlant(bSlant1) { } void SdrDragShear::TakeSdrDragComment(OUString& rStr) const { ImpTakeDescriptionStr(STR_DragMethShear, rStr); rStr += " ("; sal_Int32 nTmpAngle(nAngle); if(bUpSideDown) nTmpAngle += 18000; nTmpAngle = NormAngle180(nTmpAngle); OUString aStr; SdrModel::TakeAngleStr(nTmpAngle, aStr); rStr += aStr + ")"; if(getSdrDragView().IsDragWithCopy()) rStr += ImpGetResStr(STR_EditWithCopy); } bool SdrDragShear::BeginSdrDrag() { SdrHdlKind eRefHdl=SdrHdlKind::Move; SdrHdl* pRefHdl=nullptr; switch (GetDragHdlKind()) { case SdrHdlKind::Upper: eRefHdl=SdrHdlKind::Lower; break; case SdrHdlKind::Lower: eRefHdl=SdrHdlKind::Upper; break; case SdrHdlKind::Left : eRefHdl=SdrHdlKind::Right; bVertical=true; break; case SdrHdlKind::Right: eRefHdl=SdrHdlKind::Left ; bVertical=true; break; default: break; } if (eRefHdl!=SdrHdlKind::Move) pRefHdl=GetHdlList().GetHdl(eRefHdl); if (pRefHdl!=nullptr) { DragStat().Ref1()=pRefHdl->GetPos(); nAngle0=GetAngle(DragStat().GetStart()-DragStat().GetRef1()); } else { OSL_FAIL("SdrDragShear::BeginSdrDrag(): No reference point handle for shearing found."); return false; } Show(); return true; } basegfx::B2DHomMatrix SdrDragShear::getCurrentTransformation() { basegfx::B2DHomMatrix aRetval(basegfx::tools::createTranslateB2DHomMatrix( -DragStat().GetRef1().X(), -DragStat().GetRef1().Y())); if (bResize) { if (bVertical) { aRetval.scale(aFact, 1.0); aRetval.shearY(-nTan); } else { aRetval.scale(1.0, aFact); aRetval.shearX(-nTan); } } aRetval.translate(DragStat().GetRef1().X(), DragStat().GetRef1().Y()); return aRetval; } void SdrDragShear::MoveSdrDrag(const Point& rPnt) { if (DragStat().CheckMinMoved(rPnt)) { bResize=!getSdrDragView().IsOrtho(); long nSA=0; if (getSdrDragView().IsAngleSnapEnabled()) nSA=getSdrDragView().GetSnapAngle(); Point aP0(DragStat().GetStart()); Point aPnt(rPnt); Fraction aNeuFact(1,1); // if angle snapping not activated, snap to raster (except when using slant) if (nSA==0 && !bSlant) aPnt=GetSnapPos(aPnt); if (!bSlant && !bResize) { // shear, but no resize if (bVertical) aPnt.X()=aP0.X(); else aPnt.Y()=aP0.Y(); } Point aRef(DragStat().GetRef1()); Point aDif(aPnt-aRef); long nNewAngle=0; if (bSlant) { nNewAngle=NormAngle180(-(GetAngle(aDif)-nAngle0)); if (bVertical) nNewAngle=NormAngle180(-nNewAngle); } else { if (bVertical) nNewAngle=NormAngle180(GetAngle(aDif)); else nNewAngle=NormAngle180(-(GetAngle(aDif)-9000)); if (nNewAngle<-9000 || nNewAngle>9000) nNewAngle=NormAngle180(nNewAngle+18000); if (bResize) { Point aPt2(aPnt); if (nSA!=0) aPt2=GetSnapPos(aPnt); // snap this one in any case if (bVertical) { aNeuFact=Fraction(aPt2.X()-aRef.X(),aP0.X()-aRef.X()); } else { aNeuFact=Fraction(aPt2.Y()-aRef.Y(),aP0.Y()-aRef.Y()); } } } bool bNeg=nNewAngle<0; if (bNeg) nNewAngle=-nNewAngle; if (nSA!=0) { // angle snapping nNewAngle+=nSA/2; nNewAngle/=nSA; nNewAngle*=nSA; } nNewAngle=NormAngle360(nNewAngle); bUpSideDown=nNewAngle>9000 && nNewAngle<27000; if (bSlant) { // calculate resize for slant // when angle snapping is activated, disable 89 degree limit long nTmpAngle=nNewAngle; if (bUpSideDown) nNewAngle-=18000; if (bNeg) nTmpAngle=-nTmpAngle; bResize=true; double nCos=cos(nTmpAngle*nPi180); aNeuFact=nCos; aFact.ReduceInaccurate(10); // three decimals should be enough } if (nNewAngle>8900) nNewAngle=8900; if (bNeg) nNewAngle=-nNewAngle; if (nAngle!=nNewAngle || aFact!=aNeuFact) { nAngle=nNewAngle; aFact=aNeuFact; double a=nAngle*nPi180; double nTan1=tan(a); // calculate now, so as little time as possible passes between Hide() and Show() Hide(); nTan=nTan1; DragStat().NextMove(rPnt); Show(); } } } void SdrDragShear::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { if (bResize) { if (bVertical) { rTarget.Resize(DragStat().GetRef1(),aFact,Fraction(1,1)); } else { rTarget.Resize(DragStat().GetRef1(),Fraction(1,1),aFact); } } if (nAngle!=0) { rTarget.Shear(DragStat().GetRef1(),nAngle,tan(nAngle*nPi180),bVertical); } } bool SdrDragShear::EndSdrDrag(bool bCopy) { Hide(); if (bResize && aFact==Fraction(1,1)) bResize=false; if (nAngle!=0 || bResize) { if (nAngle!=0 && bResize) { OUString aStr; ImpTakeDescriptionStr(STR_EditShear,aStr); if (bCopy) aStr += ImpGetResStr(STR_EditWithCopy); getSdrDragView().BegUndo(aStr); } if (bResize) { if (bVertical) { getSdrDragView().ResizeMarkedObj(DragStat().GetRef1(),aFact,Fraction(1,1),bCopy); } else { getSdrDragView().ResizeMarkedObj(DragStat().GetRef1(),Fraction(1,1),aFact,bCopy); } bCopy=false; } if (nAngle!=0) { getSdrDragView().ShearMarkedObj(DragStat().GetRef1(),nAngle,bVertical,bCopy); } if (nAngle!=0 && bResize) getSdrDragView().EndUndo(); return true; } return false; } Pointer SdrDragShear::GetSdrDragPointer() const { if (bVertical) return Pointer(PointerStyle::VShear); else return Pointer(PointerStyle::HShear); } void SdrDragMirror::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { if(bMirrored) { rTarget.Mirror(DragStat().GetRef1(), DragStat().GetRef2()); } } SdrDragMirror::SdrDragMirror(SdrDragView& rNewView) : SdrDragMethod(rNewView), nAngle(0), bMirrored(false), bSide0(false) { } bool SdrDragMirror::ImpCheckSide(const Point& rPnt) const { long nAngle1=GetAngle(rPnt-DragStat().GetRef1()); nAngle1-=nAngle; nAngle1=NormAngle360(nAngle1); return nAngle1<18000; } void SdrDragMirror::TakeSdrDragComment(OUString& rStr) const { if (aDif.X()==0) ImpTakeDescriptionStr(STR_DragMethMirrorHori,rStr); else if (aDif.Y()==0) ImpTakeDescriptionStr(STR_DragMethMirrorVert,rStr); else if (std::abs(aDif.X()) == std::abs(aDif.Y())) ImpTakeDescriptionStr(STR_DragMethMirrorDiag,rStr); else ImpTakeDescriptionStr(STR_DragMethMirrorFree,rStr); if (getSdrDragView().IsDragWithCopy()) rStr+=ImpGetResStr(STR_EditWithCopy); } bool SdrDragMirror::BeginSdrDrag() { SdrHdl* pH1=GetHdlList().GetHdl(SdrHdlKind::Ref1); SdrHdl* pH2=GetHdlList().GetHdl(SdrHdlKind::Ref2); if (pH1!=nullptr && pH2!=nullptr) { DragStat().Ref1()=pH1->GetPos(); DragStat().Ref2()=pH2->GetPos(); Ref1()=pH1->GetPos(); Ref2()=pH2->GetPos(); aDif=pH2->GetPos()-pH1->GetPos(); bool b90=(aDif.X()==0) || aDif.Y()==0; bool b45=b90 || (std::abs(aDif.X()) == std::abs(aDif.Y())); nAngle=NormAngle360(GetAngle(aDif)); if (!getSdrDragView().IsMirrorAllowed() && !b45) return false; // free choice of axis angle not allowed if (!getSdrDragView().IsMirrorAllowed() && !b90) return false; // 45 degrees not allowed either bSide0=ImpCheckSide(DragStat().GetStart()); Show(); return true; } else { OSL_FAIL("SdrDragMirror::BeginSdrDrag(): Axis of reflection not found."); return false; } } basegfx::B2DHomMatrix SdrDragMirror::getCurrentTransformation() { basegfx::B2DHomMatrix aRetval; if (bMirrored) { const double fDeltaX(DragStat().GetRef2().X() - DragStat().GetRef1().X()); const double fDeltaY(DragStat().GetRef2().Y() - DragStat().GetRef1().Y()); const double fRotation(atan2(fDeltaY, fDeltaX)); aRetval = basegfx::tools::createTranslateB2DHomMatrix(-DragStat().GetRef1().X(), -DragStat().GetRef1().Y()); aRetval.rotate(-fRotation); aRetval.scale(1.0, -1.0); aRetval.rotate(fRotation); aRetval.translate(DragStat().GetRef1().X(), DragStat().GetRef1().Y()); } return aRetval; } void SdrDragMirror::MoveSdrDrag(const Point& rPnt) { if (DragStat().CheckMinMoved(rPnt)) { bool bNeuSide=ImpCheckSide(rPnt); bool bNeuMirr=bSide0!=bNeuSide; if (bMirrored!=bNeuMirr) { Hide(); bMirrored=bNeuMirr; DragStat().NextMove(rPnt); Show(); } } } bool SdrDragMirror::EndSdrDrag(bool bCopy) { Hide(); if (bMirrored) { getSdrDragView().MirrorMarkedObj(DragStat().GetRef1(),DragStat().GetRef2(),bCopy); } return true; } Pointer SdrDragMirror::GetSdrDragPointer() const { return Pointer(PointerStyle::Mirror); } SdrDragGradient::SdrDragGradient(SdrDragView& rNewView, bool bGrad) : SdrDragMethod(rNewView), pIAOHandle(nullptr), bIsGradient(bGrad) { } void SdrDragGradient::TakeSdrDragComment(OUString& rStr) const { if(IsGradient()) ImpTakeDescriptionStr(STR_DragMethGradient, rStr); else ImpTakeDescriptionStr(STR_DragMethTransparence, rStr); } bool SdrDragGradient::BeginSdrDrag() { bool bRetval(false); pIAOHandle = static_cast(GetHdlList().GetHdl(IsGradient() ? SdrHdlKind::Gradient : SdrHdlKind::Transparence)); if(pIAOHandle) { // save old values DragStat().Ref1() = pIAOHandle->GetPos(); DragStat().Ref2() = pIAOHandle->Get2ndPos(); // what was hit? bool bHit(false); SdrHdlColor* pColHdl = pIAOHandle->GetColorHdl1(); // init handling flags pIAOHandle->SetMoveSingleHandle(false); pIAOHandle->SetMoveFirstHandle(false); // test first color handle if(pColHdl) { basegfx::B2DPoint aPosition(DragStat().GetStart().X(), DragStat().GetStart().Y()); if(pColHdl->getOverlayObjectList().isHitLogic(aPosition)) { bHit = true; pIAOHandle->SetMoveSingleHandle(true); pIAOHandle->SetMoveFirstHandle(true); } } // test second color handle pColHdl = pIAOHandle->GetColorHdl2(); if(!bHit && pColHdl) { basegfx::B2DPoint aPosition(DragStat().GetStart().X(), DragStat().GetStart().Y()); if(pColHdl->getOverlayObjectList().isHitLogic(aPosition)) { bHit = true; pIAOHandle->SetMoveSingleHandle(true); } } // test gradient handle itself if(!bHit) { basegfx::B2DPoint aPosition(DragStat().GetStart().X(), DragStat().GetStart().Y()); if(pIAOHandle->getOverlayObjectList().isHitLogic(aPosition)) { bHit = true; } } // everything up and running :o} bRetval = bHit; } else { OSL_FAIL("SdrDragGradient::BeginSdrDrag(): IAOGradient not found."); } return bRetval; } void SdrDragGradient::MoveSdrDrag(const Point& rPnt) { if(pIAOHandle && DragStat().CheckMinMoved(rPnt)) { DragStat().NextMove(rPnt); // Do the Move here!!! DragStat().GetStart() Point aMoveDiff = rPnt - DragStat().GetStart(); if(pIAOHandle->IsMoveSingleHandle()) { if(pIAOHandle->IsMoveFirstHandle()) { pIAOHandle->SetPos(DragStat().Ref1() + aMoveDiff); if(pIAOHandle->GetColorHdl1()) pIAOHandle->GetColorHdl1()->SetPos(DragStat().Ref1() + aMoveDiff); } else { pIAOHandle->Set2ndPos(DragStat().Ref2() + aMoveDiff); if(pIAOHandle->GetColorHdl2()) pIAOHandle->GetColorHdl2()->SetPos(DragStat().Ref2() + aMoveDiff); } } else { pIAOHandle->SetPos(DragStat().Ref1() + aMoveDiff); pIAOHandle->Set2ndPos(DragStat().Ref2() + aMoveDiff); if(pIAOHandle->GetColorHdl1()) pIAOHandle->GetColorHdl1()->SetPos(DragStat().Ref1() + aMoveDiff); if(pIAOHandle->GetColorHdl2()) pIAOHandle->GetColorHdl2()->SetPos(DragStat().Ref2() + aMoveDiff); } // new state pIAOHandle->FromIAOToItem(getSdrDragView().GetMarkedObjectList().GetMark(0)->GetMarkedSdrObj(), false, false); } } bool SdrDragGradient::EndSdrDrag(bool /*bCopy*/) { Ref1() = pIAOHandle->GetPos(); Ref2() = pIAOHandle->Get2ndPos(); // new state pIAOHandle->FromIAOToItem(getSdrDragView().GetMarkedObjectList().GetMark(0)->GetMarkedSdrObj(), true, true); return true; } void SdrDragGradient::CancelSdrDrag() { // restore old values pIAOHandle->SetPos(DragStat().Ref1()); pIAOHandle->Set2ndPos(DragStat().Ref2()); if(pIAOHandle->GetColorHdl1()) pIAOHandle->GetColorHdl1()->SetPos(DragStat().Ref1()); if(pIAOHandle->GetColorHdl2()) pIAOHandle->GetColorHdl2()->SetPos(DragStat().Ref2()); // new state pIAOHandle->FromIAOToItem(getSdrDragView().GetMarkedObjectList().GetMark(0)->GetMarkedSdrObj(), true, false); } Pointer SdrDragGradient::GetSdrDragPointer() const { return Pointer(PointerStyle::RefHand); } SdrDragCrook::SdrDragCrook(SdrDragView& rNewView) : SdrDragMethod(rNewView), aFact(1,1), bContortionAllowed(false), bNoContortionAllowed(false), bContortion(false), bResizeAllowed(false), bResize(false), bRotateAllowed(false), bRotate(false), bVertical(false), bValid(false), bLft(false), bRgt(false), bUpr(false), bLwr(false), bAtCenter(false), nAngle(0), nMarkSize(0), eMode(SdrCrookMode::Rotate) { } void SdrDragCrook::TakeSdrDragComment(OUString& rStr) const { ImpTakeDescriptionStr(!bContortion ? STR_DragMethCrook : STR_DragMethCrookContortion, rStr); if(bValid) { rStr += " ("; sal_Int32 nVal(nAngle); if(bAtCenter) nVal *= 2; nVal = std::abs(nVal); OUString aStr; SdrModel::TakeAngleStr(nVal, aStr); rStr += aStr + ")"; } if(getSdrDragView().IsDragWithCopy()) rStr += ImpGetResStr(STR_EditWithCopy); } // These defines parametrize the created raster // for interactions #define DRAG_CROOK_RASTER_MINIMUM (4) #define DRAG_CROOK_RASTER_MAXIMUM (15) #define DRAG_CROOK_RASTER_DISTANCE (30) basegfx::B2DPolyPolygon impCreateDragRaster(SdrPageView& rPageView, const Rectangle& rMarkRect) { basegfx::B2DPolyPolygon aRetval; if(rPageView.PageWindowCount()) { OutputDevice& rOut = (rPageView.GetPageWindow(0)->GetPaintWindow().GetOutputDevice()); Rectangle aPixelSize = rOut.LogicToPixel(rMarkRect); sal_uInt32 nHorDiv(aPixelSize.GetWidth() / DRAG_CROOK_RASTER_DISTANCE); sal_uInt32 nVerDiv(aPixelSize.GetHeight() / DRAG_CROOK_RASTER_DISTANCE); if(nHorDiv > DRAG_CROOK_RASTER_MAXIMUM) nHorDiv = DRAG_CROOK_RASTER_MAXIMUM; if(nHorDiv < DRAG_CROOK_RASTER_MINIMUM) nHorDiv = DRAG_CROOK_RASTER_MINIMUM; if(nVerDiv > DRAG_CROOK_RASTER_MAXIMUM) nVerDiv = DRAG_CROOK_RASTER_MAXIMUM; if(nVerDiv < DRAG_CROOK_RASTER_MINIMUM) nVerDiv = DRAG_CROOK_RASTER_MINIMUM; const double fXLen(rMarkRect.GetWidth() / (double)nHorDiv); const double fYLen(rMarkRect.GetHeight() / (double)nVerDiv); double fYPos(rMarkRect.Top()); sal_uInt32 a, b; for(a = 0; a <= nVerDiv; a++) { // horizontal lines for(b = 0; b < nHorDiv; b++) { basegfx::B2DPolygon aHorLineSegment; const double fNewX(rMarkRect.Left() + (b * fXLen)); aHorLineSegment.append(basegfx::B2DPoint(fNewX, fYPos)); aHorLineSegment.appendBezierSegment( basegfx::B2DPoint(fNewX + (fXLen * (1.0 / 3.0)), fYPos), basegfx::B2DPoint(fNewX + (fXLen * (2.0 / 3.0)), fYPos), basegfx::B2DPoint(fNewX + fXLen, fYPos)); aRetval.append(aHorLineSegment); } // increments fYPos += fYLen; } double fXPos(rMarkRect.Left()); for(a = 0; a <= nHorDiv; a++) { // vertical lines for(b = 0; b < nVerDiv; b++) { basegfx::B2DPolygon aVerLineSegment; const double fNewY(rMarkRect.Top() + (b * fYLen)); aVerLineSegment.append(basegfx::B2DPoint(fXPos, fNewY)); aVerLineSegment.appendBezierSegment( basegfx::B2DPoint(fXPos, fNewY + (fYLen * (1.0 / 3.0))), basegfx::B2DPoint(fXPos, fNewY + (fYLen * (2.0 / 3.0))), basegfx::B2DPoint(fXPos, fNewY + fYLen)); aRetval.append(aVerLineSegment); } // increments fXPos += fXLen; } } return aRetval; } void SdrDragCrook::createSdrDragEntries() { // Add extended frame raster first, so it will be behind objects if(getSdrDragView().GetSdrPageView()) { const basegfx::B2DPolyPolygon aDragRaster(impCreateDragRaster(*getSdrDragView().GetSdrPageView(), GetMarkedRect())); if(aDragRaster.count()) { addSdrDragEntry(new SdrDragEntryPolyPolygon(aDragRaster)); } } // call parent SdrDragMethod::createSdrDragEntries(); } bool SdrDragCrook::BeginSdrDrag() { bContortionAllowed=getSdrDragView().IsCrookAllowed(); bNoContortionAllowed=getSdrDragView().IsCrookAllowed(true); bResizeAllowed=getSdrDragView().IsResizeAllowed(); bRotateAllowed=getSdrDragView().IsRotateAllowed(); if (bContortionAllowed || bNoContortionAllowed) { bVertical=(GetDragHdlKind()==SdrHdlKind::Lower || GetDragHdlKind()==SdrHdlKind::Upper); aMarkRect=GetMarkedRect(); aMarkCenter=aMarkRect.Center(); nMarkSize=bVertical ? (aMarkRect.GetHeight()-1) : (aMarkRect.GetWidth()-1); aCenter=aMarkCenter; aStart=DragStat().GetStart(); Show(); return true; } else { return false; } } void SdrDragCrook::MovAllPoints(basegfx::B2DPolyPolygon& rTarget) { SdrPageView* pPV = getSdrDragView().GetSdrPageView(); if(pPV) { XPolyPolygon aTempPolyPoly(rTarget); if (pPV->HasMarkedObjPageView()) { sal_uInt16 nPolyCount=aTempPolyPoly.Count(); if (!bContortion && !getSdrDragView().IsNoDragXorPolys()) { sal_uInt16 n1st=0,nLast=0; Point aC(aCenter); while (n1ststd::abs(dy1); else bValid = std::abs(dy1)*100>std::abs(dx1); } long nNeuRad=0; nAngle=0; if (bValid) { double a=0; // slope of the radius long nPntWink=0; if (bVertical) { a=((double)dy1)/((double)dx1); // slope of the radius nNeuRad=((long)(dy1*a)+dx1) /2; aNeuCenter.X()+=nNeuRad; nPntWink=GetAngle(aPnt-aNeuCenter); } else { a=((double)dx1)/((double)dy1); // slope of the radius nNeuRad=((long)(dx1*a)+dy1) /2; aNeuCenter.Y()+=nNeuRad; nPntWink=GetAngle(aPnt-aNeuCenter)-9000; } if (!bAtCenter) { if (nNeuRad<0) { if (bRgt) nPntWink+=18000; if (bLft) nPntWink=18000-nPntWink; if (bLwr) nPntWink=-nPntWink; } else { if (bRgt) nPntWink=-nPntWink; if (bUpr) nPntWink=18000-nPntWink; if (bLwr) nPntWink+=18000; } nPntWink=NormAngle360(nPntWink); } else { if (nNeuRad<0) nPntWink+=18000; if (bVertical) nPntWink=18000-nPntWink; nPntWink=NormAngle180(nPntWink); nPntWink = std::abs(nPntWink); } double nUmfang = 2 * std::abs(nNeuRad)*nPi; if (bResize) { long nMul=(long)(nUmfang*NormAngle360(nPntWink)/36000); if (bAtCenter) nMul*=2; aNeuFact=Fraction(nMul,nMarkSize); nAngle=nPntWink; } else { nAngle=(long)((nMarkSize*360/nUmfang)*100)/2; if (nAngle==0) bValid=false; } } if (nAngle==0 || nNeuRad==0) bValid=false; if (!bValid) nNeuRad=0; if (!bValid && bResize) { long nMul=bVertical ? dy1 : dx1; if (bLft || bUpr) nMul=-nMul; long nDiv=nMarkSize; if (bAtCenter) { nMul*=2; nMul = std::abs(nMul); } aNeuFact=Fraction(nMul,nDiv); } if (aNeuCenter!=aCenter || bNeuContortion!=bContortion || aNeuFact!=aFact || bNeuMoveOnly != getMoveOnly() || bNeuRotate!=bRotate || eNeuMode!=eMode) { Hide(); setMoveOnly(bNeuMoveOnly); bRotate=bNeuRotate; eMode=eNeuMode; bContortion=bNeuContortion; aCenter=aNeuCenter; aFact=aNeuFact; aRad=Point(nNeuRad,nNeuRad); bResize=aFact!=Fraction(1,1) && aFact.GetDenominator()!=0 && aFact.IsValid(); DragStat().NextMove(aPnt); Show(); } } } void SdrDragCrook::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { const bool bDoResize(aFact!=Fraction(1,1)); const bool bDoCrook(aCenter!=aMarkCenter && aRad.X()!=0 && aRad.Y()!=0); if (bDoCrook || bDoResize) { if (bDoResize) { Fraction aFact1(1,1); if (bContortion) { if (bVertical) { rTarget.Resize(aCenter,aFact1,aFact); } else { rTarget.Resize(aCenter,aFact,aFact1); } } else { Point aCtr0(rTarget.GetSnapRect().Center()); Point aCtr1(aCtr0); if (bVertical) { ResizePoint(aCtr1,aCenter,aFact1,aFact); } else { ResizePoint(aCtr1,aCenter,aFact,aFact1); } Size aSiz(aCtr1.X()-aCtr0.X(),aCtr1.Y()-aCtr0.Y()); rTarget.Move(aSiz); } } if (bDoCrook) { const Rectangle aLocalMarkRect(getSdrDragView().GetMarkedObjRect()); const bool bLocalRotate(!bContortion && eMode == SdrCrookMode::Rotate && getSdrDragView().IsRotateAllowed()); SdrEditView::ImpCrookObj(&rTarget,aCenter,aRad,eMode,bVertical,!bContortion,bLocalRotate,aLocalMarkRect); } } } void SdrDragCrook::applyCurrentTransformationToPolyPolygon(basegfx::B2DPolyPolygon& rTarget) { // use helper derived from old stuff MovAllPoints(rTarget); } bool SdrDragCrook::EndSdrDrag(bool bCopy) { Hide(); if (bResize && aFact==Fraction(1,1)) bResize=false; const bool bUndo = getSdrDragView().IsUndoEnabled(); bool bDoCrook=aCenter!=aMarkCenter && aRad.X()!=0 && aRad.Y()!=0; if (bDoCrook || bResize) { if (bResize && bUndo) { OUString aStr; ImpTakeDescriptionStr(!bContortion?STR_EditCrook:STR_EditCrookContortion,aStr); if (bCopy) aStr += ImpGetResStr(STR_EditWithCopy); getSdrDragView().BegUndo(aStr); } if (bResize) { Fraction aFact1(1,1); if (bContortion) { if (bVertical) getSdrDragView().ResizeMarkedObj(aCenter,aFact1,aFact,bCopy); else getSdrDragView().ResizeMarkedObj(aCenter,aFact,aFact1,bCopy); } else { if (bCopy) getSdrDragView().CopyMarkedObj(); const size_t nMarkCount=getSdrDragView().GetMarkedObjectList().GetMarkCount(); for (size_t nm=0; nmGetMarkedSdrObj(); Point aCtr0(pO->GetSnapRect().Center()); Point aCtr1(aCtr0); if (bVertical) ResizePoint(aCtr1,aCenter,aFact1,aFact); else ResizePoint(aCtr1,aCenter,aFact,aFact1); Size aSiz(aCtr1.X()-aCtr0.X(),aCtr1.Y()-aCtr0.Y()); if( bUndo ) AddUndo(getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoMoveObject(*pO,aSiz)); pO->Move(aSiz); } } bCopy=false; } if (bDoCrook) { getSdrDragView().CrookMarkedObj(aCenter,aRad,eMode,bVertical,!bContortion,bCopy); getSdrDragView().SetLastCrookCenter(aCenter); } if (bResize && bUndo) getSdrDragView().EndUndo(); return true; } return false; } Pointer SdrDragCrook::GetSdrDragPointer() const { return Pointer(PointerStyle::Crook); } SdrDragDistort::SdrDragDistort(SdrDragView& rNewView) : SdrDragMethod(rNewView), nPolyPt(0), bContortionAllowed(false), bNoContortionAllowed(false), bContortion(false) { } void SdrDragDistort::TakeSdrDragComment(OUString& rStr) const { ImpTakeDescriptionStr(STR_DragMethDistort, rStr); OUString aStr; rStr += " (x="; getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDX(), aStr); rStr += aStr + " y="; getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDY(), aStr); rStr += aStr + ")"; if(getSdrDragView().IsDragWithCopy()) rStr += ImpGetResStr(STR_EditWithCopy); } void SdrDragDistort::createSdrDragEntries() { // Add extended frame raster first, so it will be behind objects if(getSdrDragView().GetSdrPageView()) { const basegfx::B2DPolyPolygon aDragRaster(impCreateDragRaster(*getSdrDragView().GetSdrPageView(), GetMarkedRect())); if(aDragRaster.count()) { addSdrDragEntry(new SdrDragEntryPolyPolygon(aDragRaster)); } } // call parent SdrDragMethod::createSdrDragEntries(); } bool SdrDragDistort::BeginSdrDrag() { bContortionAllowed=getSdrDragView().IsDistortAllowed(); bNoContortionAllowed=getSdrDragView().IsDistortAllowed(true); if (bContortionAllowed || bNoContortionAllowed) { SdrHdlKind eKind=GetDragHdlKind(); nPolyPt=0xFFFF; if (eKind==SdrHdlKind::UpperLeft) nPolyPt=0; if (eKind==SdrHdlKind::UpperRight) nPolyPt=1; if (eKind==SdrHdlKind::LowerRight) nPolyPt=2; if (eKind==SdrHdlKind::LowerLeft) nPolyPt=3; if (nPolyPt>3) return false; aMarkRect=GetMarkedRect(); aDistortedRect=XPolygon(aMarkRect); Show(); return true; } else { return false; } } void SdrDragDistort::MovAllPoints(basegfx::B2DPolyPolygon& rTarget) { if (bContortion) { SdrPageView* pPV = getSdrDragView().GetSdrPageView(); if(pPV) { if (pPV->HasMarkedObjPageView()) { basegfx::B2DPolyPolygon aDragPolygon(rTarget); const basegfx::B2DRange aOriginalRange(aMarkRect.Left(), aMarkRect.Top(), aMarkRect.Right(), aMarkRect.Bottom()); const basegfx::B2DPoint aTopLeft(aDistortedRect[0].X(), aDistortedRect[0].Y()); const basegfx::B2DPoint aTopRight(aDistortedRect[1].X(), aDistortedRect[1].Y()); const basegfx::B2DPoint aBottomLeft(aDistortedRect[3].X(), aDistortedRect[3].Y()); const basegfx::B2DPoint aBottomRight(aDistortedRect[2].X(), aDistortedRect[2].Y()); aDragPolygon = basegfx::tools::distort(aDragPolygon, aOriginalRange, aTopLeft, aTopRight, aBottomLeft, aBottomRight); rTarget = aDragPolygon; } } } } void SdrDragDistort::MoveSdrDrag(const Point& rPnt) { if (DragStat().CheckMinMoved(rPnt)) { Point aPnt(GetSnapPos(rPnt)); if (getSdrDragView().IsOrtho()) OrthoDistance8(DragStat().GetStart(),aPnt,getSdrDragView().IsBigOrtho()); bool bNeuContortion=(bContortionAllowed && !getSdrDragView().IsCrookNoContortion()) || !bNoContortionAllowed; if (bNeuContortion!=bContortion || aDistortedRect[nPolyPt]!=aPnt) { Hide(); aDistortedRect[nPolyPt]=aPnt; bContortion=bNeuContortion; DragStat().NextMove(aPnt); Show(); } } } bool SdrDragDistort::EndSdrDrag(bool bCopy) { Hide(); bool bDoDistort=DragStat().GetDX()!=0 || DragStat().GetDY()!=0; if (bDoDistort) { getSdrDragView().DistortMarkedObj(aMarkRect,aDistortedRect,!bContortion,bCopy); return true; } return false; } Pointer SdrDragDistort::GetSdrDragPointer() const { return Pointer(PointerStyle::RefHand); } void SdrDragDistort::applyCurrentTransformationToSdrObject(SdrObject& rTarget) { const bool bDoDistort(DragStat().GetDX()!=0 || DragStat().GetDY()!=0); if (bDoDistort) { SdrEditView::ImpDistortObj(&rTarget, aMarkRect, aDistortedRect, !bContortion); } } void SdrDragDistort::applyCurrentTransformationToPolyPolygon(basegfx::B2DPolyPolygon& rTarget) { // use helper derived from old stuff MovAllPoints(rTarget); } SdrDragCrop::SdrDragCrop(SdrDragView& rNewView) : SdrDragObjOwn(rNewView) { // switch off solid dragging for crop; it just makes no sense since showing // a 50% transparent object above the original will not be visible setSolidDraggingActive(false); } void SdrDragCrop::TakeSdrDragComment(OUString& rStr) const { ImpTakeDescriptionStr(STR_DragMethCrop, rStr); OUString aStr; rStr += " (x="; getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDX(), aStr); rStr += aStr + " y="; getSdrDragView().GetModel()->TakeMetricStr(DragStat().GetDY(), aStr); rStr += aStr + ")"; if(getSdrDragView().IsDragWithCopy()) rStr += ImpGetResStr(STR_EditWithCopy); } bool SdrDragCrop::BeginSdrDrag() { // call parent bool bRetval(SdrDragObjOwn::BeginSdrDrag()); if(!GetDragHdl()) { // we need the DragHdl, break if not there bRetval = false; } return bRetval; } bool SdrDragCrop::EndSdrDrag(bool /*bCopy*/) { Hide(); if( DragStat().GetDX()==0 && DragStat().GetDY()==0 ) return false; const SdrMarkList& rMarkList = getSdrDragView().GetMarkedObjectList(); if( rMarkList.GetMarkCount() != 1 ) return false; SdrObject* pSdrObject = rMarkList.GetMark( 0 )->GetMarkedSdrObj(); // tdf 34555: in order to implement visual crop in Writer, we need to handle two // cases: // EndSdrDrag when called in Impress/Draw/...: pSdrObject is a SdrGrafObj // EndSdrDrag when called in Writer: pSdrObject is a SwVirtFlyDrawObj // Main principle: if marked object is not SdrGrafObj, we start a generic handling // based on virtual methods added to SdrObject, on MM100/Twip coordinates and so on. // If marked object is SdrGrafObj, we do all the work here with matrix based // coordinates. if (dynamic_cast( pSdrObject) == nullptr) { const bool bUndo = getSdrDragView().IsUndoEnabled(); if( bUndo ) { OUString aUndoStr; ImpTakeDescriptionStr(STR_DragMethCrop, aUndoStr); getSdrDragView().BegUndo( aUndoStr ); getSdrDragView().AddUndo( getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pSdrObject)); // also need attr undo, the SdrGrafCropItem will be changed getSdrDragView().AddUndo( getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoAttrObject(*pSdrObject)); } // We need to produce a reference point and two (X & Y) scales SdrHdl* pRef1=GetHdlList().GetHdl(SdrHdlKind::UpperLeft); SdrHdl* pRef2=GetHdlList().GetHdl(SdrHdlKind::LowerRight); if (pRef1==nullptr || pRef2==nullptr) return false; Rectangle rect(pRef1->GetPos(),pRef2->GetPos()); Point aEnd(DragStat().GetNow()); Point aStart(DragStat().GetStart()); Point aRef(rect.Center()); // Reference point is the point opposed to the dragged handle switch(GetDragHdlKind()) { case SdrHdlKind::UpperLeft: aRef = rect.BottomRight(); break; case SdrHdlKind::Upper: aRef = rect.BottomCenter(); DragStat().SetHorFixed(true); break; case SdrHdlKind::UpperRight: aRef = rect.BottomLeft(); break; case SdrHdlKind::Left : aRef = rect.RightCenter(); DragStat().SetVerFixed(true); break; case SdrHdlKind::Right: aRef = rect.LeftCenter(); DragStat().SetVerFixed(true); break; case SdrHdlKind::LowerLeft: aRef = rect.TopRight(); break; case SdrHdlKind::Lower: aRef = rect.TopCenter(); DragStat().SetHorFixed(true); break; case SdrHdlKind::LowerRight: aRef = rect.TopLeft(); break; default: break; } // By default, scale is new size / old size long nXDiv = aStart.X()-aRef.X(); if (nXDiv==0) nXDiv=1; long nYDiv = aStart.Y()-aRef.Y(); if (nYDiv==0) nYDiv=1; long nXMul = aEnd.X()-aRef.X(); long nYMul = aEnd.Y()-aRef.Y(); if (nXDiv<0) { nXDiv=-nXDiv; nXMul=-nXMul; } if (nYDiv<0) { nYDiv=-nYDiv; nYMul=-nYMul; } // Take ortho into account. bool bXNeg=nXMul<0; if (bXNeg) nXMul=-nXMul; bool bYNeg=nYMul<0; if (bYNeg) nYMul=-nYMul; bool bOrtho=getSdrDragView().IsOrtho() || !getSdrDragView().IsResizeAllowed(); if (!DragStat().IsHorFixed() && !DragStat().IsVerFixed()) { if (std::abs(nXDiv)<=1 || std::abs(nYDiv)<=1) bOrtho=false; if (bOrtho) { if ((Fraction(nXMul,nXDiv)>Fraction(nYMul,nYDiv)) !=getSdrDragView().IsBigOrtho()) { nXMul=nYMul; nXDiv=nYDiv; } else { nYMul=nXMul; nYDiv=nXDiv; } } } else { if (bOrtho) { if (DragStat().IsHorFixed()) { bXNeg=false; nXMul=nYMul; nXDiv=nYDiv; } if (DragStat().IsVerFixed()) { bYNeg=false; nYMul=nXMul; nYDiv=nXDiv; } } else { if (DragStat().IsHorFixed()) { bXNeg=false; nXMul=1; nXDiv=1; } if (DragStat().IsVerFixed()) { bYNeg=false; nYMul=1; nYDiv=1; } } } Fraction aXFact(nXMul,nXDiv); Fraction aYFact(nYMul,nYDiv); Fraction aMaxFact(0x7FFFFFFF,1); if (bOrtho) { if (aXFact>aMaxFact) { aXFact=aMaxFact; aYFact=aMaxFact; } if (aYFact>aMaxFact) { aXFact=aMaxFact; aYFact=aMaxFact; } } if (bXNeg) aXFact=Fraction(-aXFact.GetNumerator(),aXFact.GetDenominator()); if (bYNeg) aYFact=Fraction(-aYFact.GetNumerator(),aYFact.GetDenominator()); // With Ref point (opposed to dragged point), X scale and Y scale, // we call crop (virtual method) on pSdrObject which calls VirtFlyDrawObj // crop pSdrObject->Crop(aRef, aXFact, aYFact); if( bUndo ) getSdrDragView().EndUndo(); // Job's done return true; } // This part of code handles the case where pSdrObject is SdrGrafObj SdrGrafObj* pObj = dynamic_cast( pSdrObject ); if( !pObj || (pObj->GetGraphicType() == GraphicType::NONE) || (pObj->GetGraphicType() == GraphicType::Default) ) return false; const GraphicObject& rGraphicObject = pObj->GetGraphicObject(); const MapMode aMapMode100thmm(MapUnit::Map100thMM); Size aGraphicSize(rGraphicObject.GetPrefSize()); if( MapUnit::MapPixel == rGraphicObject.GetPrefMapMode().GetMapUnit() ) aGraphicSize = Application::GetDefaultDevice()->PixelToLogic( aGraphicSize, aMapMode100thmm ); else aGraphicSize = OutputDevice::LogicToLogic( aGraphicSize, rGraphicObject.GetPrefMapMode(), aMapMode100thmm); if( aGraphicSize.Width() == 0 || aGraphicSize.Height() == 0 ) return false; const SdrGrafCropItem& rOldCrop = static_cast(pObj->GetMergedItem(SDRATTR_GRAFCROP)); const bool bUndo = getSdrDragView().IsUndoEnabled(); if( bUndo ) { OUString aUndoStr; ImpTakeDescriptionStr(STR_DragMethCrop, aUndoStr); getSdrDragView().BegUndo( aUndoStr ); getSdrDragView().AddUndo( getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoGeoObject(*pObj)); // also need attr undo, the SdrGrafCropItem will be changed getSdrDragView().AddUndo( getSdrDragView().GetModel()->GetSdrUndoFactory().CreateUndoAttrObject(*pObj)); } // new part to commute the user's drag activities // get the original objects transformation basegfx::B2DHomMatrix aOriginalMatrix; basegfx::B2DPolyPolygon aPolyPolygon; bool bShearCorrected(false); // get transformation from object pObj->TRGetBaseGeometry(aOriginalMatrix, aPolyPolygon); { // TTTT correct shear, it comes currently mirrored from TRGetBaseGeometry, can be removed with aw080 basegfx::B2DTuple aScale; basegfx::B2DTuple aTranslate; double fRotate(0.0), fShearX(0.0); aOriginalMatrix.decompose(aScale, aTranslate, fRotate, fShearX); if(!basegfx::fTools::equalZero(fShearX)) { bShearCorrected = true; aOriginalMatrix = basegfx::tools::createScaleShearXRotateTranslateB2DHomMatrix( aScale, -fShearX, fRotate, aTranslate); } } // invert it to be able to work on unit coordinates basegfx::B2DHomMatrix aInverse(aOriginalMatrix); aInverse.invert(); // generate start point of original drag vector in unit coordinates (the // vis-a-vis of the drag point) basegfx::B2DPoint aLocalStart(0.0, 0.0); bool bOnAxis(false); switch(GetDragHdlKind()) { case SdrHdlKind::UpperLeft: aLocalStart.setX(1.0); aLocalStart.setY(1.0); break; case SdrHdlKind::Upper: aLocalStart.setX(0.5); aLocalStart.setY(1.0); bOnAxis = true; break; case SdrHdlKind::UpperRight: aLocalStart.setX(0.0); aLocalStart.setY(1.0); break; case SdrHdlKind::Left : aLocalStart.setX(1.0); aLocalStart.setY(0.5); bOnAxis = true; break; case SdrHdlKind::Right: aLocalStart.setX(0.0); aLocalStart.setY(0.5); bOnAxis = true; break; case SdrHdlKind::LowerLeft: aLocalStart.setX(1.0); aLocalStart.setY(0.0); break; case SdrHdlKind::Lower: aLocalStart.setX(0.5); aLocalStart.setY(0.0); bOnAxis = true; break; case SdrHdlKind::LowerRight: aLocalStart.setX(0.0); aLocalStart.setY(0.0); break; default: break; } // create the current drag position in unit coordinates basegfx::B2DPoint aLocalCurrent(aInverse * basegfx::B2DPoint(DragStat().GetNow().X(), DragStat().GetNow().Y())); // if one of the edge handles is used, limit to X or Y drag only if(bOnAxis) { if(basegfx::fTools::equal(aLocalStart.getX(), 0.5)) { aLocalCurrent.setX(aLocalStart.getX()); } else { aLocalCurrent.setY(aLocalStart.getY()); } } // create internal change in unit coordinates basegfx::B2DHomMatrix aDiscreteChangeMatrix; if(!basegfx::fTools::equal(aLocalCurrent.getX(), aLocalStart.getX())) { if(aLocalStart.getX() < 0.5) { aDiscreteChangeMatrix.scale(aLocalCurrent.getX(), 1.0); } else { aDiscreteChangeMatrix.scale(1.0 - aLocalCurrent.getX(), 1.0); aDiscreteChangeMatrix.translate(aLocalCurrent.getX(), 0.0); } } if(!basegfx::fTools::equal(aLocalCurrent.getY(), aLocalStart.getY())) { if(aLocalStart.getY() < 0.5) { aDiscreteChangeMatrix.scale(1.0, aLocalCurrent.getY()); } else { aDiscreteChangeMatrix.scale(1.0, 1.0 - aLocalCurrent.getY()); aDiscreteChangeMatrix.translate(0.0, aLocalCurrent.getY()); } } // preparematrix to apply to object; evtl. back-correct shear basegfx::B2DHomMatrix aNewObjectMatrix(aOriginalMatrix * aDiscreteChangeMatrix); if(bShearCorrected) { // TTTT back-correct shear basegfx::B2DTuple aScale; basegfx::B2DTuple aTranslate; double fRotate(0.0), fShearX(0.0); aNewObjectMatrix.decompose(aScale, aTranslate, fRotate, fShearX); aNewObjectMatrix = basegfx::tools::createScaleShearXRotateTranslateB2DHomMatrix( aScale, -fShearX, fRotate, aTranslate); } // apply change to object by applying the unit coordinate change followed // by the original change pObj->TRSetBaseGeometry(aNewObjectMatrix, aPolyPolygon); // the following old code uses aOldRect/aNewRect to calculate the crop change for // the crop item. It implies unrotated objects, so create the unrotated original // rectangle and the unrotated modified rectangle. Latter can in case of shear and/or // rotation not be fetched by using //Rectangle aNewRect( pObj->GetLogicRect() ); // as it was done before because the top-left of that new rect *will* have an offset // caused by the evtl. existing shear and/or rotation, so calculate a unrotated // rectangle how it would be as a result when applying the unit coordinate change // to the unrotated original transformation. basegfx::B2DTuple aScale; basegfx::B2DTuple aTranslate; double fRotate, fShearX; // get access to scale and translate aOriginalMatrix.decompose(aScale, aTranslate, fRotate, fShearX); // prepare unsheared/unrotated versions of the old and new transformation const basegfx::B2DHomMatrix aMatrixOriginalNoShearNoRotate( basegfx::tools::createScaleTranslateB2DHomMatrix( basegfx::absolute(aScale), aTranslate)); // create the ranges for these basegfx::B2DRange aRangeOriginalNoShearNoRotate(0.0, 0.0, 1.0, 1.0); basegfx::B2DRange aRangeNewNoShearNoRotate(0.0, 0.0, 1.0, 1.0); aRangeOriginalNoShearNoRotate.transform(aMatrixOriginalNoShearNoRotate); aRangeNewNoShearNoRotate.transform(aMatrixOriginalNoShearNoRotate * aDiscreteChangeMatrix); // extract the old Rectangle structures Rectangle aOldRect( basegfx::fround(aRangeOriginalNoShearNoRotate.getMinX()), basegfx::fround(aRangeOriginalNoShearNoRotate.getMinY()), basegfx::fround(aRangeOriginalNoShearNoRotate.getMaxX()), basegfx::fround(aRangeOriginalNoShearNoRotate.getMaxY())); Rectangle aNewRect( basegfx::fround(aRangeNewNoShearNoRotate.getMinX()), basegfx::fround(aRangeNewNoShearNoRotate.getMinY()), basegfx::fround(aRangeNewNoShearNoRotate.getMaxX()), basegfx::fround(aRangeNewNoShearNoRotate.getMaxY())); // continue with the old original stuff if (!aOldRect.GetWidth() || !aOldRect.GetHeight()) throw o3tl::divide_by_zero(); double fScaleX = ( aGraphicSize.Width() - rOldCrop.GetLeft() - rOldCrop.GetRight() ) / (double)aOldRect.GetWidth(); double fScaleY = ( aGraphicSize.Height() - rOldCrop.GetTop() - rOldCrop.GetBottom() ) / (double)aOldRect.GetHeight(); sal_Int32 nDiffLeft = aNewRect.Left() - aOldRect.Left(); sal_Int32 nDiffTop = aNewRect.Top() - aOldRect.Top(); sal_Int32 nDiffRight = aNewRect.Right() - aOldRect.Right(); sal_Int32 nDiffBottom = aNewRect.Bottom() - aOldRect.Bottom(); if(pObj->IsMirrored()) { // mirrored X or Y, for old stuff, exchange X // TTTT: check for aw080 sal_Int32 nTmp(nDiffLeft); nDiffLeft = -nDiffRight; nDiffRight = -nTmp; } sal_Int32 nLeftCrop = static_cast( rOldCrop.GetLeft() + nDiffLeft * fScaleX ); sal_Int32 nTopCrop = static_cast( rOldCrop.GetTop() + nDiffTop * fScaleY ); sal_Int32 nRightCrop = static_cast( rOldCrop.GetRight() - nDiffRight * fScaleX ); sal_Int32 nBottomCrop = static_cast( rOldCrop.GetBottom() - nDiffBottom * fScaleY ); SfxItemPool& rPool = getSdrDragView().GetModel()->GetItemPool(); SfxItemSet aSet( rPool, SDRATTR_GRAFCROP, SDRATTR_GRAFCROP ); aSet.Put( SdrGrafCropItem( nLeftCrop, nTopCrop, nRightCrop, nBottomCrop ) ); getSdrDragView().SetAttributes( aSet, false ); if( bUndo ) getSdrDragView().EndUndo(); return true; } Pointer SdrDragCrop::GetSdrDragPointer() const { return Pointer(PointerStyle::Crop); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */