/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include SdrViewEvent::SdrViewEvent() : mpHdl(nullptr), mpObj(nullptr), mpRootObj(nullptr), mpPV(nullptr), mpURLField(nullptr), meHit(SdrHitKind::NONE), meEvent(SdrEventKind::NONE), mnMouseClicks(0), mnMouseMode(MouseEventModifiers::NONE), mnMouseCode(0), mnHlplIdx(0), mnGlueId(0), mbMouseDown(false), mbMouseUp(false), mbIsAction(false), mbIsTextEdit(false), mbAddMark(false), mbUnmark(false), mbPrevNextMark(false), mbMarkPrev(false) { } // helper class for all D&D overlays void SdrDropMarkerOverlay::ImplCreateOverlays( const SdrView& rView, const basegfx::B2DPolyPolygon& rLinePolyPolygon) { for(sal_uInt32 a(0); a < rView.PaintWindowCount(); a++) { SdrPaintWindow* pCandidate = rView.GetPaintWindow(a); const rtl::Reference< sdr::overlay::OverlayManager >& xTargetOverlay = pCandidate->GetOverlayManager(); if (xTargetOverlay.is()) { std::unique_ptr pNew(new sdr::overlay::OverlayPolyPolygonStripedAndFilled( rLinePolyPolygon)); xTargetOverlay->add(*pNew); maObjects.append(std::move(pNew)); } } } SdrDropMarkerOverlay::SdrDropMarkerOverlay(const SdrView& rView, const SdrObject& rObject) { ImplCreateOverlays( rView, rObject.TakeXorPoly()); } SdrDropMarkerOverlay::SdrDropMarkerOverlay(const SdrView& rView, const tools::Rectangle& rRectangle) { basegfx::B2DPolygon aB2DPolygon; aB2DPolygon.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Top())); aB2DPolygon.append(basegfx::B2DPoint(rRectangle.Right(), rRectangle.Top())); aB2DPolygon.append(basegfx::B2DPoint(rRectangle.Right(), rRectangle.Bottom())); aB2DPolygon.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Bottom())); aB2DPolygon.setClosed(true); ImplCreateOverlays( rView, basegfx::B2DPolyPolygon(aB2DPolygon)); } SdrDropMarkerOverlay::SdrDropMarkerOverlay(const SdrView& rView, const Point& rStart, const Point& rEnd) { basegfx::B2DPolygon aB2DPolygon; aB2DPolygon.append(basegfx::B2DPoint(rStart.X(), rStart.Y())); aB2DPolygon.append(basegfx::B2DPoint(rEnd.X(), rEnd.Y())); aB2DPolygon.setClosed(true); ImplCreateOverlays( rView, basegfx::B2DPolyPolygon(aB2DPolygon)); } SdrDropMarkerOverlay::~SdrDropMarkerOverlay() { // The OverlayObjects are cleared using the destructor of OverlayObjectList. // That destructor calls clear() at the list which removes all objects from the // OverlayManager and deletes them. } SdrView::SdrView( SdrModel& rSdrModel, OutputDevice* pOut) : SdrCreateView(rSdrModel, pOut), mbNoExtendedMouseDispatcher(false), mbNoExtendedKeyDispatcher(false), mbMasterPagePaintCaching(false) { } SdrView::~SdrView() { } bool SdrView::KeyInput(const KeyEvent& rKEvt, vcl::Window* pWin) { SetActualWin(pWin ? pWin->GetOutDev() : nullptr); bool bRet = SdrCreateView::KeyInput(rKEvt,pWin); if (!bRet && !IsExtendedKeyInputDispatcherEnabled()) { bRet = true; switch (rKEvt.GetKeyCode().GetFullFunction()) { case KeyFuncType::DELETE: DeleteMarked(); break; case KeyFuncType::UNDO: GetModel().Undo(); break; case KeyFuncType::REDO: GetModel().Redo(); break; default: { switch (rKEvt.GetKeyCode().GetFullCode()) { case KEY_ESCAPE: { if (IsTextEdit()) SdrEndTextEdit(); if (IsAction()) BrkAction(); if (pWin!=nullptr) pWin->ReleaseMouse(); } break; case KEY_DELETE: DeleteMarked(); break; case KEY_UNDO: case KEY_BACKSPACE+KEY_MOD2: GetModel().Undo(); break; case KEY_BACKSPACE+KEY_MOD2+KEY_SHIFT: GetModel().Redo(); break; case KEY_REPEAT: case KEY_BACKSPACE+KEY_MOD2+KEY_MOD1: GetModel().Repeat(*this); break; case KEY_MOD1+KEY_A: MarkAll(); break; default: bRet=false; } // switch } } // switch if (bRet && pWin!=nullptr) { pWin->SetPointer(GetPreferredPointer( pWin->PixelToLogic(pWin->ScreenToOutputPixel( pWin->GetPointerPosPixel() ) ), pWin->GetOutDev(), rKEvt.GetKeyCode().GetModifier())); } } return bRet; } bool SdrView::MouseButtonDown(const MouseEvent& rMEvt, OutputDevice* pWin) { SetActualWin(pWin); if (rMEvt.IsLeft()) maDragStat.SetMouseDown(true); bool bRet = SdrCreateView::MouseButtonDown(rMEvt,pWin); if (!bRet && !IsExtendedMouseEventDispatcherEnabled()) { SdrViewEvent aVEvt; PickAnything(rMEvt,SdrMouseEventKind::BUTTONDOWN,aVEvt); bRet = DoMouseEvent(aVEvt); } return bRet; } bool SdrView::MouseButtonUp(const MouseEvent& rMEvt, OutputDevice* pWin) { SetActualWin(pWin); if (rMEvt.IsLeft()) maDragStat.SetMouseDown(false); bool bAction = IsAction(); bool bRet = !bAction && SdrCreateView::MouseButtonUp(rMEvt,pWin); if (!bRet && !IsExtendedMouseEventDispatcherEnabled()) { SdrViewEvent aVEvt; PickAnything(rMEvt,SdrMouseEventKind::BUTTONUP,aVEvt); bRet = DoMouseEvent(aVEvt); } return bRet; } bool SdrView::MouseMove(const MouseEvent& rMEvt, OutputDevice* pWin) { SetActualWin(pWin); maDragStat.SetMouseDown(rMEvt.IsLeft()); bool bRet = SdrCreateView::MouseMove(rMEvt,pWin); if (!IsExtendedMouseEventDispatcherEnabled() && !IsTextEditInSelectionMode()) { SdrViewEvent aVEvt; PickAnything(rMEvt,SdrMouseEventKind::MOVE,aVEvt); if (DoMouseEvent(aVEvt)) bRet=true; } return bRet; } bool SdrView::Command(const CommandEvent& rCEvt, vcl::Window* pWin) { SetActualWin(pWin->GetOutDev()); bool bRet = SdrCreateView::Command(rCEvt,pWin); return bRet; } void SdrView::GetAttributes(SfxItemSet& rTargetSet, bool bOnlyHardAttr) const { SdrCreateView::GetAttributes(rTargetSet, bOnlyHardAttr); } SdrHitKind SdrView::PickAnything(const MouseEvent& rMEvt, SdrMouseEventKind nEventKind, SdrViewEvent& rVEvt) const { rVEvt.mbMouseDown = nEventKind==SdrMouseEventKind::BUTTONDOWN; rVEvt.mbMouseUp = nEventKind==SdrMouseEventKind::BUTTONUP; rVEvt.mnMouseClicks = rMEvt.GetClicks(); rVEvt.mnMouseMode = rMEvt.GetMode(); rVEvt.mnMouseCode = rMEvt.GetButtons() | rMEvt.GetModifier(); const OutputDevice* pOut=mpActualOutDev; if (pOut==nullptr) { pOut = GetFirstOutputDevice(); } Point aPnt(rMEvt.GetPosPixel()); if (pOut!=nullptr) aPnt=pOut->PixelToLogic(aPnt); if (mbNegativeX) { // Shape's x coordinates are all negated, // Hence negate mouse event's x coord to match. aPnt.setX(-aPnt.X()); } rVEvt.maLogicPos = aPnt; return PickAnything(aPnt,rVEvt); } // Dragging with the Mouse (Move) // Example when creating a rectangle: MouseDown has to happen without a ModKey, // else we usually force a selection (see below). // When pressing Shift, Ctrl and Alt at the same time while doing a MouseMove, // a centered, not snapped square is created. // The dual allocation of Ortho and Shift won't usually create a problem, as the // two functions are in most cases mutually exclusive. Only shearing (the kind // that happens when contorting, not when rotating) can use both functions at // the same time. To get around this, the user can use e. g. help lines. #define MODKEY_NoSnap bCtrl /* temporarily disable snapping */ #define MODKEY_Ortho bShift /* ortho */ #define MODKEY_Center bAlt /* create/resize centeredly */ #define MODKEY_AngleSnap bShift #define MODKEY_CopyDrag bCtrl /* drag and copy */ // click somewhere (MouseDown) #define MODKEY_PolyPoly bAlt /* new Poly at InsPt and at Create */ #define MODKEY_MultiMark bShift /* MarkObj without doing UnmarkAll first */ #define MODKEY_Unmark bAlt /* deselect with a dragged frame */ #define MODKEY_ForceMark bCtrl /* force dragging a frame, even if there's an object at cursor position */ #define MODKEY_DeepMark bAlt /* MarkNextObj */ #define MODKEY_DeepBackw bShift /* MarkNextObj but backwards */ SdrHitKind SdrView::PickAnything(const Point& rLogicPos, SdrViewEvent& rVEvt) const { const OutputDevice* pOut=mpActualOutDev; if (pOut==nullptr) { pOut = GetFirstOutputDevice(); } // #i73628# Use a non-changeable copy of he logic position const Point aLocalLogicPosition(rLogicPos); bool bEditMode=IsEditMode(); bool bPointMode=bEditMode && HasMarkablePoints(); bool bGluePointMode=IsGluePointEditMode(); bool bInsPolyPt=bPointMode && IsInsObjPointMode() && IsInsObjPointPossible(); bool bInsGluePt=bGluePointMode && IsInsGluePointMode() && IsInsGluePointPossible(); bool bIsTextEdit=IsTextEdit(); bool bTextEditHit=IsTextEditHit(aLocalLogicPosition); bool bTextEditSel=IsTextEditInSelectionMode(); bool bShift = (rVEvt.mnMouseCode & KEY_SHIFT) != 0; bool bCtrl = (rVEvt.mnMouseCode & KEY_MOD1) != 0; bool bAlt = (rVEvt.mnMouseCode & KEY_MOD2) != 0; SdrHitKind eHit=SdrHitKind::NONE; SdrHdl* pHdl=pOut!=nullptr && !bTextEditSel ? PickHandle(aLocalLogicPosition) : nullptr; SdrPageView* pPV=nullptr; SdrObject* pObj=nullptr; SdrObject* pHitObj=nullptr; bool bHitPassDirect=true; sal_uInt16 nHlplIdx=0; sal_uInt16 nGlueId=0; if (bTextEditHit || bTextEditSel) { eHit=SdrHitKind::TextEdit; } else if (pHdl!=nullptr) { eHit=SdrHitKind::Handle; // handle is hit: highest priority } else if (bEditMode && IsHlplVisible() && IsHlplFront() && pOut!=nullptr && PickHelpLine(aLocalLogicPosition,mnHitTolLog,*pOut,nHlplIdx,pPV)) { eHit=SdrHitKind::HelpLine; // help line in the foreground hit: can be moved now } else if (bGluePointMode && PickGluePoint(aLocalLogicPosition,pObj,nGlueId,pPV)) { eHit=SdrHitKind::Gluepoint; // deselected gluepoint hit } else if ((pHitObj = PickObj(aLocalLogicPosition,mnHitTolLog,pPV,SdrSearchOptions::DEEP|SdrSearchOptions::MARKED,&pObj,&bHitPassDirect))) { eHit=SdrHitKind::MarkedObject; sdr::table::SdrTableObj* pTableObj = dynamic_cast< sdr::table::SdrTableObj* >( pObj ); if( pTableObj ) { sal_Int32 nX = 0, nY = 0; switch( pTableObj->CheckTableHit( aLocalLogicPosition, nX, nY ) ) { case sdr::table::TableHitKind::Cell: eHit = SdrHitKind::Cell; break; case sdr::table::TableHitKind::CellTextArea: eHit = SdrHitKind::TextEditObj; break; default: break; } } } else if ((pHitObj = PickObj(aLocalLogicPosition,mnHitTolLog,pPV,SdrSearchOptions::DEEP|SdrSearchOptions::ALSOONMASTER|SdrSearchOptions::WHOLEPAGE,&pObj,&bHitPassDirect))) { // MasterPages and WholePage for Macro and URL eHit=SdrHitKind::UnmarkedObject; sdr::table::SdrTableObj* pTableObj = dynamic_cast< sdr::table::SdrTableObj* >( pObj ); if( pTableObj ) { sal_Int32 nX = 0, nY = 0; switch( pTableObj->CheckTableHit( aLocalLogicPosition, nX, nY, mnHitTolLog ) ) { case sdr::table::TableHitKind::Cell: eHit = SdrHitKind::Cell; break; case sdr::table::TableHitKind::CellTextArea: // Keep state on UnmarkedObject to allow the below // 'check for URL field' to be executed, else popups // for e.g. URL links when hoovering and clicking // them will not work. Tried several other changes, // but this one safely keeps existing behaviour as-is. // Except for the LOK. LOK doesn't have hoovering popup // feature. eHit = comphelper::LibreOfficeKit::isActive() ? SdrHitKind::TextEditObj : SdrHitKind::UnmarkedObject; break; default: break; } } } else if (bEditMode && IsHlplVisible() && !IsHlplFront() && pOut!=nullptr && PickHelpLine(aLocalLogicPosition,mnHitTolLog,*pOut,nHlplIdx,pPV)) { eHit=SdrHitKind::HelpLine; // help line in foreground hit: can be moved now } if (eHit==SdrHitKind::UnmarkedObject) { bool bRoot=pObj->HasMacro(); bool bDeep=pObj!=pHitObj && pHitObj->HasMacro(); bool bMid=false; // Have we hit upon a grouped group with a macro? SdrObject* pMidObj=nullptr; if (pObj!=pHitObj) { SdrObject* pObjTmp=pHitObj->getParentSdrObjectFromSdrObject(); if (pObjTmp==pObj) pObjTmp=nullptr; while (pObjTmp!=nullptr) { if (pObjTmp->HasMacro()) { bMid=true; pMidObj=pObjTmp; } pObjTmp=pObjTmp->getParentSdrObjectFromSdrObject(); if (pObjTmp==pObj) pObjTmp=nullptr; } } if (bDeep || bMid || bRoot) { SdrObjMacroHitRec aHitRec; aHitRec.aPos=aLocalLogicPosition; aHitRec.nTol=mnHitTolLog; aHitRec.pVisiLayer=&pPV->GetVisibleLayers(); aHitRec.pPageView=pPV; if (bDeep) bDeep=pHitObj->IsMacroHit(aHitRec); if (bMid ) bMid =pMidObj->IsMacroHit(aHitRec); if (bRoot) bRoot=pObj->IsMacroHit(aHitRec); if (bRoot || bMid || bDeep) { // Priorities: 1. Root, 2. Mid, 3. Deep rVEvt.mpRootObj = pObj; if (!bRoot) pObj=pMidObj; if (!bRoot && !bMid) pObj=pHitObj; eHit=SdrHitKind::Macro; } } } // check for URL field if (eHit==SdrHitKind::UnmarkedObject) { SdrTextObj* pTextObj=DynCastSdrTextObj( pHitObj ); if (pTextObj!=nullptr && pTextObj->HasText()) { // use the primitive-based HitTest which is more accurate anyways. It // will correctly handle rotated/mirrored/sheared/scaled text and can // now return a HitContainer containing the primitive hierarchy of the // primitive that triggered the hit. The first entry is that primitive, // the others are the full stack of primitives leading to that one which // includes grouping primitives (like TextHierarchyPrimitives we deed here) // but also all decomposed ones which lead to the creation of that primitive drawinglayer::primitive2d::Primitive2DContainer aHitContainer; const bool bTEHit(pPV && SdrObjectPrimitiveHit(*pTextObj, aLocalLogicPosition, {0, 0}, *pPV, &pPV->GetVisibleLayers(), true, &aHitContainer)); if (bTEHit && !aHitContainer.empty()) { // search for TextHierarchyFieldPrimitive2D which contains the needed information // about a possible URLField const drawinglayer::primitive2d::TextHierarchyFieldPrimitive2D* pTextHierarchyFieldPrimitive2D = nullptr; for (const drawinglayer::primitive2d::Primitive2DReference& xReference : aHitContainer) { auto pBasePrimitive = xReference.get(); if (pBasePrimitive && pBasePrimitive->getPrimitive2DID() == PRIMITIVE2D_ID_TEXTHIERARCHYFIELDPRIMITIVE2D) { pTextHierarchyFieldPrimitive2D = static_cast(pBasePrimitive); break; } } if (nullptr != pTextHierarchyFieldPrimitive2D) { if (drawinglayer::primitive2d::FieldType::FIELD_TYPE_URL == pTextHierarchyFieldPrimitive2D->getType()) { // problem with the old code is that a *pointer* to an instance of // SvxURLField is set in the Event which is per se not good since that // data comes from a temporary EditEngine's data and could vanish any // moment. Have to replace for now with a static instance that gets // filled/initialized from the original data held in the TextHierarchyField- // Primitive2D (see impTextBreakupHandler::impCheckFieldPrimitive). // Unfortunately things like 'TargetFrame' are still used in Calc, so this // can currently not get replaced. For the future the Name/Value vector or // the TextHierarchyFieldPrimitive2D itself should/will be used for handling // that data static SvxURLField aSvxURLField; aSvxURLField.SetURL(pTextHierarchyFieldPrimitive2D->getValue(u"URL"_ustr)); aSvxURLField.SetRepresentation(pTextHierarchyFieldPrimitive2D->getValue(u"Representation"_ustr)); aSvxURLField.SetTargetFrame(pTextHierarchyFieldPrimitive2D->getValue(u"TargetFrame"_ustr)); const OUString aFormat(pTextHierarchyFieldPrimitive2D->getValue(u"SvxURLFormat"_ustr)); if (!aFormat.isEmpty()) { aSvxURLField.SetFormat(static_cast(aFormat.toInt32())); } // set HitKind and pointer to local static instance in the Event // to comply to old stuff eHit = SdrHitKind::UrlField; rVEvt.mpURLField = &aSvxURLField; } } } } if (eHit==SdrHitKind::UnmarkedObject && !pHitObj->getHyperlink().isEmpty()) { static SvxURLField aSvxURLField; aSvxURLField.SetURL(pHitObj->getHyperlink()); rVEvt.mpURLField = &aSvxURLField; eHit = SdrHitKind::UrlField; } } if (bHitPassDirect && (eHit==SdrHitKind::MarkedObject || eHit==SdrHitKind::UnmarkedObject) && (IsTextTool() || (IsEditMode() && IsQuickTextEditMode())) && pHitObj->HasTextEdit()) { auto pTextObj = DynCastSdrTextObj(pHitObj); // Around the TextEditArea there's a border to select without going into text edit mode. tools::Rectangle aBoundRect; const GeoStat& rGeo = pTextObj->GetGeoStat(); if (pTextObj && !rGeo.m_nRotationAngle && !rGeo.m_nShearAngle) { pTextObj->TakeTextEditArea(nullptr, nullptr, &aBoundRect, nullptr); } else aBoundRect = pHitObj->GetCurrentBoundRect(); // Force to SnapRect when Fontwork if( pTextObj && pTextObj->IsFontwork() ) aBoundRect = pHitObj->GetSnapRect(); sal_Int32 nTolerance(mnHitTolLog); bool bBoundRectHit(false); if(pOut) { nTolerance = pOut->PixelToLogic(Size(2, 0)).Width(); } if( (aLocalLogicPosition.X() >= aBoundRect.Left() - nTolerance && aLocalLogicPosition.X() <= aBoundRect.Left() + nTolerance) || (aLocalLogicPosition.X() >= aBoundRect.Right() - nTolerance && aLocalLogicPosition.X() <= aBoundRect.Right() + nTolerance) || (aLocalLogicPosition.Y() >= aBoundRect.Top() - nTolerance && aLocalLogicPosition.Y() <= aBoundRect.Top() + nTolerance) || (aLocalLogicPosition.Y() >= aBoundRect.Bottom() - nTolerance && aLocalLogicPosition.Y() <= aBoundRect.Bottom() + nTolerance)) { bBoundRectHit = true; } if(!bBoundRectHit && aBoundRect.Contains(aLocalLogicPosition)) { bool bTEHit(pPV && SdrObjectPrimitiveHit(*pHitObj, aLocalLogicPosition, { 2000.0, 0.0 }, *pPV, &pPV->GetVisibleLayers(), true)); // TextEdit attached to an object in a locked layer if (bTEHit && pPV->GetLockedLayers().IsSet(pHitObj->GetLayer())) { bTEHit=false; } if (bTEHit) { rVEvt.mpRootObj=pObj; pObj=pHitObj; eHit=SdrHitKind::TextEditObj; } } } if (!bHitPassDirect && eHit==SdrHitKind::UnmarkedObject) { eHit=SdrHitKind::NONE; pObj=nullptr; pPV=nullptr; } bool bMouseLeft = (rVEvt.mnMouseCode & MOUSE_LEFT) != 0; bool bMouseRight = (rVEvt.mnMouseCode & MOUSE_RIGHT) != 0; bool bMouseDown = rVEvt.mbMouseDown; bool bMouseUp = rVEvt.mbMouseUp; SdrEventKind eEvent=SdrEventKind::NONE; bool bIsAction=IsAction(); if (bIsAction) { if (bMouseDown) { if (bMouseRight) eEvent=SdrEventKind::BackAction; } else if (bMouseUp) { if (bMouseLeft) { eEvent=SdrEventKind::EndAction; if (IsDragObj()) { eEvent=SdrEventKind::EndDrag; } else if (IsCreateObj() || IsInsObjPoint()) { eEvent=IsCreateObj() ? SdrEventKind::EndCreate : SdrEventKind::EndInsertObjPoint; } else if (IsMarking()) { eEvent=SdrEventKind::EndMark; if (!maDragStat.IsMinMoved()) { eEvent=SdrEventKind::BrkMark; rVEvt.mbAddMark = MODKEY_MultiMark; } } } } else { eEvent=SdrEventKind::MoveAction; } } else if (eHit==SdrHitKind::TextEdit) { eEvent=SdrEventKind::TextEdit; } else if (bMouseDown && bMouseLeft) { if (rVEvt.mnMouseClicks == 2 && rVEvt.mnMouseCode == MOUSE_LEFT && pObj!=nullptr && pHitObj!=nullptr && pHitObj->HasTextEdit() && eHit==SdrHitKind::MarkedObject) { rVEvt.mpRootObj = pObj; pObj=pHitObj; eEvent=SdrEventKind::BeginTextEdit; } else if (MODKEY_ForceMark && eHit!=SdrHitKind::UrlField) { eEvent=SdrEventKind::BeginMark; // AddMark,Unmark */ } else if (eHit==SdrHitKind::HelpLine) { eEvent=SdrEventKind::BeginDragHelpline; // nothing, actually } else if (eHit==SdrHitKind::Gluepoint) { eEvent=SdrEventKind::MarkGluePoint; // AddMark+Drag rVEvt.mbAddMark = MODKEY_MultiMark || MODKEY_DeepMark; // if not hit with Deep } else if (eHit==SdrHitKind::Handle) { eEvent=SdrEventKind::BeginDragObj; // Mark+Drag,AddMark+Drag,DeepMark+Drag,Unmark bool bGlue=pHdl->GetKind()==SdrHdlKind::Glue; bool bPoly=!bGlue && IsPointMarkable(*pHdl); bool bMarked=bGlue || (bPoly && pHdl->IsSelected()); if (bGlue || bPoly) { eEvent=bGlue ? SdrEventKind::MarkGluePoint : SdrEventKind::MarkPoint; if (MODKEY_DeepMark) { rVEvt.mbAddMark = true; rVEvt.mbPrevNextMark = true; rVEvt.mbMarkPrev = MODKEY_DeepBackw; } else if (MODKEY_MultiMark) { rVEvt.mbAddMark = true; rVEvt.mbUnmark = bMarked; // Toggle if (bGlue) { pObj=pHdl->GetObj(); nGlueId=static_cast(pHdl->GetObjHdlNum()); } } else if (bMarked) { eEvent=SdrEventKind::BeginDragObj; // don't change MarkState, only change Drag } } } else if (bInsPolyPt && (MODKEY_PolyPoly || (!MODKEY_MultiMark && !MODKEY_DeepMark))) { eEvent=SdrEventKind::BeginInsertObjPoint; } else if (bInsGluePt && !MODKEY_MultiMark && !MODKEY_DeepMark) { eEvent=SdrEventKind::BeginInsertGluePoint; } else if (eHit==SdrHitKind::TextEditObj) { eEvent=SdrEventKind::BeginTextEdit; // AddMark+Drag,DeepMark+Drag,Unmark if (MODKEY_MultiMark || MODKEY_DeepMark) { // if not hit with Deep eEvent=SdrEventKind::MarkObj; } } else if (eHit==SdrHitKind::Macro) { eEvent=SdrEventKind::BeginMacroObj; // AddMark+Drag if (MODKEY_MultiMark || MODKEY_DeepMark) { // if not hit with Deep eEvent=SdrEventKind::MarkObj; } } else if (eHit==SdrHitKind::UrlField) { eEvent=SdrEventKind::ExecuteUrl; // AddMark+Drag if (MODKEY_MultiMark || MODKEY_DeepMark) { // if not hit with Deep eEvent=SdrEventKind::MarkObj; } } else if (eHit==SdrHitKind::MarkedObject) { eEvent=SdrEventKind::BeginDragObj; // DeepMark+Drag,Unmark if (MODKEY_MultiMark || MODKEY_DeepMark) { // if not hit with Deep eEvent=SdrEventKind::MarkObj; } } else if (IsCreateMode()) { eEvent=SdrEventKind::BeginCreateObj; // nothing, actually } else if (eHit==SdrHitKind::UnmarkedObject) { eEvent=SdrEventKind::MarkObj; // AddMark+Drag } else { eEvent=SdrEventKind::BeginMark; } if (eEvent==SdrEventKind::MarkObj) { rVEvt.mbAddMark = MODKEY_MultiMark || MODKEY_DeepMark; // if not hit with Deep rVEvt.mbPrevNextMark = MODKEY_DeepMark; rVEvt.mbMarkPrev = MODKEY_DeepMark && MODKEY_DeepBackw; } if (eEvent==SdrEventKind::BeginMark) { rVEvt.mbAddMark = MODKEY_MultiMark; rVEvt.mbUnmark = MODKEY_Unmark; } } rVEvt.mbIsAction = bIsAction; rVEvt.mbIsTextEdit = bIsTextEdit; rVEvt.maLogicPos = aLocalLogicPosition; rVEvt.mpHdl = pHdl; rVEvt.mpObj = pObj; if (rVEvt.mpRootObj == nullptr) rVEvt.mpRootObj = pObj; rVEvt.mpPV = pPV; rVEvt.mnHlplIdx = nHlplIdx; rVEvt.mnGlueId = nGlueId; rVEvt.meHit = eHit; rVEvt.meEvent = eEvent; #ifdef DGB_UTIL if (rVEvt.mpRootObj != nullptr) { if (rVEvt.mpRootObj->getParentSdrObjListFromSdrObject() != rVEvt.mpPV->GetObjList()) { OSL_FAIL("SdrView::PickAnything(): pRootObj->getParentSdrObjListFromSdrObject()!=pPV->GetObjList() !"); } } #endif return eHit; } bool SdrView::DoMouseEvent(const SdrViewEvent& rVEvt) { bool bRet=false; SdrHitKind eHit = rVEvt.meHit; Point aLogicPos(rVEvt.maLogicPos); bool bShift = (rVEvt.mnMouseCode & KEY_SHIFT) != 0; bool bCtrl = (rVEvt.mnMouseCode & KEY_MOD1) != 0; bool bAlt = (rVEvt.mnMouseCode & KEY_MOD2) != 0; bool bMouseLeft = (rVEvt.mnMouseCode & MOUSE_LEFT) != 0; bool bMouseDown = rVEvt.mbMouseDown; bool bMouseUp = rVEvt.mbMouseUp; if (bMouseDown) { if (bMouseLeft) maDragStat.SetMouseDown(true); } else if (bMouseUp) { if (bMouseLeft) maDragStat.SetMouseDown(false); } else { // else, MouseMove maDragStat.SetMouseDown(bMouseLeft); } #ifdef MODKEY_NoSnap SetSnapEnabled(!MODKEY_NoSnap); #endif #ifdef MODKEY_Ortho SetOrtho(MODKEY_Ortho!=IsOrthoDesired()); #endif #ifdef MODKEY_AngleSnap SetAngleSnapEnabled(MODKEY_AngleSnap); #endif #ifdef MODKEY_CopyDrag SetDragWithCopy(MODKEY_CopyDrag); #endif #ifdef MODKEY_Center SetCreate1stPointAsCenter(MODKEY_Center); SetResizeAtCenter(MODKEY_Center); SetCrookAtCenter(MODKEY_Center); #endif if (bMouseLeft && bMouseDown && rVEvt.mbIsTextEdit && (eHit==SdrHitKind::UnmarkedObject || eHit==SdrHitKind::NONE)) { SdrEndTextEdit(); // User has clicked beneath object, exit edit mode. // pHdl is invalid, then, that shouldn't matter, though, as we expect // pHdl==NULL (because of eHit). } switch (rVEvt.meEvent) { case SdrEventKind::NONE: bRet=false; break; case SdrEventKind::TextEdit: bRet=false; break; // Events handled by the OutlinerView are not taken into account here. case SdrEventKind::MoveAction: MovAction(aLogicPos); bRet=true; break; case SdrEventKind::EndAction: EndAction(); bRet=true; break; case SdrEventKind::BackAction: BckAction(); bRet=true; break; case SdrEventKind::EndMark : EndAction(); bRet=true; break; case SdrEventKind::BrkMark : { BrkAction(); if (!MarkObj(aLogicPos, mnHitTolLog, rVEvt.mbAddMark)) { // No object hit. Do the following: // 1. deselect any selected gluepoints // 2. deselect any selected polygon points // 3. deselect any selected objects if (!rVEvt.mbAddMark) UnmarkAll(); } bRet=true; } break; case SdrEventKind::EndCreate: { // if necessary, MarkObj SdrCreateCmd eCmd=SdrCreateCmd::NextPoint; if (MODKEY_PolyPoly) eCmd=SdrCreateCmd::NextObject; if (rVEvt.mnMouseClicks > 1) eCmd=SdrCreateCmd::ForceEnd; if (!EndCreateObj(eCmd)) { // Don't evaluate event for Create? -> Select if (eHit==SdrHitKind::UnmarkedObject || eHit==SdrHitKind::TextEdit) { MarkObj(rVEvt.mpRootObj, rVEvt.mpPV); if (eHit==SdrHitKind::TextEdit) { bool bRet2(mpActualOutDev && OUTDEV_WINDOW == mpActualOutDev->GetOutDevType() && SdrBeginTextEdit(rVEvt.mpObj, rVEvt.mpPV, mpActualOutDev->GetOwnerWindow())); if(bRet2) { MouseEvent aMEvt(mpActualOutDev->LogicToPixel(aLogicPos), 1, rVEvt.mnMouseMode,rVEvt.mnMouseCode,rVEvt.mnMouseCode); OutlinerView* pOLV=GetTextEditOutlinerView(); if (pOLV!=nullptr) { pOLV->MouseButtonDown(aMEvt); // event for the Outliner, but without double-click pOLV->MouseButtonUp(aMEvt); // event for the Outliner, but without double-click } } } bRet=true; // object is selected and (if necessary) TextEdit is started } else bRet=false; // canceled Create, nothing else } else bRet=true; // return true for EndCreate } break; case SdrEventKind::EndDrag: { bRet=EndDragObj(IsDragWithCopy()); ForceMarkedObjToAnotherPage(); // TODO: Undo+bracing missing! } break; case SdrEventKind::MarkObj: { // + (if applicable) BegDrag if (!rVEvt.mbAddMark) UnmarkAllObj(); bool bUnmark = rVEvt.mbUnmark; if (rVEvt.mbPrevNextMark) { bRet=MarkNextObj(aLogicPos, mnHitTolLog, rVEvt.mbMarkPrev); } else { const SdrMarkList& rMarkList = GetMarkedObjectList(); rMarkList.ForceSort(); const size_t nCount0=rMarkList.GetMarkCount(); bRet=MarkObj(aLogicPos, mnHitTolLog, rVEvt.mbAddMark); rMarkList.ForceSort(); const size_t nCount1=rMarkList.GetMarkCount(); bUnmark=nCount1 1) eCmd = SdrCreateCmd::ForceEnd; EndInsObjPoint(eCmd); bRet=true; } break; case SdrEventKind::BeginInsertGluePoint: bRet=BegInsGluePoint(aLogicPos); break; case SdrEventKind::BeginDragHelpline: bRet = BegDragHelpLine(rVEvt.mnHlplIdx,rVEvt.mpPV); break; case SdrEventKind::BeginDragObj: bRet=BegDragObj(aLogicPos, nullptr, rVEvt.mpHdl, mnMinMovLog); break; case SdrEventKind::BeginCreateObj: { if (mnCurrentInvent==SdrInventor::Default && mnCurrentIdent==SdrObjKind::Caption) { tools::Long nHgt=SdrEngineDefaults::GetFontHeight(); bRet=BegCreateCaptionObj(aLogicPos,Size(5*nHgt,2*nHgt)); } else bRet=BegCreateObj(aLogicPos); } break; case SdrEventKind::BeginMacroObj: { BegMacroObj(aLogicPos, mnHitTolLog, rVEvt.mpObj, rVEvt.mpPV, mpActualOutDev->GetOwnerWindow()); bRet=false; } break; case SdrEventKind::BeginTextEdit: { if (!IsObjMarked(rVEvt.mpObj)) { UnmarkAllObj(); MarkObj(rVEvt.mpRootObj,rVEvt.mpPV); } bRet = mpActualOutDev && OUTDEV_WINDOW == mpActualOutDev->GetOutDevType()&& SdrBeginTextEdit(rVEvt.mpObj, rVEvt.mpPV, mpActualOutDev->GetOwnerWindow()); if(bRet) { MouseEvent aMEvt(mpActualOutDev->LogicToPixel(aLogicPos), 1, rVEvt.mnMouseMode, rVEvt.mnMouseCode, rVEvt.mnMouseCode); OutlinerView* pOLV=GetTextEditOutlinerView(); if (pOLV!=nullptr) pOLV->MouseButtonDown(aMEvt); // event for the Outliner, but without double-click } } break; default: break; } // switch if (bRet && mpActualOutDev && mpActualOutDev->GetOutDevType()==OUTDEV_WINDOW) { vcl::Window* pWin=mpActualOutDev->GetOwnerWindow(); // left mouse button pressed? bool bLeftDown = (rVEvt.mnMouseCode & MOUSE_LEFT) != 0 && rVEvt.mbMouseDown; // left mouse button released? bool bLeftUp = (rVEvt.mnMouseCode & MOUSE_LEFT) != 0 && rVEvt.mbMouseUp; // left mouse button pressed or held? bool bLeftDown1=(rVEvt.mnMouseCode & MOUSE_LEFT) != 0 && !rVEvt.mbMouseUp; pWin->SetPointer(GetPreferredPointer(rVEvt.maLogicPos, pWin->GetOutDev(), rVEvt.mnMouseCode & (KEY_SHIFT|KEY_MOD1|KEY_MOD2),bLeftDown1)); bool bAction=IsAction(); if (bLeftDown && bAction) pWin->CaptureMouse(); else if (bLeftUp || (rVEvt.mbIsAction && !bAction)) pWin->ReleaseMouse(); } return bRet; } PointerStyle SdrView::GetPreferredPointer(const Point& rMousePos, const OutputDevice* pOut, sal_uInt16 nModifier, bool bLeftDown) const { // Actions if (IsCreateObj()) { return mpCurrentCreate->GetCreatePointer(); } if (mpCurrentSdrDragMethod) { return mpCurrentSdrDragMethod->GetSdrDragPointer(); } if (IsMarkObj() || IsMarkPoints() || IsMarkGluePoints() || IsSetPageOrg()) return PointerStyle::Arrow; if (IsDragHelpLine()) return GetDraggedHelpLinePointer(); if (IsMacroObj()) { SdrObjMacroHitRec aHitRec; aHitRec.aPos=pOut->LogicToPixel(rMousePos); aHitRec.nTol=nMacroTol; aHitRec.pVisiLayer=&pMacroPV->GetVisibleLayers(); aHitRec.pPageView=pMacroPV; return pMacroObj->GetMacroPointer(aHitRec); } // TextEdit, ObjEdit, Macro if (IsTextEdit() && (IsTextEditInSelectionMode() || IsTextEditHit(rMousePos))) { if(!pOut || IsTextEditInSelectionMode()) { if (mpTextEditOutliner->IsVertical()) return PointerStyle::TextVertical; else return PointerStyle::Text; } // Outliner should return something here... Point aPos(pOut->LogicToPixel(rMousePos)); PointerStyle aPointer(mpTextEditOutlinerView->GetPointer(aPos)); if (aPointer==PointerStyle::Arrow) { if (mpTextEditOutliner->IsVertical()) aPointer = PointerStyle::TextVertical; else aPointer = PointerStyle::Text; } return aPointer; } SdrViewEvent aVEvt; aVEvt.mnMouseCode = (nModifier&(KEY_SHIFT|KEY_MOD1|KEY_MOD2))|MOUSE_LEFT; // to see what would happen on MouseLeftDown aVEvt.mbMouseDown = !bLeftDown; // What if ..? aVEvt.mbMouseUp = bLeftDown; // What if ..? if (pOut!=nullptr) const_cast(this)->SetActualWin(pOut); SdrHitKind eHit=PickAnything(rMousePos,aVEvt); SdrEventKind eEvent = aVEvt.meEvent; switch (eEvent) { case SdrEventKind::BeginCreateObj: return maCurrentCreatePointer; case SdrEventKind::MarkObj: return PointerStyle::Move; case SdrEventKind::BeginMark: return PointerStyle::Arrow; case SdrEventKind::MarkPoint: case SdrEventKind::MarkGluePoint: return PointerStyle::MovePoint; case SdrEventKind::BeginInsertObjPoint: case SdrEventKind::BeginInsertGluePoint: return PointerStyle::Cross; case SdrEventKind::ExecuteUrl: return PointerStyle::RefHand; case SdrEventKind::BeginMacroObj: { SdrObjMacroHitRec aHitRec; aHitRec.aPos = aVEvt.maLogicPos; aHitRec.nTol=mnHitTolLog; aHitRec.pVisiLayer = &aVEvt.mpPV->GetVisibleLayers(); aHitRec.pPageView = aVEvt.mpPV; return aVEvt.mpObj->GetMacroPointer(aHitRec); } default: break; } // switch switch(eHit) { case SdrHitKind::Cell: return PointerStyle::Arrow; case SdrHitKind::HelpLine : return aVEvt.mpPV->GetHelpLines()[aVEvt.mnHlplIdx].GetPointer(); case SdrHitKind::Gluepoint: return PointerStyle::MovePoint; case SdrHitKind::TextEdit : case SdrHitKind::TextEditObj: { SdrTextObj* pText = DynCastSdrTextObj(aVEvt.mpObj); if(pText && pText->HasText()) { OutlinerParaObject* pParaObj = pText->GetOutlinerParaObject(); if(pParaObj && pParaObj->IsEffectivelyVertical()) return PointerStyle::TextVertical; } return PointerStyle::Text; } default: break; } bool bMarkHit=eHit==SdrHitKind::MarkedObject; SdrHdl* pHdl = aVEvt.mpHdl; // now check the pointers for dragging if (pHdl!=nullptr || bMarkHit) { SdrHdlKind eHdl= pHdl!=nullptr ? pHdl->GetKind() : SdrHdlKind::Move; bool bCorner=pHdl!=nullptr && pHdl->IsCornerHdl(); bool bVertex=pHdl!=nullptr && pHdl->IsVertexHdl(); bool bMov=eHdl==SdrHdlKind::Move; if (bMov && (meDragMode==SdrDragMode::Move || meDragMode==SdrDragMode::Resize || mbMarkedHitMovesAlways)) { if (!IsMoveAllowed()) return PointerStyle::Arrow; // because double click or drag & drop is possible return PointerStyle::Move; } switch (meDragMode) { case SdrDragMode::Rotate: { if ((bCorner || bMov) && !IsRotateAllowed(true)) return PointerStyle::NotAllowed; // are 3D objects selected? bool b3DObjSelected = false; const SdrMarkList& rMarkList = GetMarkedObjectList(); for (size_t a=0; !b3DObjSelected && aGetMarkedSdrObj(); if(DynCastE3dObject(pObj)) b3DObjSelected = true; } // If we have a 3D object, go on despite !IsShearAllowed, // because then we have a rotation instead of a shear. if (bVertex && !IsShearAllowed() && !b3DObjSelected) return PointerStyle::NotAllowed; if (bMov) return PointerStyle::Rotate; } break; case SdrDragMode::Shear: { if (bCorner) { if (!IsDistortAllowed(true) && !IsDistortAllowed()) return PointerStyle::NotAllowed; else return PointerStyle::RefHand; } if (bVertex && !IsShearAllowed()) return PointerStyle::NotAllowed; if (bMov) { if (!IsMoveAllowed()) return PointerStyle::Arrow; // because double click or drag & drop is possible return PointerStyle::Move; } } break; case SdrDragMode::Mirror: { if (bCorner || bVertex || bMov) { SdrHdl* pH1=maHdlList.GetHdl(SdrHdlKind::Ref1); SdrHdl* pH2=maHdlList.GetHdl(SdrHdlKind::Ref2); bool b90=false; bool b45=false; if (pH1!=nullptr && pH2!=nullptr) { Point aDif = pH2->GetPos()-pH1->GetPos(); b90=(aDif.X()==0) || aDif.Y()==0; b45=b90 || (std::abs(aDif.X())==std::abs(aDif.Y())); } bool bNo=false; if (!IsMirrorAllowed(true,true)) bNo=true; // any mirroring is forbidden if (!IsMirrorAllowed() && !b45) bNo=true; // mirroring freely is forbidden if (!IsMirrorAllowed(true) && !b90) bNo=true; // mirroring horizontally/vertically is allowed if (bNo) return PointerStyle::NotAllowed; if (b90) { return PointerStyle::Mirror; } return PointerStyle::Mirror; } } break; case SdrDragMode::Transparence: { if(!IsTransparenceAllowed()) return PointerStyle::NotAllowed; return PointerStyle::RefHand; } case SdrDragMode::Gradient: { if(!IsGradientAllowed()) return PointerStyle::NotAllowed; return PointerStyle::RefHand; } case SdrDragMode::Crook: { if (bCorner || bVertex || bMov) { if (!IsCrookAllowed(true) && !IsCrookAllowed()) return PointerStyle::NotAllowed; return PointerStyle::Crook; } break; } case SdrDragMode::Crop: { return PointerStyle::Crop; } default: { if ((bCorner || bVertex) && !IsResizeAllowed(true)) return PointerStyle::NotAllowed; } } if (pHdl!=nullptr) return pHdl->GetPointer(); if (bMov) { if (!IsMoveAllowed()) return PointerStyle::Arrow; // because double click or drag & drop is possible return PointerStyle::Move; } } if (meEditMode==SdrViewEditMode::Create) return maCurrentCreatePointer; return PointerStyle::Arrow; } constexpr OUString STR_NOTHING = u"nothing"_ustr; OUString SdrView::GetStatusText() { OUString aName; OUString aStr = STR_NOTHING; if (mpCurrentCreate!=nullptr) { aStr=mpCurrentCreate->getSpecialDragComment(maDragStat); if(aStr.isEmpty()) { aName = mpCurrentCreate->TakeObjNameSingul(); aStr = SvxResId(STR_ViewCreateObj); } } else if (mpCurrentSdrDragMethod) { if (mbInsPolyPoint || IsInsertGluePoint()) { aStr=maInsPointUndoStr; } else { if (maDragStat.IsMinMoved()) { SAL_INFO( "svx.svdraw", "(" << this << ") " << mpCurrentSdrDragMethod.get()); aStr = mpCurrentSdrDragMethod->GetSdrDragComment(); } } } else if(IsMarkObj()) { const SdrMarkList& rMarkList = GetMarkedObjectList(); if(rMarkList.GetMarkCount() != 0) { aStr = SvxResId(STR_ViewMarkMoreObjs); } else { aStr = SvxResId(STR_ViewMarkObjs); } } else if(IsMarkPoints()) { if(HasMarkedPoints()) { aStr = SvxResId(STR_ViewMarkMorePoints); } else { aStr = SvxResId(STR_ViewMarkPoints); } } else if (IsMarkGluePoints()) { if(HasMarkedGluePoints()) { aStr = SvxResId(STR_ViewMarkMoreGluePoints); } else { aStr = SvxResId(STR_ViewMarkGluePoints); } } else if (IsTextEdit() && mpTextEditOutlinerView != nullptr) { aStr=SvxResId(STR_ViewTextEdit); // "TextEdit - Row y, Column x"; ESelection aSel(mpTextEditOutlinerView->GetSelection()); tools::Long nPar = aSel.nEndPara,nLin=0,nCol=aSel.nEndPos; if (aSel.nEndPara>0) { for (sal_Int32 nParaNum=0; nParaNumGetLineCount(nParaNum); } } // A little imperfection: // At the end of a line of any multi-line paragraph, we display the // position of the next line of the same paragraph, if there is one. sal_uInt16 nParaLine = 0; sal_uInt32 nParaLineCount = mpTextEditOutliner->GetLineCount(aSel.nEndPara); bool bBrk = false; while (!bBrk) { sal_uInt16 nLen = mpTextEditOutliner->GetLineLen(aSel.nEndPara, nParaLine); bool bLastLine = (nParaLine == nParaLineCount - 1); if (nCol>nLen || (!bLastLine && nCol == nLen)) { nCol -= nLen; nLin++; nParaLine++; } else bBrk = true; if (nLen == 0) bBrk = true; // to be sure } aStr = aStr.replaceFirst("%1", OUString::number(nPar + 1)); aStr = aStr.replaceFirst("%2", OUString::number(nLin + 1)); aStr = aStr.replaceFirst("%3", OUString::number(nCol + 1)); #ifdef DBG_UTIL aStr += ", Level " + OUString::number(mpTextEditOutliner->GetDepth( aSel.nEndPara )); #endif } if(aStr == STR_NOTHING) { const SdrMarkList& rMarkList = GetMarkedObjectList(); if (rMarkList.GetMarkCount() != 0) { aStr = ImpGetDescriptionString(STR_ViewMarked); if (IsGluePointEditMode()) { if (HasMarkedGluePoints()) { aStr = ImpGetDescriptionString(STR_ViewMarked, ImpGetDescriptionOptions::GLUEPOINTS); } } else { if (HasMarkedPoints()) { aStr = ImpGetDescriptionString(STR_ViewMarked, ImpGetDescriptionOptions::POINTS); } } } else { aStr.clear(); } } else if(!aName.isEmpty()) { aStr = aStr.replaceFirst("%1", aName); } if(!aStr.isEmpty()) { // capitalize first letter aStr = aStr.replaceAt(0, 1, OUString(aStr[0]).toAsciiUpperCase()); } return aStr; } SdrViewContext SdrView::GetContext() const { if( IsGluePointEditMode() ) return SdrViewContext::GluePointEdit; const SdrMarkList& rMarkList = GetMarkedObjectList(); const size_t nMarkCount = rMarkList.GetMarkCount(); if( HasMarkablePoints() && !IsFrameHandles() ) { bool bPath=true; for( size_t nMarkNum = 0; nMarkNum < nMarkCount && bPath; ++nMarkNum ) if (dynamic_cast(rMarkList.GetMark(nMarkNum)->GetMarkedSdrObj()) == nullptr) bPath=false; if( bPath ) return SdrViewContext::PointEdit; } if( rMarkList.GetMarkCount() ) { bool bGraf = true, bMedia = true, bTable = true; for( size_t nMarkNum = 0; nMarkNum < nMarkCount && ( bGraf || bMedia ); ++nMarkNum ) { const SdrObject* pMarkObj = rMarkList.GetMark(nMarkNum)->GetMarkedSdrObj(); DBG_ASSERT( pMarkObj, "SdrView::GetContext(), null pointer in mark list!" ); if( !pMarkObj ) continue; if( dynamic_cast( pMarkObj) == nullptr ) bGraf = false; if( dynamic_cast( pMarkObj) == nullptr ) bMedia = false; if( dynamic_cast( pMarkObj ) == nullptr ) bTable = false; } if( bGraf ) return SdrViewContext::Graphic; else if( bMedia ) return SdrViewContext::Media; else if( bTable ) return SdrViewContext::Table; } return SdrViewContext::Standard; } void SdrView::MarkAll() { if (IsTextEdit()) { GetTextEditOutlinerView()->SetSelection(ESelection::All()); } else if (IsGluePointEditMode()) MarkAllGluePoints(); else if (HasMarkablePoints()) MarkAllPoints(); else { // check for table bool bMarkAll = true; const SdrMarkList& rMarkList = GetMarkedObjectList(); if (rMarkList.GetMarkCount() == 1) { const SdrObject* pObj(rMarkList.GetMark(0)->GetMarkedSdrObj()); SdrView* pView = this; if (pObj && pView && (pObj->GetObjInventor() == SdrInventor::Default) && (pObj->GetObjIdentifier() == SdrObjKind::Table)) { mxSelectionController.clear(); mxSelectionController = sdr::table::CreateTableController( *pView, static_cast(*pObj), mxLastSelectionController); if (mxSelectionController.is()) { mxLastSelectionController.clear(); mxSelectionController->onSelectAll(); bMarkAll = false; } } } if ( bMarkAll ) MarkAllObj(); } } void SdrView::UnmarkAll() { if (IsTextEdit()) { ESelection eSel=GetTextEditOutlinerView()->GetSelection(); eSel.nStartPara=eSel.nEndPara; eSel.nStartPos=eSel.nEndPos; GetTextEditOutlinerView()->SetSelection(eSel); } else if (HasMarkedGluePoints()) UnmarkAllGluePoints(); else if (HasMarkedPoints()) UnmarkAllPoints(); // Marked, not Markable! else UnmarkAllObj(); } const tools::Rectangle& SdrView::GetMarkedRect() const { if (IsGluePointEditMode() && HasMarkedGluePoints()) { return GetMarkedGluePointsRect(); } if (HasMarkedPoints()) { return GetMarkedPointsRect(); } return GetMarkedObjRect(); } void SdrView::DeleteMarked() { if (IsTextEdit()) { SdrObjEditView::KeyInput(KeyEvent(0, vcl::KeyCode(KeyFuncType::DELETE)), mpTextEditWin); } else { if( mxSelectionController.is() && mxSelectionController->DeleteMarked() ) { // action already performed by current selection controller, do nothing } else if (IsGluePointEditMode() && HasMarkedGluePoints()) { DeleteMarkedGluePoints(); } else if (GetContext()==SdrViewContext::PointEdit && HasMarkedPoints()) { DeleteMarkedPoints(); } else { DeleteMarkedObj(); } } } bool SdrView::BegMark(const Point& rPnt, bool bAddMark, bool bUnmark) { if (bUnmark) bAddMark=true; if (IsGluePointEditMode()) { if (!bAddMark) UnmarkAllGluePoints(); return BegMarkGluePoints(rPnt,bUnmark); } else if (HasMarkablePoints()) { if (!bAddMark) UnmarkAllPoints(); return BegMarkPoints(rPnt,bUnmark); } else { if (!bAddMark) UnmarkAllObj(); BegMarkObj(rPnt,bUnmark); return true; } } bool SdrView::MoveShapeHandle(const sal_uInt32 handleNum, const Point& aEndPoint, const sal_Int32 aObjectOrdNum) { if (GetHdlList().IsMoveOutside()) return false; const SdrMarkList& rMarkList = GetMarkedObjectList(); if (!rMarkList.GetMarkCount()) return false; SdrHdl * pHdl = GetHdlList().GetHdl(handleNum); if (pHdl == nullptr) return false; SdrDragStat& rDragStat = const_cast(GetDragStat()); // start dragging BegDragObj(pHdl->GetPos(), nullptr, pHdl, 0); if (!IsDragObj()) return false; bool bWasNoSnap = rDragStat.IsNoSnap(); bool bWasSnapEnabled = IsSnapEnabled(); // switch snapping off if(!bWasNoSnap) rDragStat.SetNoSnap(); if(bWasSnapEnabled) SetSnapEnabled(false); if (aObjectOrdNum != -1) { rDragStat.GetGlueOptions().objectOrdNum = aObjectOrdNum; } MovDragObj(aEndPoint); EndDragObj(); // Clear Glue Options rDragStat.GetGlueOptions().objectOrdNum = -1; if (!bWasNoSnap) rDragStat.SetNoSnap(bWasNoSnap); if (bWasSnapEnabled) SetSnapEnabled(bWasSnapEnabled); return true; } void SdrView::SetMasterPagePaintCaching(bool bOn) { if(mbMasterPagePaintCaching == bOn) return; mbMasterPagePaintCaching = bOn; // reset at all SdrPageWindows SdrPageView* pPageView = GetSdrPageView(); if(!pPageView) return; for(sal_uInt32 b(0); b < pPageView->PageWindowCount(); b++) { SdrPageWindow* pPageWindow = pPageView->GetPageWindow(b); assert(pPageWindow && "SdrView::SetMasterPagePaintCaching: Corrupt SdrPageWindow list (!)"); // force deletion of ObjectContact, so at re-display all VOCs // will be re-created with updated flag setting pPageWindow->ResetObjectContact(); } // force redraw of this view pPageView->InvalidateAllWin(); } // Default ObjectContact is ObjectContactOfPageView sdr::contact::ObjectContact* SdrView::createViewSpecificObjectContact( SdrPageWindow& rPageWindow, const char* pDebugName) const { return new sdr::contact::ObjectContactOfPageView(rPageWindow, pDebugName); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */