/* -*- 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 "fuconbez.hxx" #include #include #include #include #include #include #include #include #include #include #include "app.hrc" #include "ViewShell.hxx" #include "ViewShellBase.hxx" #include "View.hxx" #include "Window.hxx" #include "ToolBarManager.hxx" #include "drawdoc.hxx" #include "res_bmp.hrc" #include #include #include "CustomAnimationEffect.hxx" using namespace ::com::sun::star::uno; namespace sd { FuConstructBezierPolygon::FuConstructBezierPolygon ( ViewShell* pViewSh, ::sd::Window* pWin, ::sd::View* pView, SdDrawDocument* pDoc, SfxRequest& rReq) : FuConstruct(pViewSh, pWin, pView, pDoc, rReq), nEditMode(SID_BEZIER_MOVE) { } rtl::Reference FuConstructBezierPolygon::Create( ViewShell* pViewSh, ::sd::Window* pWin, ::sd::View* pView, SdDrawDocument* pDoc, SfxRequest& rReq, bool bPermanent ) { FuConstructBezierPolygon* pFunc; rtl::Reference xFunc( pFunc = new FuConstructBezierPolygon( pViewSh, pWin, pView, pDoc, rReq ) ); xFunc->DoExecute(rReq); pFunc->SetPermanent(bPermanent); return xFunc; } void FuConstructBezierPolygon::DoExecute( SfxRequest& rReq ) { FuConstruct::DoExecute( rReq ); const SfxItemSet* pArgs = rReq.GetArgs(); if( pArgs ) { const SfxPoolItem* pPoolItem = nullptr; if( SfxItemState::SET == pArgs->GetItemState( SID_ADD_MOTION_PATH, true, &pPoolItem ) ) maTargets = static_cast( pPoolItem )->GetValue(); } } bool FuConstructBezierPolygon::MouseButtonDown(const MouseEvent& rMEvt) { bool bReturn = FuConstruct::MouseButtonDown(rMEvt); SdrViewEvent aVEvt; SdrHitKind eHit = mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); if (eHit == SDRHIT_HANDLE || rMEvt.IsMod1()) { mpView->SetEditMode(SDREDITMODE_EDIT); } else { mpView->SetEditMode(SDREDITMODE_CREATE); } if (aVEvt.eEvent == SDREVENT_BEGTEXTEDIT) { // here, we do not allow text input aVEvt.eEvent = SDREVENT_BEGDRAGOBJ; mpView->EnableExtendedMouseEventDispatcher(false); } else { mpView->EnableExtendedMouseEventDispatcher(true); } if (eHit == SDRHIT_MARKEDOBJECT && nEditMode == SID_BEZIER_INSERT) { // insert glue point mpView->BegInsObjPoint(aMDPos, rMEvt.IsMod1()); } else { mpView->MouseButtonDown(rMEvt, mpWindow); SdrObject* pObj = mpView->GetCreateObj(); if (pObj) { SfxItemSet aAttr(mpDoc->GetPool()); SetStyleSheet(aAttr, pObj); pObj->SetMergedItemSet(aAttr); } } return bReturn; } bool FuConstructBezierPolygon::MouseMove(const MouseEvent& rMEvt) { return FuConstruct::MouseMove(rMEvt); } bool FuConstructBezierPolygon::MouseButtonUp(const MouseEvent& rMEvt ) { bool bReturn = false; bool bCreated = false; SdrViewEvent aVEvt; mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONUP, aVEvt); const size_t nCount = mpView->GetSdrPageView()->GetObjList()->GetObjCount(); if (mpView->IsInsObjPoint()) { mpView->EndInsObjPoint(SDRCREATE_FORCEEND); } else { mpView->MouseButtonUp(rMEvt, mpWindow); } if (aVEvt.eEvent == SDREVENT_ENDCREATE) { bReturn = true; if (nCount+1 == mpView->GetSdrPageView()->GetObjList()->GetObjCount()) { bCreated = true; } // trick to suppress FuDraw::DoubleClick bMBDown = false; } bReturn = FuConstruct::MouseButtonUp(rMEvt) || bReturn; bool bDeleted = false; if( bCreated && maTargets.hasValue() ) { SdrPathObj* pPathObj = dynamic_cast< SdrPathObj* >( mpView->GetSdrPageView()->GetObjList()->GetObj( nCount ) ); SdPage* pPage = dynamic_cast< SdPage* >( pPathObj ? pPathObj->GetPage() : nullptr ); if( pPage ) { std::shared_ptr< sd::MainSequence > pMainSequence( pPage->getMainSequence() ); if( pMainSequence.get() ) { Sequence< Any > aTargets; maTargets >>= aTargets; sal_Int32 nTCount = aTargets.getLength(); if( nTCount > 1 ) { const Any* pTarget = aTargets.getConstArray(); double fDuration = 0.0; *pTarget++ >>= fDuration; bool bFirst = true; while( --nTCount ) { CustomAnimationEffectPtr pCreated( pMainSequence->append( *pPathObj, *pTarget++, fDuration ) ); if( bFirst ) bFirst = false; else pCreated->setNodeType( css::presentation::EffectNodeType::WITH_PREVIOUS ); } } } } mpView->DeleteMarked(); bDeleted = true; } if ((!bPermanent && bCreated) || bDeleted) { mpViewShell->GetViewFrame()->GetDispatcher()->Execute(SID_OBJECT_SELECT, SfxCallMode::ASYNCHRON); } return bReturn; } /** * Process keyboard input * @returns sal_True if a KeyEvent is being processed, sal_False otherwise */ bool FuConstructBezierPolygon::KeyInput(const KeyEvent& rKEvt) { return FuConstruct::KeyInput(rKEvt); } void FuConstructBezierPolygon::Activate() { mpView->EnableExtendedMouseEventDispatcher(true); SdrObjKind eKind; switch (nSlotId) { case SID_DRAW_POLYGON_NOFILL: case SID_DRAW_XPOLYGON_NOFILL: { eKind = OBJ_PLIN; } break; case SID_DRAW_POLYGON: case SID_DRAW_XPOLYGON: { eKind = OBJ_POLY; } break; case SID_DRAW_BEZIER_NOFILL: { eKind = OBJ_PATHLINE; } break; case SID_DRAW_BEZIER_FILL: { eKind = OBJ_PATHFILL; } break; case SID_DRAW_FREELINE_NOFILL: { eKind = OBJ_FREELINE; } break; case SID_DRAW_FREELINE: { eKind = OBJ_FREEFILL; } break; default: { eKind = OBJ_PATHLINE; } break; } mpView->SetCurrentObj((sal_uInt16)eKind); FuConstruct::Activate(); } void FuConstructBezierPolygon::Deactivate() { mpView->EnableExtendedMouseEventDispatcher(false); FuConstruct::Deactivate(); } void FuConstructBezierPolygon::SelectionHasChanged() { FuDraw::SelectionHasChanged(); mpViewShell->GetViewShellBase().GetToolBarManager()->SelectionHasChanged( *mpViewShell, *mpView); } /** * Set current bezier edit mode */ void FuConstructBezierPolygon::SetEditMode(sal_uInt16 nMode) { nEditMode = nMode; ForcePointer(); SfxBindings& rBindings = mpViewShell->GetViewFrame()->GetBindings(); rBindings.Invalidate(SID_BEZIER_MOVE); rBindings.Invalidate(SID_BEZIER_INSERT); } SdrObject* FuConstructBezierPolygon::CreateDefaultObject(const sal_uInt16 nID, const Rectangle& rRectangle) { // case SID_DRAW_POLYGON: // case SID_DRAW_POLYGON_NOFILL: // case SID_DRAW_XPOLYGON: // case SID_DRAW_XPOLYGON_NOFILL: // case SID_DRAW_FREELINE: // case SID_DRAW_FREELINE_NOFILL: // case SID_DRAW_BEZIER_FILL: // BASIC // case SID_DRAW_BEZIER_NOFILL: // BASIC SdrObject* pObj = SdrObjFactory::MakeNewObject( mpView->GetCurrentObjInventor(), mpView->GetCurrentObjIdentifier(), nullptr, mpDoc); if(pObj) { if( dynamic_cast< const SdrPathObj *>( pObj ) != nullptr) { basegfx::B2DPolyPolygon aPoly; switch(nID) { case SID_DRAW_BEZIER_FILL: { const sal_Int32 nWdt(rRectangle.GetWidth() / 2); const sal_Int32 nHgt(rRectangle.GetHeight() / 2); const basegfx::B2DPolygon aInnerPoly(basegfx::tools::createPolygonFromEllipse(basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Center().Y()), nWdt, nHgt)); aPoly.append(aInnerPoly); break; } case SID_DRAW_BEZIER_NOFILL: { basegfx::B2DPolygon aInnerPoly; aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Bottom())); const basegfx::B2DPoint aCenterBottom(rRectangle.Center().X(), rRectangle.Bottom()); aInnerPoly.appendBezierSegment( aCenterBottom, aCenterBottom, basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Center().Y())); const basegfx::B2DPoint aCenterTop(rRectangle.Center().X(), rRectangle.Top()); aInnerPoly.appendBezierSegment( aCenterTop, aCenterTop, basegfx::B2DPoint(rRectangle.Right(), rRectangle.Top())); aPoly.append(aInnerPoly); break; } case SID_DRAW_FREELINE: case SID_DRAW_FREELINE_NOFILL: { basegfx::B2DPolygon aInnerPoly; aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Bottom())); aInnerPoly.appendBezierSegment( basegfx::B2DPoint(rRectangle.Left(), rRectangle.Top()), basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Top()), basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Center().Y())); aInnerPoly.appendBezierSegment( basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Bottom()), basegfx::B2DPoint(rRectangle.Right(), rRectangle.Bottom()), basegfx::B2DPoint(rRectangle.Right(), rRectangle.Top())); if(SID_DRAW_FREELINE == nID) { aInnerPoly.append(basegfx::B2DPoint(rRectangle.Right(), rRectangle.Bottom())); } else { aInnerPoly.setClosed(true); } aPoly.append(aInnerPoly); break; } case SID_DRAW_XPOLYGON: case SID_DRAW_XPOLYGON_NOFILL: { basegfx::B2DPolygon aInnerPoly; aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Bottom())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Top())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Top())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Center().Y())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Right(), rRectangle.Center().Y())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Right(), rRectangle.Bottom())); if(SID_DRAW_XPOLYGON_NOFILL == nID) { aInnerPoly.append(basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Bottom())); } else { aInnerPoly.setClosed(true); } aPoly.append(aInnerPoly); break; } case SID_DRAW_POLYGON: case SID_DRAW_POLYGON_NOFILL: { basegfx::B2DPolygon aInnerPoly; const sal_Int32 nWdt(rRectangle.GetWidth()); const sal_Int32 nHgt(rRectangle.GetHeight()); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Bottom())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left() + (nWdt * 30) / 100, rRectangle.Top() + (nHgt * 70) / 100)); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left(), rRectangle.Top() + (nHgt * 15) / 100)); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left() + (nWdt * 65) / 100, rRectangle.Top())); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left() + nWdt, rRectangle.Top() + (nHgt * 30) / 100)); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left() + (nWdt * 80) / 100, rRectangle.Top() + (nHgt * 50) / 100)); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Left() + (nWdt * 80) / 100, rRectangle.Top() + (nHgt * 75) / 100)); aInnerPoly.append(basegfx::B2DPoint(rRectangle.Bottom(), rRectangle.Right())); if(SID_DRAW_POLYGON_NOFILL == nID) { aInnerPoly.append(basegfx::B2DPoint(rRectangle.Center().X(), rRectangle.Bottom())); } else { aInnerPoly.setClosed(true); } aPoly.append(aInnerPoly); break; } } static_cast(pObj)->SetPathPoly(aPoly); } else { OSL_FAIL("Object is NO path object"); } pObj->SetLogicRect(rRectangle); } return pObj; } } // end of namespace sd /* vim:set shiftwidth=4 softtabstop=4 expandtab: */