/* -*- 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 "UndoGuard.hxx" #include #include #include #include "DragMethod_PieSegment.hxx" #include "DragMethod_RotateDiagram.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "DrawCommandDispatch.hxx" #include #include "ControllerCommandDispatch.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define DRGPIX 2 // Drag MinMove in Pixel using namespace ::com::sun::star; using namespace ::com::sun::star::chart2; using namespace ::chart::DataSeriesProperties; using ::com::sun::star::uno::Reference; namespace chart { namespace { bool lcl_GrowAndShiftLogic( RelativePosition & rInOutRelPos, RelativeSize & rInOutRelSize, const awt::Size & rRefSize, double fGrowLogicX, double fGrowLogicY ) { if( rRefSize.Width == 0 || rRefSize.Height == 0 ) return false; double fRelativeGrowX = fGrowLogicX / rRefSize.Width; double fRelativeGrowY = fGrowLogicY / rRefSize.Height; return ::chart::RelativePositionHelper::centerGrow( rInOutRelPos, rInOutRelSize, fRelativeGrowX, fRelativeGrowY ); } bool lcl_MoveObjectLogic( RelativePosition & rInOutRelPos, RelativeSize const & rObjectSize, const awt::Size & rRefSize, double fShiftLogicX, double fShiftLogicY ) { if( rRefSize.Width == 0 || rRefSize.Height == 0 ) return false; double fRelativeShiftX = fShiftLogicX / rRefSize.Width; double fRelativeShiftY = fShiftLogicY / rRefSize.Height; return ::chart::RelativePositionHelper::moveObject( rInOutRelPos, rObjectSize, fRelativeShiftX, fRelativeShiftY ); } void lcl_insertMenuCommand( const uno::Reference< awt::XPopupMenu > & xMenu, sal_Int16 nId, const OUString & rCommand ) { xMenu->insertItem( nId, "", 0, -1 ); xMenu->setCommand( nId, rCommand ); } OUString lcl_getFormatCommandForObjectCID( std::u16string_view rCID ) { OUString aDispatchCommand( ".uno:FormatSelection" ); ObjectType eObjectType = ObjectIdentifier::getObjectType( rCID ); switch(eObjectType) { case OBJECTTYPE_DIAGRAM: case OBJECTTYPE_DIAGRAM_WALL: aDispatchCommand = ".uno:FormatWall"; break; case OBJECTTYPE_DIAGRAM_FLOOR: aDispatchCommand = ".uno:FormatFloor"; break; case OBJECTTYPE_PAGE: aDispatchCommand = ".uno:FormatChartArea"; break; case OBJECTTYPE_LEGEND: aDispatchCommand = ".uno:FormatLegend"; break; case OBJECTTYPE_TITLE: aDispatchCommand = ".uno:FormatTitle"; break; case OBJECTTYPE_LEGEND_ENTRY: aDispatchCommand = ".uno:FormatDataSeries"; break; case OBJECTTYPE_AXIS: case OBJECTTYPE_AXIS_UNITLABEL: aDispatchCommand = ".uno:FormatAxis"; break; case OBJECTTYPE_GRID: aDispatchCommand = ".uno:FormatMajorGrid"; break; case OBJECTTYPE_SUBGRID: aDispatchCommand = ".uno:FormatMinorGrid"; break; case OBJECTTYPE_DATA_LABELS: aDispatchCommand = ".uno:FormatDataLabels"; break; case OBJECTTYPE_DATA_SERIES: aDispatchCommand = ".uno:FormatDataSeries"; break; case OBJECTTYPE_DATA_LABEL: aDispatchCommand = ".uno:FormatDataLabel"; break; case OBJECTTYPE_DATA_POINT: aDispatchCommand = ".uno:FormatDataPoint"; break; case OBJECTTYPE_DATA_AVERAGE_LINE: aDispatchCommand = ".uno:FormatMeanValue"; break; case OBJECTTYPE_DATA_ERRORS_X: aDispatchCommand = ".uno:FormatXErrorBars"; break; case OBJECTTYPE_DATA_ERRORS_Y: aDispatchCommand = ".uno:FormatYErrorBars"; break; case OBJECTTYPE_DATA_ERRORS_Z: aDispatchCommand = ".uno:FormatZErrorBars"; break; case OBJECTTYPE_DATA_CURVE: aDispatchCommand = ".uno:FormatTrendline"; break; case OBJECTTYPE_DATA_CURVE_EQUATION: aDispatchCommand = ".uno:FormatTrendlineEquation"; break; case OBJECTTYPE_DATA_STOCK_RANGE: aDispatchCommand = ".uno:FormatSelection"; break; case OBJECTTYPE_DATA_STOCK_LOSS: aDispatchCommand = ".uno:FormatStockLoss"; break; case OBJECTTYPE_DATA_STOCK_GAIN: aDispatchCommand = ".uno:FormatStockGain"; break; default: //OBJECTTYPE_UNKNOWN break; } return aDispatchCommand; } } // anonymous namespace // awt::XWindow void SAL_CALL ChartController::setPosSize( sal_Int32 X, sal_Int32 Y, sal_Int32 Width, sal_Int32 Height, sal_Int16 Flags ) { SolarMutexGuard aGuard; uno::Reference xWindow = m_xViewWindow; auto pChartWindow(GetChartWindow()); if(!(xWindow.is() && pChartWindow)) return; Size aLogicSize = pChartWindow->PixelToLogic( Size( Width, Height ), MapMode( MapUnit::Map100thMM ) ); //todo: for standalone chart: detect whether we are standalone //change map mode to fit new size awt::Size aModelPageSize = ChartModelHelper::getPageSize( getChartModel() ); sal_Int32 nScaleXNumerator = aLogicSize.Width(); sal_Int32 nScaleXDenominator = aModelPageSize.Width; sal_Int32 nScaleYNumerator = aLogicSize.Height(); sal_Int32 nScaleYDenominator = aModelPageSize.Height; MapMode aNewMapMode( MapUnit::Map100thMM, Point(0,0), Fraction(nScaleXNumerator, nScaleXDenominator), Fraction(nScaleYNumerator, nScaleYDenominator) ); pChartWindow->SetMapMode(aNewMapMode); pChartWindow->setPosSizePixel( X, Y, Width, Height, static_cast(Flags) ); //#i75867# poor quality of ole's alternative view with 3D scenes and zoomfactors besides 100% if( m_xChartView.is() ) { auto aZoomFactors(::comphelper::InitPropertySequence({ { "ScaleXNumerator", uno::Any( nScaleXNumerator ) }, { "ScaleXDenominator", uno::Any( nScaleXDenominator ) }, { "ScaleYNumerator", uno::Any( nScaleYNumerator ) }, { "ScaleYDenominator", uno::Any( nScaleYDenominator ) } })); m_xChartView->setPropertyValue( "ZoomFactors", uno::Any( aZoomFactors )); } //a correct work area is at least necessary for correct values in the position and size dialog and for dragging area if(m_pDrawViewWrapper) { tools::Rectangle aRect(Point(0,0), pChartWindow->GetOutDev()->GetOutputSize()); m_pDrawViewWrapper->SetWorkArea( aRect ); } pChartWindow->Invalidate(); } awt::Rectangle SAL_CALL ChartController::getPosSize() { //@todo awt::Rectangle aRet(0, 0, 0, 0); uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) aRet = xWindow->getPosSize(); return aRet; } void SAL_CALL ChartController::setVisible( sal_Bool Visible ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->setVisible( Visible ); } void SAL_CALL ChartController::setEnable( sal_Bool Enable ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->setEnable( Enable ); } void SAL_CALL ChartController::setFocus() { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->setFocus(); } void SAL_CALL ChartController::addWindowListener( const uno::Reference< awt::XWindowListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->addWindowListener( xListener ); } void SAL_CALL ChartController::removeWindowListener( const uno::Reference< awt::XWindowListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->removeWindowListener( xListener ); } void SAL_CALL ChartController::addFocusListener( const uno::Reference< awt::XFocusListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->addFocusListener( xListener ); } void SAL_CALL ChartController::removeFocusListener( const uno::Reference< awt::XFocusListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->removeFocusListener( xListener ); } void SAL_CALL ChartController::addKeyListener( const uno::Reference< awt::XKeyListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->addKeyListener( xListener ); } void SAL_CALL ChartController::removeKeyListener( const uno::Reference< awt::XKeyListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->removeKeyListener( xListener ); } void SAL_CALL ChartController::addMouseListener( const uno::Reference< awt::XMouseListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->addMouseListener( xListener ); } void SAL_CALL ChartController::removeMouseListener( const uno::Reference< awt::XMouseListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->removeMouseListener( xListener ); } void SAL_CALL ChartController::addMouseMotionListener( const uno::Reference< awt::XMouseMotionListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->addMouseMotionListener( xListener ); } void SAL_CALL ChartController::removeMouseMotionListener( const uno::Reference< awt::XMouseMotionListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->removeMouseMotionListener( xListener ); } void SAL_CALL ChartController::addPaintListener( const uno::Reference< awt::XPaintListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->addPaintListener( xListener ); } void SAL_CALL ChartController::removePaintListener( const uno::Reference< awt::XPaintListener >& xListener ) { //@todo uno::Reference xWindow = m_xViewWindow; if(xWindow.is()) xWindow->removePaintListener( xListener ); } // impl vcl window controller methods void ChartController::PrePaint() { // forward VCLs PrePaint window event to DrawingLayer DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); if (pDrawViewWrapper) { pDrawViewWrapper->PrePaint(); } } void ChartController::execute_Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle& rRect) { try { rtl::Reference xModel(getChartModel()); //OSL_ENSURE( xModel.is(), "ChartController::execute_Paint: have no model to paint"); if (!xModel.is()) return; //better performance for big data if (m_xChartView.is()) { awt::Size aResolution(1000, 1000); { SolarMutexGuard aGuard; auto pChartWindow(GetChartWindow()); if (pChartWindow) { aResolution.Width = pChartWindow->GetSizePixel().Width(); aResolution.Height = pChartWindow->GetSizePixel().Height(); } } m_xChartView->setPropertyValue( "Resolution", uno::Any( aResolution )); } if (m_xChartView.is()) m_xChartView->update(); { SolarMutexGuard aGuard; DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); if (pDrawViewWrapper) pDrawViewWrapper->CompleteRedraw(&rRenderContext, vcl::Region(rRect)); } } catch( const uno::Exception & ) { DBG_UNHANDLED_EXCEPTION("chart2"); } catch( ... ) { } } static bool isDoubleClick( const MouseEvent& rMEvt ) { return rMEvt.GetClicks() == 2 && rMEvt.IsLeft() && !rMEvt.IsMod1() && !rMEvt.IsMod2() && !rMEvt.IsShift(); } void ChartController::startDoubleClickWaiting() { SolarMutexGuard aGuard; m_bWaitingForDoubleClick = true; sal_uInt64 nDblClkTime = 500; auto pChartWindow(GetChartWindow()); if( pChartWindow ) { const MouseSettings& rMSettings = pChartWindow->GetSettings().GetMouseSettings(); nDblClkTime = rMSettings.GetDoubleClickTime(); } m_aDoubleClickTimer.SetTimeout( nDblClkTime ); m_aDoubleClickTimer.Start(); } void ChartController::stopDoubleClickWaiting() { m_aDoubleClickTimer.Stop(); m_bWaitingForDoubleClick = false; } IMPL_LINK_NOARG(ChartController, DoubleClickWaitingHdl, Timer *, void) { m_bWaitingForDoubleClick = false; if( m_bWaitingForMouseUp || !m_aSelection.maybeSwitchSelectionAfterSingleClickWasEnsured() ) return; impl_selectObjectAndNotiy(); SolarMutexGuard aGuard; auto pChartWindow(GetChartWindow()); if( pChartWindow ) { vcl::Window::PointerState aPointerState( pChartWindow->GetPointerState() ); MouseEvent aMouseEvent( aPointerState.maPos, 1/*nClicks*/, MouseEventModifiers::NONE, static_cast< sal_uInt16 >( aPointerState.mnState )/*nButtons*/, 0/*nModifier*/ ); impl_SetMousePointer( aMouseEvent ); } } void ChartController::execute_MouseButtonDown( const MouseEvent& rMEvt ) { SolarMutexGuard aGuard; m_bWaitingForMouseUp = true; m_bFieldButtonDown = false; if( isDoubleClick(rMEvt) ) stopDoubleClickWaiting(); else startDoubleClickWaiting(); m_aSelection.remindSelectionBeforeMouseDown(); DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); auto pChartWindow(GetChartWindow()); if(!pChartWindow || !pDrawViewWrapper ) return; Point aMPos = pChartWindow->PixelToLogic(rMEvt.GetPosPixel()); // Check if button was clicked SdrObject* pObject = pDrawViewWrapper->getHitObject(aMPos); if (pObject) { OUString aCID = pObject->GetName(); if (aCID.startsWith("FieldButton")) { m_bFieldButtonDown = true; return; // Don't take any action if button was clicked } } if ( rMEvt.GetButtons() == MOUSE_LEFT ) { pChartWindow->GrabFocus(); pChartWindow->CaptureMouse(); } if( pDrawViewWrapper->IsTextEdit() ) { SdrViewEvent aVEvt; if ( pDrawViewWrapper->IsTextEditHit( aMPos ) || // #i12587# support for shapes in chart ( rMEvt.IsRight() && pDrawViewWrapper->PickAnything( rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt ) == SdrHitKind::MarkedObject ) ) { pDrawViewWrapper->MouseButtonDown(rMEvt, pChartWindow->GetOutDev()); return; } else { EndTextEdit(); } } //abort running action if( pDrawViewWrapper->IsAction() ) { if( rMEvt.IsRight() ) pDrawViewWrapper->BckAction(); return; } if( isDoubleClick(rMEvt) ) //do not change selection if double click return;//double click is handled further in mousebutton up SdrHdl* pHitSelectionHdl = nullptr; //switch from move to resize if handle is hit on a resizable object if( m_aSelection.isResizeableObjectSelected() ) pHitSelectionHdl = pDrawViewWrapper->PickHandle( aMPos ); //only change selection if no selection handles are hit if( !pHitSelectionHdl ) { // #i12587# support for shapes in chart if ( m_eDrawMode == CHARTDRAW_INSERT && ( !pDrawViewWrapper->IsMarkedHit( aMPos ) || !m_aSelection.isDragableObjectSelected() ) ) { if ( m_aSelection.hasSelection() ) { m_aSelection.clearSelection(); } if ( !pDrawViewWrapper->IsAction() ) { if ( pDrawViewWrapper->GetCurrentObjIdentifier() == SdrObjKind::Caption ) { Size aCaptionSize( 2268, 1134 ); pDrawViewWrapper->BegCreateCaptionObj( aMPos, aCaptionSize ); } else { pDrawViewWrapper->BegCreateObj( aMPos); } SdrObject* pObj = pDrawViewWrapper->GetCreateObj(); DrawCommandDispatch* pDrawCommandDispatch = m_aDispatchContainer.getDrawCommandDispatch(); if ( pObj && m_pDrawModelWrapper && pDrawCommandDispatch ) { SfxItemSet aSet( m_pDrawModelWrapper->GetItemPool() ); pDrawCommandDispatch->setAttributes( pObj ); pDrawCommandDispatch->setLineEnds( aSet ); pObj->SetMergedItemSet( aSet ); } } impl_SetMousePointer( rMEvt ); return; } m_aSelection.adaptSelectionToNewPos( aMPos, pDrawViewWrapper, rMEvt.IsRight(), m_bWaitingForDoubleClick ); if( !m_aSelection.isRotateableObjectSelected( getChartModel() ) ) { m_eDragMode = SdrDragMode::Move; pDrawViewWrapper->SetDragMode(m_eDragMode); } m_aSelection.applySelection(pDrawViewWrapper); } if( m_aSelection.isDragableObjectSelected() && !rMEvt.IsRight() ) { //start drag sal_uInt16 nDrgLog = static_cast(pChartWindow->PixelToLogic(Size(DRGPIX,0)).Width()); SdrDragMethod* pDragMethod = nullptr; //change selection to 3D scene if rotate mode SdrDragMode eDragMode = pDrawViewWrapper->GetDragMode(); if( eDragMode==SdrDragMode::Rotate ) { E3dScene* pScene = SelectionHelper::getSceneToRotate( pDrawViewWrapper->getNamedSdrObject( m_aSelection.getSelectedCID() ) ); if( pScene ) { DragMethod_RotateDiagram::RotationDirection eRotationDirection(DragMethod_RotateDiagram::ROTATIONDIRECTION_FREE); if(pHitSelectionHdl) { SdrHdlKind eKind = pHitSelectionHdl->GetKind(); if( eKind==SdrHdlKind::Upper || eKind==SdrHdlKind::Lower ) eRotationDirection = DragMethod_RotateDiagram::ROTATIONDIRECTION_X; else if( eKind==SdrHdlKind::Left || eKind==SdrHdlKind::Right ) eRotationDirection = DragMethod_RotateDiagram::ROTATIONDIRECTION_Y; else if( eKind==SdrHdlKind::UpperLeft || eKind==SdrHdlKind::UpperRight || eKind==SdrHdlKind::LowerLeft || eKind==SdrHdlKind::LowerRight ) eRotationDirection = DragMethod_RotateDiagram::ROTATIONDIRECTION_Z; } pDragMethod = new DragMethod_RotateDiagram( *pDrawViewWrapper, m_aSelection.getSelectedCID(), getChartModel(), eRotationDirection ); } } else { std::u16string_view aDragMethodServiceName( ObjectIdentifier::getDragMethodServiceName( m_aSelection.getSelectedCID() ) ); if( aDragMethodServiceName == ObjectIdentifier::getPieSegmentDragMethodServiceName() ) pDragMethod = new DragMethod_PieSegment( *pDrawViewWrapper, m_aSelection.getSelectedCID(), getChartModel() ); } pDrawViewWrapper->SdrView::BegDragObj(aMPos, nullptr, pHitSelectionHdl, nDrgLog, pDragMethod); } impl_SetMousePointer( rMEvt ); } void ChartController::execute_MouseMove( const MouseEvent& rMEvt ) { SolarMutexGuard aGuard; DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); auto pChartWindow(GetChartWindow()); if(!pChartWindow || !pDrawViewWrapper) return; if( m_pDrawViewWrapper->IsTextEdit() ) { if( m_pDrawViewWrapper->MouseMove(rMEvt,pChartWindow->GetOutDev()) ) return; } if(pDrawViewWrapper->IsAction()) { pDrawViewWrapper->MovAction( pChartWindow->PixelToLogic( rMEvt.GetPosPixel() ) ); } impl_SetMousePointer( rMEvt ); } void ChartController::execute_MouseButtonUp( const MouseEvent& rMEvt ) { ControllerLockGuardUNO aCLGuard( getChartModel() ); bool bMouseUpWithoutMouseDown = !m_bWaitingForMouseUp; m_bWaitingForMouseUp = false; bool bNotifySelectionChange = false; { SolarMutexGuard aGuard; DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); auto pChartWindow(GetChartWindow()); if(!pChartWindow || !pDrawViewWrapper) return; Point aMPos = pChartWindow->PixelToLogic(rMEvt.GetPosPixel()); // Check if button was clicked if (m_bFieldButtonDown) { m_bFieldButtonDown = false; SdrObject* pObject = pDrawViewWrapper->getHitObject(aMPos); if (pObject) { OUString aCID = pObject->GetName(); if (aCID.startsWith("FieldButton")) { sendPopupRequest(aCID, pObject->GetCurrentBoundRect()); return; } } } if(pDrawViewWrapper->IsTextEdit()) { if( pDrawViewWrapper->MouseButtonUp(rMEvt,pChartWindow->GetOutDev()) ) return; } // #i12587# support for shapes in chart if ( m_eDrawMode == CHARTDRAW_INSERT && pDrawViewWrapper->IsCreateObj() ) { pDrawViewWrapper->EndCreateObj( SdrCreateCmd::ForceEnd ); { HiddenUndoContext aUndoContext( m_xUndoManager ); // don't want the positioning Undo action to appear in the UI impl_switchDiagramPositioningToExcludingPositioning(); } if ( pDrawViewWrapper->AreObjectsMarked() ) { if ( pDrawViewWrapper->GetCurrentObjIdentifier() == SdrObjKind::Text ) { executeDispatch_EditText(); } else { SdrObject* pObj = pDrawViewWrapper->getSelectedObject(); if ( pObj ) { uno::Reference< drawing::XShape > xShape( pObj->getUnoShape(), uno::UNO_QUERY ); if ( xShape.is() ) { m_aSelection.setSelection( xShape ); m_aSelection.applySelection( pDrawViewWrapper ); } } } } else { m_aSelection.adaptSelectionToNewPos( aMPos, pDrawViewWrapper, rMEvt.IsRight(), m_bWaitingForDoubleClick ); m_aSelection.applySelection( pDrawViewWrapper ); setDrawMode( CHARTDRAW_SELECT ); } } else if ( pDrawViewWrapper->IsDragObj() ) { bool bDraggingDone = false; SdrDragMethod* pDragMethod = pDrawViewWrapper->SdrView::GetDragMethod(); bool bIsMoveOnly = pDragMethod && pDragMethod->getMoveOnly(); DragMethod_Base* pChartDragMethod = dynamic_cast< DragMethod_Base* >(pDragMethod); if( pChartDragMethod ) { UndoGuard aUndoGuard( pChartDragMethod->getUndoDescription(), m_xUndoManager ); if( pDrawViewWrapper->EndDragObj() ) { bDraggingDone = true; aUndoGuard.commit(); } } if( !bDraggingDone && pDrawViewWrapper->EndDragObj() ) { try { //end move or size SdrObject* pObj = pDrawViewWrapper->getSelectedObject(); if( pObj ) { tools::Rectangle aObjectRect = pObj->GetSnapRect(); tools::Rectangle aOldObjectRect = pObj->GetLastBoundRect(); awt::Size aPageSize( ChartModelHelper::getPageSize( getChartModel() ) ); tools::Rectangle aPageRect( 0,0,aPageSize.Width,aPageSize.Height ); const E3dObject* pE3dObject(DynCastE3dObject(pObj)); if(nullptr != pE3dObject) { E3dScene* pScene(pE3dObject->getRootE3dSceneFromE3dObject()); if(nullptr != pScene) { aObjectRect = pScene->GetSnapRect(); } } ActionDescriptionProvider::ActionType eActionType(ActionDescriptionProvider::ActionType::Move); if( !bIsMoveOnly && m_aSelection.isResizeableObjectSelected() ) eActionType = ActionDescriptionProvider::ActionType::Resize; ObjectType eObjectType = ObjectIdentifier::getObjectType( m_aSelection.getSelectedCID() ); UndoGuard aUndoGuard( ActionDescriptionProvider::createDescription( eActionType, ObjectNameProvider::getName( eObjectType)), m_xUndoManager ); bool bChanged = false; rtl::Reference< ChartModel > xModel = getChartModel(); if ( eObjectType == OBJECTTYPE_LEGEND ) bChanged = DiagramHelper::switchDiagramPositioningToExcludingPositioning( *xModel, false , true ); bool bMoved = PositionAndSizeHelper::moveObject( m_aSelection.getSelectedCID() , xModel , awt::Rectangle(aObjectRect.Left(),aObjectRect.Top(),aObjectRect.getOpenWidth(),aObjectRect.getOpenHeight()) , awt::Rectangle(aOldObjectRect.Left(), aOldObjectRect.Top(), 0, 0) , awt::Rectangle(aPageRect.Left(),aPageRect.Top(),aPageRect.getOpenWidth(),aPageRect.getOpenHeight()) ); if( bMoved || bChanged ) { bDraggingDone = true; aUndoGuard.commit(); } } } catch( const uno::Exception & ) { DBG_UNHANDLED_EXCEPTION("chart2"); } //all wanted model changes will take effect //and all unwanted view modifications are cleaned } if( !bDraggingDone ) //mouse wasn't moved while dragging { bool bClickedTwiceOnDragableObject = SelectionHelper::isDragableObjectHitTwice( aMPos, m_aSelection.getSelectedCID(), *pDrawViewWrapper ); bool bIsRotateable = m_aSelection.isRotateableObjectSelected( getChartModel() ); //toggle between move and rotate if( bIsRotateable && bClickedTwiceOnDragableObject && m_eDragMode==SdrDragMode::Move ) m_eDragMode=SdrDragMode::Rotate; else m_eDragMode=SdrDragMode::Move; pDrawViewWrapper->SetDragMode(m_eDragMode); if( !m_bWaitingForDoubleClick && m_aSelection.maybeSwitchSelectionAfterSingleClickWasEnsured() ) { impl_selectObjectAndNotiy(); } } else m_aSelection.resetPossibleSelectionAfterSingleClickWasEnsured(); } //@todo ForcePointer(&rMEvt); pChartWindow->ReleaseMouse(); // In tiled rendering drag mode could be not yet over on the call // that should handle the double-click, so better to perform this check // always. if( isDoubleClick(rMEvt) && !bMouseUpWithoutMouseDown /*#i106966#*/ ) { Point aMousePixel = rMEvt.GetPosPixel(); execute_DoubleClick( &aMousePixel ); } if( m_aSelection.isSelectionDifferentFromBeforeMouseDown() ) bNotifySelectionChange = true; } impl_SetMousePointer( rMEvt ); if(bNotifySelectionChange) impl_notifySelectionChangeListeners(); } void ChartController::execute_DoubleClick( const Point* pMousePixel ) { const SfxViewShell* pViewShell = SfxViewShell::Current(); bool isMobilePhone = pViewShell && pViewShell->isLOKMobilePhone(); if (isMobilePhone) return; bool bEditText = false; if ( m_aSelection.hasSelection() ) { OUString aCID( m_aSelection.getSelectedCID() ); if ( !aCID.isEmpty() ) { ObjectType eObjectType = ObjectIdentifier::getObjectType( aCID ); if ( eObjectType == OBJECTTYPE_TITLE ) { bEditText = true; } } else { // #i12587# support for shapes in chart SdrObject* pObj = DrawViewWrapper::getSdrObject( m_aSelection.getSelectedAdditionalShape() ); if ( DynCastSdrTextObj(pObj) != nullptr ) { bEditText = true; } } } if ( bEditText ) { executeDispatch_EditText( pMousePixel ); } else { executeDispatch_ObjectProperties(); } } void ChartController::execute_Resize() { SolarMutexGuard aGuard; auto pChartWindow(GetChartWindow()); if(pChartWindow) pChartWindow->Invalidate(); } void ChartController::execute_Command( const CommandEvent& rCEvt ) { SolarMutexGuard aGuard; auto pChartWindow(GetChartWindow()); bool bIsAction = false; { DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); if(!pChartWindow || !pDrawViewWrapper) return; bIsAction = m_pDrawViewWrapper->IsAction(); } // pop-up menu if(rCEvt.GetCommand() == CommandEventId::ContextMenu && !bIsAction) { { if(pChartWindow) pChartWindow->ReleaseMouse(); } if( m_aSelection.isSelectionDifferentFromBeforeMouseDown() ) impl_notifySelectionChangeListeners(); rtl::Reference< VCLXPopupMenu > xPopupMenu = new VCLXPopupMenu(); Point aPos( rCEvt.GetMousePosPixel() ); if( !rCEvt.IsMouseEvent() ) { if(pChartWindow) aPos = pChartWindow->GetPointerState().maPos; } OUString aMenuName; if ( isShapeContext() ) // #i12587# support for shapes in chart aMenuName = m_pDrawViewWrapper->IsTextEdit() ? std::u16string_view( u"drawtext" ) : std::u16string_view( u"draw" ); else { ObjectType eObjectType = ObjectIdentifier::getObjectType( m_aSelection.getSelectedCID() ); // todo: the context menu should be specified by an xml file in uiconfig sal_Int16 nUniqueId = 1; if (eObjectType != OBJECTTYPE_DATA_TABLE) { lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:Cut" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:Copy" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:Paste" ); xPopupMenu->insertSeparator( -1 ); } rtl::Reference< Diagram > xDiagram = getFirstDiagram(); OUString aFormatCommand( lcl_getFormatCommandForObjectCID( m_aSelection.getSelectedCID() ) ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, aFormatCommand ); //some commands for dataseries and points: if( eObjectType == OBJECTTYPE_DATA_SERIES || eObjectType == OBJECTTYPE_DATA_POINT ) { bool bIsPoint = ( eObjectType == OBJECTTYPE_DATA_POINT ); rtl::Reference< DataSeries > xSeries = ObjectIdentifier::getDataSeriesForCID( m_aSelection.getSelectedCID(), getChartModel() ); rtl::Reference< RegressionCurveModel > xTrendline = RegressionCurveHelper::getFirstCurveNotMeanValueLine( xSeries ); bool bHasEquation = RegressionCurveHelper::hasEquation( xTrendline ); rtl::Reference< RegressionCurveModel > xMeanValue = RegressionCurveHelper::getMeanValueLine( xSeries ); bool bHasYErrorBars = StatisticsHelper::hasErrorBars( xSeries ); bool bHasXErrorBars = StatisticsHelper::hasErrorBars( xSeries, false ); bool bHasDataLabelsAtSeries = DataSeriesHelper::hasDataLabelsAtSeries( xSeries ); bool bHasDataLabelsAtPoints = DataSeriesHelper::hasDataLabelsAtPoints( xSeries ); bool bHasDataLabelAtPoint = false; sal_Int32 nPointIndex = -1; if( bIsPoint ) { nPointIndex = ObjectIdentifier::getIndexFromParticleOrCID( m_aSelection.getSelectedCID() ); bHasDataLabelAtPoint = DataSeriesHelper::hasDataLabelAtPoint( xSeries, nPointIndex ); } bool bSelectedPointIsFormatted = false; bool bHasFormattedDataPointsOtherThanSelected = false; if( xSeries.is() ) { uno::Sequence< sal_Int32 > aAttributedDataPointIndexList; // "AttributedDataPoints" if( xSeries->getFastPropertyValue( PROP_DATASERIES_ATTRIBUTED_DATA_POINTS ) >>= aAttributedDataPointIndexList ) { if( aAttributedDataPointIndexList.hasElements() ) { if( bIsPoint ) { auto aIt = std::find( std::as_const(aAttributedDataPointIndexList).begin(), std::as_const(aAttributedDataPointIndexList).end(), nPointIndex ); if( aIt != std::as_const(aAttributedDataPointIndexList).end()) bSelectedPointIsFormatted = true; else bHasFormattedDataPointsOtherThanSelected = true; } else bHasFormattedDataPointsOtherThanSelected = true; } } } if( bIsPoint ) { if( bHasDataLabelAtPoint ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatDataLabel" ); if( !bHasDataLabelAtPoint ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertDataLabel" ); else lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteDataLabel" ); if( bSelectedPointIsFormatted ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:ResetDataPoint" ); xPopupMenu->insertSeparator( -1 ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatDataSeries" ); } rtl::Reference< ChartType > xChartType( xDiagram->getChartTypeOfSeries( xSeries ) ); if( xChartType->getChartType() == CHART2_SERVICE_NAME_CHARTTYPE_CANDLESTICK ) { try { bool bJapaneseStyle = false; xChartType->getPropertyValue( "Japanese" ) >>= bJapaneseStyle; if( bJapaneseStyle ) { lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatStockLoss" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatStockGain" ); } } catch( const uno::Exception & ) { DBG_UNHANDLED_EXCEPTION("chart2"); } } if( bHasDataLabelsAtSeries ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatDataLabels" ); if( bHasEquation ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatTrendlineEquation" ); if( xMeanValue.is() ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatMeanValue" ); if( bHasXErrorBars ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatXErrorBars" ); if( bHasYErrorBars ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatYErrorBars" ); xPopupMenu->insertSeparator( -1 ); if( !bHasDataLabelsAtSeries ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertDataLabels" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertTrendline" ); if( !xMeanValue.is() ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertMeanValue" ); if( !bHasXErrorBars ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertXErrorBars" ); if( !bHasYErrorBars ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertYErrorBars" ); if( bHasDataLabelsAtSeries || ( bHasDataLabelsAtPoints && bHasFormattedDataPointsOtherThanSelected ) ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteDataLabels" ); if( bHasEquation ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteTrendlineEquation" ); if( xMeanValue.is() ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteMeanValue" ); if( bHasXErrorBars ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteXErrorBars" ); if( bHasYErrorBars ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteYErrorBars" ); if( bHasFormattedDataPointsOtherThanSelected ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:ResetAllDataPoints" ); xPopupMenu->insertSeparator( -1 ); lcl_insertMenuCommand( xPopupMenu, nUniqueId, ".uno:ArrangeRow" ); rtl::Reference< VCLXPopupMenu > xArrangePopupMenu = new VCLXPopupMenu(); sal_Int16 nSubId = nUniqueId + 1; lcl_insertMenuCommand( xArrangePopupMenu, nSubId++, ".uno:Forward" ); lcl_insertMenuCommand( xArrangePopupMenu, nSubId, ".uno:Backward" ); xPopupMenu->setPopupMenu( nUniqueId, xArrangePopupMenu ); nUniqueId = nSubId; ++nUniqueId; } else if( eObjectType == OBJECTTYPE_DATA_CURVE ) { lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteTrendline" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatTrendlineEquation" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertTrendlineEquation" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertTrendlineEquationAndR2" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertR2Value" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteTrendlineEquation" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteR2Value" ); } else if( eObjectType == OBJECTTYPE_DATA_CURVE_EQUATION ) { lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertR2Value" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteR2Value" ); } //some commands for axes: and grids else if( eObjectType == OBJECTTYPE_AXIS || eObjectType == OBJECTTYPE_GRID || eObjectType == OBJECTTYPE_SUBGRID ) { rtl::Reference< Axis > xAxis = ObjectIdentifier::getAxisForCID( m_aSelection.getSelectedCID(), getChartModel() ); if( xAxis.is() && xDiagram.is() ) { sal_Int32 nDimensionIndex = -1; sal_Int32 nCooSysIndex = -1; sal_Int32 nAxisIndex = -1; AxisHelper::getIndicesForAxis( xAxis, xDiagram, nCooSysIndex, nDimensionIndex, nAxisIndex ); bool bIsSecondaryAxis = nAxisIndex!=0; bool bIsAxisVisible = AxisHelper::isAxisVisible( xAxis ); bool bIsMajorGridVisible = AxisHelper::isGridShown( nDimensionIndex, nCooSysIndex, true /*bMainGrid*/, xDiagram ); bool bIsMinorGridVisible = AxisHelper::isGridShown( nDimensionIndex, nCooSysIndex, false /*bMainGrid*/, xDiagram ); bool bHasTitle = !TitleHelper::getCompleteString( xAxis->getTitleObject2() ).isEmpty(); if( eObjectType != OBJECTTYPE_AXIS && bIsAxisVisible ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatAxis" ); if( eObjectType != OBJECTTYPE_GRID && bIsMajorGridVisible && !bIsSecondaryAxis ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatMajorGrid" ); if( eObjectType != OBJECTTYPE_SUBGRID && bIsMinorGridVisible && !bIsSecondaryAxis ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatMinorGrid" ); xPopupMenu->insertSeparator( -1 ); if( eObjectType != OBJECTTYPE_AXIS && !bIsAxisVisible ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertAxis" ); if( eObjectType != OBJECTTYPE_GRID && !bIsMajorGridVisible && !bIsSecondaryAxis ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertMajorGrid" ); if( eObjectType != OBJECTTYPE_SUBGRID && !bIsMinorGridVisible && !bIsSecondaryAxis ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertMinorGrid" ); if( !bHasTitle ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertAxisTitle" ); if( bIsAxisVisible ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteAxis" ); if( bIsMajorGridVisible && !bIsSecondaryAxis ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteMajorGrid" ); if( bIsMinorGridVisible && !bIsSecondaryAxis ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteMinorGrid" ); if (bIsAxisVisible) lcl_insertMenuCommand(xPopupMenu, nUniqueId++, ".uno:InsertDataTable"); } } else if (eObjectType == OBJECTTYPE_DATA_TABLE) { lcl_insertMenuCommand(xPopupMenu, nUniqueId++, ".uno:DeleteDataTable"); } if( eObjectType == OBJECTTYPE_DATA_STOCK_LOSS ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatStockGain" ); else if( eObjectType == OBJECTTYPE_DATA_STOCK_GAIN ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:FormatStockLoss" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:TransformDialog" ); if( eObjectType == OBJECTTYPE_PAGE || eObjectType == OBJECTTYPE_DIAGRAM || eObjectType == OBJECTTYPE_DIAGRAM_WALL || eObjectType == OBJECTTYPE_DIAGRAM_FLOOR || eObjectType == OBJECTTYPE_UNKNOWN ) { if( eObjectType != OBJECTTYPE_UNKNOWN ) xPopupMenu->insertSeparator( -1 ); bool bHasLegend = LegendHelper::hasLegend( xDiagram ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertTitles" ); if( !bHasLegend ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertLegend" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:InsertRemoveAxes" ); if( bHasLegend ) lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DeleteLegend" ); } xPopupMenu->insertSeparator( -1 ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DiagramType" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DataRanges" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:DiagramData" ); lcl_insertMenuCommand( xPopupMenu, nUniqueId++, ".uno:View3D" ); } css::uno::Sequence< css::uno::Any > aArgs{ css::uno::Any(comphelper::makePropertyValue( "IsContextMenu", true )), css::uno::Any(comphelper::makePropertyValue( "Frame", m_xFrame )), css::uno::Any(comphelper::makePropertyValue( "Value", aMenuName )) }; css::uno::Reference< css::frame::XPopupMenuController > xPopupController( m_xCC->getServiceManager()->createInstanceWithArgumentsAndContext( "com.sun.star.comp.framework.ResourceMenuController", aArgs, m_xCC ), css::uno::UNO_QUERY ); if ( !xPopupController.is() || !xPopupMenu.is() ) return; xPopupController->setPopupMenu( xPopupMenu ); if (comphelper::LibreOfficeKit::isActive()) { if (SfxViewShell* pViewShell = SfxViewShell::Current()) { ControllerCommandDispatch* pCommandDispatch = dynamic_cast(m_aDispatchContainer.getChartDispatcher().get()); if (pCommandDispatch) { for (int nPos = 0, nCount = xPopupMenu->getItemCount(); nPos < nCount; ++nPos) { auto nItemId = xPopupMenu->getItemId(nPos); OUString aCommandURL = xPopupMenu->getCommand(nItemId); if (!pCommandDispatch->commandAvailable(aCommandURL)) xPopupMenu->enableItem(nItemId, false); } } boost::property_tree::ptree aMenu = SfxDispatcher::fillPopupMenu(xPopupMenu); boost::property_tree::ptree aRoot; aRoot.add_child("menu", aMenu); std::stringstream aStream; boost::property_tree::write_json(aStream, aRoot, true); pViewShell->libreOfficeKitViewCallback(LOK_CALLBACK_CONTEXT_MENU, OString(aStream.str())); } } else { xPopupMenu->execute( css::uno::Reference< css::awt::XWindowPeer >( m_xFrame->getContainerWindow(), css::uno::UNO_QUERY ), css::awt::Rectangle( aPos.X(), aPos.Y(), 0, 0 ), css::awt::PopupMenuDirection::EXECUTE_DEFAULT ); } css::uno::Reference< css::lang::XComponent > xComponent( xPopupController, css::uno::UNO_QUERY ); if ( xComponent.is() ) xComponent->dispose(); } else if( ( rCEvt.GetCommand() == CommandEventId::StartExtTextInput ) || ( rCEvt.GetCommand() == CommandEventId::ExtTextInput ) || ( rCEvt.GetCommand() == CommandEventId::EndExtTextInput ) || ( rCEvt.GetCommand() == CommandEventId::InputContextChange ) ) { //#i84417# enable editing with IME m_pDrawViewWrapper->Command( rCEvt, pChartWindow ); } } bool ChartController::execute_KeyInput( const KeyEvent& rKEvt ) { SolarMutexGuard aGuard; bool bReturn=false; DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); auto pChartWindow(GetChartWindow()); if (!pChartWindow || !pDrawViewWrapper) return bReturn; // handle accelerators if (!m_apAccelExecute && m_xFrame.is() && m_xCC.is()) { m_apAccelExecute = ::svt::AcceleratorExecute::createAcceleratorHelper(); OSL_ASSERT(m_apAccelExecute); if (m_apAccelExecute) m_apAccelExecute->init( m_xCC, m_xFrame ); } vcl::KeyCode aKeyCode( rKEvt.GetKeyCode()); sal_uInt16 nCode = aKeyCode.GetCode(); bool bAlternate = aKeyCode.IsMod2(); bool bCtrl = aKeyCode.IsMod1(); if (m_apAccelExecute) bReturn = m_apAccelExecute->execute( aKeyCode ); if( bReturn ) return bReturn; { if( pDrawViewWrapper->IsTextEdit() ) { if( pDrawViewWrapper->KeyInput(rKEvt, pChartWindow) ) { bReturn = true; if( nCode == KEY_ESCAPE ) { EndTextEdit(); } } } } // keyboard accessibility ObjectType eObjectType = ObjectIdentifier::getObjectType( m_aSelection.getSelectedCID() ); if( ! bReturn ) { // Navigation (Tab/F3/Home/End) rtl::Reference<::chart::ChartModel> xChartDoc( getChartModel() ); ObjectKeyNavigation aObjNav( m_aSelection.getSelectedOID(), xChartDoc, m_xChartView.get() ); awt::KeyEvent aKeyEvent( ::svt::AcceleratorExecute::st_VCLKey2AWTKey( aKeyCode )); bReturn = aObjNav.handleKeyEvent( aKeyEvent ); if( bReturn ) { const ObjectIdentifier& aNewOID = aObjNav.getCurrentSelection(); uno::Any aNewSelection; if ( aNewOID.isValid() && !ObjectHierarchy::isRootNode( aNewOID ) ) { aNewSelection = aNewOID.getAny(); } if ( m_eDragMode == SdrDragMode::Rotate && !SelectionHelper::isRotateableObject( aNewOID.getObjectCID(), getChartModel() ) ) { m_eDragMode = SdrDragMode::Move; } bReturn = select( aNewSelection ); } } // Position and Size (+/-/arrow-keys) or pie segment dragging if( ! bReturn ) { // pie segment dragging // note: could also be done for data series if( eObjectType == OBJECTTYPE_DATA_POINT && ObjectIdentifier::getDragMethodServiceName( m_aSelection.getSelectedCID() ) == ObjectIdentifier::getPieSegmentDragMethodServiceName()) { bool bDrag = false; bool bDragInside = false; if( nCode == KEY_ADD || nCode == KEY_SUBTRACT ) { bDrag = true; bDragInside = ( nCode == KEY_SUBTRACT ); } else if( nCode == KEY_LEFT || nCode == KEY_RIGHT || nCode == KEY_UP || nCode == KEY_DOWN ) { bDrag = true; std::u16string_view aParameter( ObjectIdentifier::getDragParameterString( m_aSelection.getSelectedCID() )); sal_Int32 nOffsetPercentDummy( 0 ); awt::Point aMinimumPosition( 0, 0 ); awt::Point aMaximumPosition( 0, 0 ); ObjectIdentifier::parsePieSegmentDragParameterString( aParameter, nOffsetPercentDummy, aMinimumPosition, aMaximumPosition ); aMaximumPosition.Y -= aMinimumPosition.Y; aMaximumPosition.X -= aMinimumPosition.X; bDragInside = (nCode == KEY_RIGHT && (aMaximumPosition.X < 0)) || (nCode == KEY_LEFT && (aMaximumPosition.X > 0)) || (nCode == KEY_DOWN && (aMaximumPosition.Y < 0)) || (nCode == KEY_UP && (aMaximumPosition.Y > 0)); } if( bDrag ) { double fAmount = bAlternate ? 0.01 : 0.05; if( bDragInside ) fAmount *= -1.0; bReturn = impl_DragDataPoint( m_aSelection.getSelectedCID(), fAmount ); } } else { // size if( nCode == KEY_ADD || nCode == KEY_SUBTRACT ) { if( eObjectType == OBJECTTYPE_DIAGRAM ) { // default 1 mm in each direction double fGrowAmountX = 200.0; double fGrowAmountY = 200.0; if (bAlternate) { // together with Alt-key: 1 px in each direction Size aPixelSize = pChartWindow->PixelToLogic( Size( 2, 2 )); fGrowAmountX = static_cast< double >( aPixelSize.Width()); fGrowAmountY = static_cast< double >( aPixelSize.Height()); } if( nCode == KEY_SUBTRACT ) { fGrowAmountX = -fGrowAmountX; fGrowAmountY = -fGrowAmountY; } bReturn = impl_moveOrResizeObject( m_aSelection.getSelectedCID(), CENTERED_RESIZE_OBJECT, fGrowAmountX, fGrowAmountY ); } } // position else if( nCode == KEY_LEFT || nCode == KEY_RIGHT || nCode == KEY_UP || nCode == KEY_DOWN ) { if( m_aSelection.isDragableObjectSelected() ) { // default 1 mm double fShiftAmountX = 100.0; double fShiftAmountY = 100.0; if (bAlternate) { // together with Alt-key: 1 px Size aPixelSize = pChartWindow->PixelToLogic( Size( 1, 1 )); fShiftAmountX = static_cast< double >( aPixelSize.Width()); fShiftAmountY = static_cast< double >( aPixelSize.Height()); } switch( nCode ) { case KEY_LEFT: fShiftAmountX = -fShiftAmountX; fShiftAmountY = 0.0; break; case KEY_RIGHT: fShiftAmountY = 0.0; break; case KEY_UP: fShiftAmountX = 0.0; fShiftAmountY = -fShiftAmountY; break; case KEY_DOWN: fShiftAmountX = 0.0; break; } if( !m_aSelection.getSelectedCID().isEmpty() ) { //move chart objects if (eObjectType == OBJECTTYPE_DATA_LABEL) { SdrObject* pObj = pDrawViewWrapper->getSelectedObject(); if (pObj) { tools::Rectangle aRect = pObj->GetSnapRect(); awt::Size aPageSize(ChartModelHelper::getPageSize(getChartModel())); if ((fShiftAmountX > 0.0 && (aRect.Right() + fShiftAmountX > aPageSize.Width)) || (fShiftAmountX < 0.0 && (aRect.Left() + fShiftAmountX < 0)) || (fShiftAmountY > 0.0 && (aRect.Bottom() + fShiftAmountY > aPageSize.Height)) || (fShiftAmountY < 0.0 && (aRect.Top() + fShiftAmountY < 0))) bReturn = false; else bReturn = PositionAndSizeHelper::moveObject( m_aSelection.getSelectedCID(), getChartModel(), awt::Rectangle(aRect.Left() + fShiftAmountX, aRect.Top() + fShiftAmountY, aRect.getOpenWidth(), aRect.getOpenHeight()), awt::Rectangle(aRect.Left(), aRect.Top(), 0, 0), awt::Rectangle(0, 0, aPageSize.Width, aPageSize.Height)); } } else bReturn = impl_moveOrResizeObject( m_aSelection.getSelectedCID(), MOVE_OBJECT, fShiftAmountX, fShiftAmountY ); } else { //move additional shapes uno::Reference< drawing::XShape > xShape( m_aSelection.getSelectedAdditionalShape() ); if( xShape.is() ) { awt::Point aPos( xShape->getPosition() ); awt::Size aSize( xShape->getSize() ); awt::Size aPageSize( ChartModelHelper::getPageSize( getChartModel() ) ); aPos.X = static_cast< tools::Long >( static_cast< double >( aPos.X ) + fShiftAmountX ); aPos.Y = static_cast< tools::Long >( static_cast< double >( aPos.Y ) + fShiftAmountY ); if( aPos.X + aSize.Width > aPageSize.Width ) aPos.X = aPageSize.Width - aSize.Width; if( aPos.X < 0 ) aPos.X = 0; if( aPos.Y + aSize.Height > aPageSize.Height ) aPos.Y = aPageSize.Height - aSize.Height; if( aPos.Y < 0 ) aPos.Y = 0; xShape->setPosition( aPos ); } } } } } } // dumping the shape if( !bReturn && bCtrl && nCode == KEY_F12) { rtl::Reference< ChartModel > xChartModel = getChartModel(); if(xChartModel.is()) { OUString aDump = xChartModel->dump("shapes"); SAL_WARN("chart2", aDump); } } // text edit if( ! bReturn && nCode == KEY_F2 ) { if( eObjectType == OBJECTTYPE_TITLE ) { executeDispatch_EditText(); bReturn = true; } } // deactivate inplace mode (this code should be unnecessary, but // unfortunately is not) if( ! bReturn && nCode == KEY_ESCAPE ) { uno::Reference< frame::XDispatchHelper > xDispatchHelper( frame::DispatchHelper::create(m_xCC) ); uno::Sequence< beans::PropertyValue > aArgs; xDispatchHelper->executeDispatch( uno::Reference< frame::XDispatchProvider >( m_xFrame, uno::UNO_QUERY ), ".uno:TerminateInplaceActivation", "_parent", frame::FrameSearchFlag::PARENT, aArgs ); bReturn = true; } if( ! bReturn && (nCode == KEY_DELETE || nCode == KEY_BACKSPACE )) { bReturn = executeDispatch_Delete(); if( ! bReturn ) { std::unique_ptr xInfoBox(Application::CreateMessageDialog(pChartWindow->GetFrameWeld(), VclMessageType::Info, VclButtonsType::Ok, SchResId(STR_ACTION_NOTPOSSIBLE))); xInfoBox->run(); } } return bReturn; } bool ChartController::requestQuickHelp( ::Point aAtLogicPosition, bool bIsBalloonHelp, OUString & rOutQuickHelpText, awt::Rectangle & rOutEqualRect ) { rtl::Reference<::chart::ChartModel> xChartModel; if( m_aModel.is()) xChartModel = getChartModel(); if( !xChartModel.is()) return false; // help text OUString aCID; if( m_pDrawViewWrapper ) { aCID = SelectionHelper::getHitObjectCID( aAtLogicPosition, *m_pDrawViewWrapper ); } bool bResult( !aCID.isEmpty()); if( bResult ) { // get help text rOutQuickHelpText = ObjectNameProvider::getHelpText( aCID, xChartModel, bIsBalloonHelp /* bVerbose */ ); // set rectangle if( m_xChartView ) rOutEqualRect = m_xChartView->getRectangleOfObject( aCID, true ); } return bResult; } // XSelectionSupplier (optional interface) sal_Bool SAL_CALL ChartController::select( const uno::Any& rSelection ) { bool bSuccess = false; if ( rSelection.hasValue() ) { const uno::Type& rType = rSelection.getValueType(); if ( rType == cppu::UnoType< OUString >::get() ) { OUString aNewCID; if ( ( rSelection >>= aNewCID ) && m_aSelection.setSelection( aNewCID ) ) { bSuccess = true; } } else if ( rType == cppu::UnoType::get() ) { uno::Reference< drawing::XShape > xShape; if ( ( rSelection >>= xShape ) && m_aSelection.setSelection( xShape ) ) { bSuccess = true; } } } else { if ( m_aSelection.hasSelection() ) { m_aSelection.clearSelection(); bSuccess = true; } } if ( bSuccess ) { SolarMutexGuard aGuard; if ( m_pDrawViewWrapper && m_pDrawViewWrapper->IsTextEdit() ) { EndTextEdit(); } impl_selectObjectAndNotiy(); auto pChartWindow(GetChartWindow()); if ( pChartWindow ) { pChartWindow->Invalidate(); } return true; } return false; } uno::Any SAL_CALL ChartController::getSelection() { uno::Any aReturn; if ( m_aSelection.hasSelection() ) { OUString aCID( m_aSelection.getSelectedCID() ); if ( !aCID.isEmpty() ) { aReturn <<= aCID; } else { // #i12587# support for shapes in chart aReturn <<= m_aSelection.getSelectedAdditionalShape(); } } return aReturn; } void SAL_CALL ChartController::addSelectionChangeListener( const uno::Reference & xListener ) { SolarMutexGuard aGuard; if( impl_isDisposedOrSuspended() )//@todo? allow adding of listeners in suspend mode? return; //behave passive if already disposed or suspended //--add listener std::unique_lock aGuard2(m_aLifeTimeManager.m_aAccessMutex); m_aLifeTimeManager.m_aSelectionChangeListeners.addInterface( aGuard2, xListener ); } void SAL_CALL ChartController::removeSelectionChangeListener( const uno::Reference & xListener ) { SolarMutexGuard aGuard; if( impl_isDisposedOrSuspended() ) //@todo? allow removing of listeners in suspend mode? return; //behave passive if already disposed or suspended //--remove listener std::unique_lock aGuard2(m_aLifeTimeManager.m_aAccessMutex); m_aLifeTimeManager.m_aSelectionChangeListeners.removeInterface( aGuard2, xListener ); } void ChartController::impl_notifySelectionChangeListeners() { std::unique_lock aGuard(m_aLifeTimeManager.m_aAccessMutex); if( m_aLifeTimeManager.m_aSelectionChangeListeners.getLength(aGuard) ) { uno::Reference< view::XSelectionSupplier > xSelectionSupplier(this); lang::EventObject aEvent( xSelectionSupplier ); m_aLifeTimeManager.m_aSelectionChangeListeners.notifyEach(aGuard, &view::XSelectionChangeListener::selectionChanged, aEvent); } } void ChartController::impl_selectObjectAndNotiy() { { SolarMutexGuard aGuard; DrawViewWrapper* pDrawViewWrapper = m_pDrawViewWrapper.get(); if( pDrawViewWrapper ) { pDrawViewWrapper->SetDragMode( m_eDragMode ); m_aSelection.applySelection( m_pDrawViewWrapper.get() ); } } impl_notifySelectionChangeListeners(); } bool ChartController::impl_moveOrResizeObject( const OUString & rCID, eMoveOrResizeType eType, double fAmountLogicX, double fAmountLogicY ) { bool bResult = false; bool bNeedResize = ( eType == CENTERED_RESIZE_OBJECT ); rtl::Reference<::chart::ChartModel> xChartModel( getChartModel() ); uno::Reference< beans::XPropertySet > xObjProp( ObjectIdentifier::getObjectPropertySet( rCID, xChartModel )); if( xObjProp.is()) { awt::Size aRefSize = ChartModelHelper::getPageSize( xChartModel ); chart2::RelativePosition aRelPos; chart2::RelativeSize aRelSize; bool bDeterminePos = !(xObjProp->getPropertyValue( "RelativePosition") >>= aRelPos); bool bDetermineSize = !bNeedResize || !(xObjProp->getPropertyValue( "RelativeSize") >>= aRelSize); if( ( bDeterminePos || bDetermineSize ) && ( aRefSize.Width > 0 && aRefSize.Height > 0 ) ) { ExplicitValueProvider * pValueProvider( m_xChartView.get() ); if( pValueProvider ) { awt::Rectangle aRect( pValueProvider->getRectangleOfObject( rCID )); double fWidth = static_cast< double >( aRefSize.Width ); double fHeight = static_cast< double >( aRefSize.Height ); if( bDetermineSize ) { aRelSize.Primary = static_cast< double >( aRect.Width ) / fWidth; aRelSize.Secondary = static_cast< double >( aRect.Height ) / fHeight; } if( bDeterminePos ) { if( bNeedResize && aRelSize.Primary > 0.0 && aRelSize.Secondary > 0.0 ) { aRelPos.Primary = (static_cast< double >( aRect.X ) / fWidth) + (aRelSize.Primary / 2.0); aRelPos.Secondary = (static_cast< double >( aRect.Y ) / fHeight) + (aRelSize.Secondary / 2.0); aRelPos.Anchor = drawing::Alignment_CENTER; } else { aRelPos.Primary = static_cast< double >( aRect.X ) / fWidth; aRelPos.Secondary = static_cast< double >( aRect.Y ) / fHeight; aRelPos.Anchor = drawing::Alignment_TOP_LEFT; } } } } if( eType == CENTERED_RESIZE_OBJECT ) bResult = lcl_GrowAndShiftLogic( aRelPos, aRelSize, aRefSize, fAmountLogicX, fAmountLogicY ); else if( eType == MOVE_OBJECT ) bResult = lcl_MoveObjectLogic( aRelPos, aRelSize, aRefSize, fAmountLogicX, fAmountLogicY ); if( bResult ) { ActionDescriptionProvider::ActionType eActionType(ActionDescriptionProvider::ActionType::Move); if( bNeedResize ) eActionType = ActionDescriptionProvider::ActionType::Resize; ObjectType eObjectType = ObjectIdentifier::getObjectType( rCID ); UndoGuard aUndoGuard( ActionDescriptionProvider::createDescription( eActionType, ObjectNameProvider::getName( eObjectType )), m_xUndoManager ); { ControllerLockGuardUNO aCLGuard( xChartModel ); xObjProp->setPropertyValue( "RelativePosition", uno::Any( aRelPos )); if( bNeedResize || (eObjectType == OBJECTTYPE_DIAGRAM) )//Also set an explicit size at the diagram when an explicit position is set xObjProp->setPropertyValue( "RelativeSize", uno::Any( aRelSize )); } aUndoGuard.commit(); } } return bResult; } bool ChartController::impl_DragDataPoint( std::u16string_view rCID, double fAdditionalOffset ) { bool bResult = false; if( fAdditionalOffset < -1.0 || fAdditionalOffset > 1.0 || fAdditionalOffset == 0.0 ) return bResult; sal_Int32 nDataPointIndex = ObjectIdentifier::getIndexFromParticleOrCID( rCID ); rtl::Reference< DataSeries > xSeries = ObjectIdentifier::getDataSeriesForCID( rCID, getChartModel() ); if( xSeries.is()) { try { uno::Reference< beans::XPropertySet > xPointProp( xSeries->getDataPointByIndex( nDataPointIndex )); double fOffset = 0.0; if( xPointProp.is() && (xPointProp->getPropertyValue( "Offset" ) >>= fOffset ) && (( fAdditionalOffset > 0.0 && fOffset < 1.0 ) || (fOffset > 0.0)) ) { fOffset += fAdditionalOffset; if( fOffset > 1.0 ) fOffset = 1.0; else if( fOffset < 0.0 ) fOffset = 0.0; xPointProp->setPropertyValue( "Offset", uno::Any( fOffset )); bResult = true; } } catch( const uno::Exception & ) { DBG_UNHANDLED_EXCEPTION("chart2"); } } return bResult; } void ChartController::impl_SetMousePointer( const MouseEvent & rEvent ) { SolarMutexGuard aGuard; auto pChartWindow(GetChartWindow()); if (!m_pDrawViewWrapper || !pChartWindow) return; Point aMousePos( pChartWindow->PixelToLogic( rEvent.GetPosPixel())); sal_uInt16 nModifier = rEvent.GetModifier(); bool bLeftDown = rEvent.IsLeft(); // Check if object is for field button and set the normal arrow pointer in this case SdrObject* pObject = m_pDrawViewWrapper->getHitObject(aMousePos); if (pObject && pObject->GetName().startsWith("FieldButton")) { pChartWindow->SetPointer(PointerStyle::Arrow); return; } if ( m_pDrawViewWrapper->IsTextEdit() ) { if( m_pDrawViewWrapper->IsTextEditHit( aMousePos ) ) { pChartWindow->SetPointer( m_pDrawViewWrapper->GetPreferredPointer( aMousePos, pChartWindow->GetOutDev(), nModifier, bLeftDown ) ); return; } } else if( m_pDrawViewWrapper->IsAction() ) { return;//don't change pointer during running action } SdrHdl* pHitSelectionHdl = nullptr; if( m_aSelection.isResizeableObjectSelected() ) pHitSelectionHdl = m_pDrawViewWrapper->PickHandle( aMousePos ); if( pHitSelectionHdl ) { PointerStyle aPointer = m_pDrawViewWrapper->GetPreferredPointer( aMousePos, pChartWindow->GetOutDev(), nModifier, bLeftDown ); bool bForceArrowPointer = false; ObjectIdentifier aOID( m_aSelection.getSelectedOID() ); switch( aPointer) { case PointerStyle::NSize: case PointerStyle::SSize: case PointerStyle::WSize: case PointerStyle::ESize: case PointerStyle::NWSize: case PointerStyle::NESize: case PointerStyle::SWSize: case PointerStyle::SESize: if( ! m_aSelection.isResizeableObjectSelected() ) bForceArrowPointer = true; break; case PointerStyle::Move: if ( !aOID.isDragableObject() ) bForceArrowPointer = true; break; case PointerStyle::MovePoint: case PointerStyle::MoveBezierWeight: // there is no point-editing in a chart // the PointerStyle::MoveBezierWeight appears in 3d data points bForceArrowPointer = true; break; default: break; } if( bForceArrowPointer ) pChartWindow->SetPointer( PointerStyle::Arrow ); else pChartWindow->SetPointer( aPointer ); return; } // #i12587# support for shapes in chart if ( m_eDrawMode == CHARTDRAW_INSERT && ( !m_pDrawViewWrapper->IsMarkedHit( aMousePos ) || !m_aSelection.isDragableObjectSelected() ) ) { PointerStyle ePointerStyle = PointerStyle::DrawRect; SdrObjKind eKind = m_pDrawViewWrapper->GetCurrentObjIdentifier(); switch ( eKind ) { case SdrObjKind::Line: { ePointerStyle = PointerStyle::DrawLine; } break; case SdrObjKind::Rectangle: case SdrObjKind::CustomShape: { ePointerStyle = PointerStyle::DrawRect; } break; case SdrObjKind::CircleOrEllipse: { ePointerStyle = PointerStyle::DrawEllipse; } break; case SdrObjKind::FreehandLine: { ePointerStyle = PointerStyle::DrawPolygon; } break; case SdrObjKind::Text: { ePointerStyle = PointerStyle::DrawText; } break; case SdrObjKind::Caption: { ePointerStyle = PointerStyle::DrawCaption; } break; default: { ePointerStyle = PointerStyle::DrawRect; } break; } pChartWindow->SetPointer( ePointerStyle ); return; } OUString aHitObjectCID( SelectionHelper::getHitObjectCID( aMousePos, *m_pDrawViewWrapper, true /*bGetDiagramInsteadOf_Wall*/ )); if( m_pDrawViewWrapper->IsTextEdit() ) { if( aHitObjectCID == m_aSelection.getSelectedCID() ) { pChartWindow->SetPointer( PointerStyle::Arrow ); return; } } if( aHitObjectCID.isEmpty() ) { //additional shape was hit pChartWindow->SetPointer( PointerStyle::Move ); } else if( ObjectIdentifier::isDragableObject( aHitObjectCID ) ) { if( (m_eDragMode == SdrDragMode::Rotate) && SelectionHelper::isRotateableObject( aHitObjectCID , getChartModel() ) ) pChartWindow->SetPointer( PointerStyle::Rotate ); else { ObjectType eHitObjectType = ObjectIdentifier::getObjectType( aHitObjectCID ); if( eHitObjectType == OBJECTTYPE_DATA_POINT ) { if( !ObjectIdentifier::areSiblings(aHitObjectCID,m_aSelection.getSelectedCID()) && !ObjectIdentifier::areIdenticalObjects(aHitObjectCID,m_aSelection.getSelectedCID()) ) { pChartWindow->SetPointer( PointerStyle::Arrow ); return; } } pChartWindow->SetPointer( PointerStyle::Move ); } } else pChartWindow->SetPointer( PointerStyle::Arrow ); } void ChartController::sendPopupRequest(std::u16string_view rCID, tools::Rectangle aRectangle) { ChartModel* pChartModel = m_aModel->getModel().get(); if (!pChartModel) return; uno::Reference xPivotTableDataProvider; xPivotTableDataProvider.set(pChartModel->getDataProvider(), uno::UNO_QUERY); if (!xPivotTableDataProvider.is()) return; OUString sPivotTableName = xPivotTableDataProvider->getPivotTableName(); css::uno::Reference xPopupRequest = pChartModel->getPopupRequest(); PopupRequest* pPopupRequest = dynamic_cast(xPopupRequest.get()); if (!pPopupRequest) return; // Get dimension index from CID size_t nStartPos = rCID.rfind('.'); nStartPos++; sal_Int32 nEndPos = rCID.size(); std::u16string_view sDimensionIndex = rCID.substr(nStartPos, nEndPos - nStartPos); sal_Int32 nDimensionIndex = o3tl::toInt32(sDimensionIndex); awt::Rectangle xRectangle { sal_Int32(aRectangle.Left()), sal_Int32(aRectangle.Top()), sal_Int32(aRectangle.GetWidth()), sal_Int32(aRectangle.GetHeight()) }; uno::Sequence aCallbackData = comphelper::InitPropertySequence( { {"Rectangle", uno::Any(xRectangle)}, {"DimensionIndex", uno::Any(sal_Int32(nDimensionIndex))}, {"PivotTableName", uno::Any(sPivotTableName)}, }); pPopupRequest->getCallback()->notify(uno::Any(aCallbackData)); } } //namespace chart /* vim:set shiftwidth=4 softtabstop=4 expandtab: */