/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::linguistic2; namespace sd { const sal_uInt16 SidArray[] = { SID_STYLE_FAMILY2, // 5542 SID_STYLE_FAMILY5, // 5545 SID_REDO, // 5700 SID_UNDO, // 5701 SID_CUT, // 5710 SID_COPY, // 5711 SID_ATTR_TABSTOP, // 10002 SID_ATTR_CHAR_FONT, // 10007 SID_ATTR_CHAR_POSTURE, // 10008 SID_ATTR_CHAR_WEIGHT, // 10009 SID_ATTR_CHAR_SHADOWED, // 10010 SID_ATTR_CHAR_STRIKEOUT, // 10013 SID_ATTR_CHAR_UNDERLINE, // 10014 SID_ATTR_CHAR_FONTHEIGHT, // 10015 SID_ATTR_CHAR_COLOR, // 10017 SID_ATTR_CHAR_KERNING, // 10018 SID_ATTR_CHAR_CASEMAP, // 10019 SID_ATTR_PARA_ADJUST_LEFT, // 10028 SID_ATTR_PARA_ADJUST_RIGHT, // 10029 SID_ATTR_PARA_ADJUST_CENTER, // 10030 SID_ATTR_PARA_ADJUST_BLOCK, // 10031 SID_ATTR_PARA_LINESPACE_10, // 10034 SID_ATTR_PARA_LINESPACE_15, // 10035 SID_ATTR_PARA_LINESPACE_20, // 10036 SID_ATTR_PARA_ULSPACE, // 10042 SID_ATTR_PARA_LRSPACE, // 10043 SID_ATTR_TRANSFORM_POS_X, // 10088 SID_ATTR_TRANSFORM_POS_Y, // 10089 SID_ATTR_TRANSFORM_WIDTH, // 10090 SID_ATTR_TRANSFORM_HEIGHT, // 10091 SID_ATTR_TRANSFORM_ROT_X, // 10093 SID_ATTR_TRANSFORM_ROT_Y, // 10094 SID_ATTR_TRANSFORM_ANGLE, // 10095 //Added SID_OUTLINE_UP, // 10150 SID_OUTLINE_DOWN, // 10151 SID_OUTLINE_LEFT, // 10152 SID_OUTLINE_RIGHT, // 10153 SID_ATTR_TRANSFORM_PROTECT_POS, // 10236 SID_ATTR_TRANSFORM_PROTECT_SIZE, // 10237 //Added SID_FORMTEXT_STYLE, // 10257 SID_SET_SUPER_SCRIPT, // 10294 SID_SET_SUB_SCRIPT, // 10295 SID_ATTR_TRANSFORM_AUTOWIDTH, // 10310 SID_ATTR_TRANSFORM_AUTOHEIGHT, // 10311 //Added SID_HYPERLINK_GETLINK, // 10361 SID_DEC_INDENT, // 10461 SID_INC_INDENT, // 10462 SID_CHARMAP, // 10503 SID_TEXTDIRECTION_LEFT_TO_RIGHT, // 10907 SID_TEXTDIRECTION_TOP_TO_BOTTOM, // 10908 SID_ATTR_PARA_LEFT_TO_RIGHT, // 10950 SID_ATTR_PARA_RIGHT_TO_LEFT, // 10951 SID_PARASPACE_INCREASE, // 11145 SID_PARASPACE_DECREASE, // 11146 FN_NUM_BULLET_ON, // 20138 0 }; /** * base class for text functions */ FuText::FuText( ViewShell* pViewSh, ::sd::Window* pWin, ::sd::View* pView, SdDrawDocument* pDoc, SfxRequest& rReq ) : FuConstruct(pViewSh, pWin, pView, pDoc, rReq) , bFirstObjCreated(false) , bJustEndedEdit(false) , rRequest (rReq) { } rtl::Reference FuText::Create( ViewShell* pViewSh, ::sd::Window* pWin, ::sd::View* pView, SdDrawDocument* pDoc, SfxRequest& rReq ) { rtl::Reference xFunc( new FuText( pViewSh, pWin, pView, pDoc, rReq ) ); return xFunc; } void FuText::disposing() { if(mpView) { if(mpView->SdrEndTextEdit() == SdrEndTextEditKind::Deleted) mxTextObj = nullptr; // reset the RequestHandler of the used Outliner to the handler of the document ::Outliner* pOutliner = mpView->GetTextEditOutliner(); if (pOutliner) pOutliner->SetStyleSheetPool(static_cast(mpDoc->GetStyleSheetPool())); } } /************************************************************************* |* |* Execute functionality of this class: |* |* #71422: Start the functionality of this class in this method |* and not in the ctor. |* If you construct an object of this class and you put the |* address of this object to pFuActual you've got a problem, |* because some methods inside DoExecute use the pFuActual-Pointer. |* If the code inside DoExecute is executed inside the ctor, |* the value of pFuActual is not right. And the value will not |* be right until the ctor finished !!! |* \************************************************************************/ void FuText::DoExecute( SfxRequest& ) { mpViewShell->GetViewShellBase().GetToolBarManager()->SetToolBarShell( ToolBarManager::ToolBarGroup::Function, ToolbarId::Draw_Text_Toolbox_Sd); mpView->SetCurrentObj(SdrObjKind::Text); mpView->SetEditMode(SdrViewEditMode::Edit); MouseEvent aMEvt(mpWindow->GetPointerPosPixel()); if (nSlotId == SID_TEXTEDIT) { // Try to select an object SdrPageView* pPV = mpView->GetSdrPageView(); SdrViewEvent aVEvt; mpView->PickAnything(aMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); mpView->MarkObj(aVEvt.mpRootObj, pPV); mxTextObj = DynCastSdrTextObj( aVEvt.mpObj ); } else if (mpView->AreObjectsMarked()) { const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); if (rMarkList.GetMarkCount() == 1) { SdrObject* pObj = rMarkList.GetMark(0)->GetMarkedSdrObj(); mxTextObj = DynCastSdrTextObj( pObj ); } } // check for table if (mpView->AreObjectsMarked()) { const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); if (rMarkList.GetMarkCount() == 1) { SdrObject* pObj = rMarkList.GetMark(0)->GetMarkedSdrObj(); if( pObj && (pObj->GetObjInventor() == SdrInventor::Default ) && (pObj->GetObjIdentifier() == SdrObjKind::Table) ) { mpViewShell->GetViewShellBase().GetToolBarManager()->AddToolBarShell(ToolBarManager::ToolBarGroup::Function, ToolbarId::Draw_Table_Toolbox); } } } bool bQuickDrag = true; const SfxItemSet* pArgs = rRequest.GetArgs(); if (pArgs // test for type before using && SID_TEXTEDIT == nSlotId && SfxItemState::SET == pArgs->GetItemState(SID_TEXTEDIT) && pArgs->Get(SID_TEXTEDIT).GetValue() == 2) { // Selection by doubleclick -> don't allow QuickDrag bQuickDrag = false; } SetInEditMode(aMEvt, bQuickDrag); } bool FuText::MouseButtonDown(const MouseEvent& rMEvt) { bMBDown = true; bJustEndedEdit = false; bool bReturn = FuDraw::MouseButtonDown(rMEvt); SdrViewEvent aVEvt; SdrHitKind eHit = mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); // handle URL also during the text editing if (rMEvt.GetClicks() == 1 && rMEvt.IsLeft() && rMEvt.IsMod1()) { OutlinerView* pOLV = mpView->GetTextEditOutlinerView(); if (mxTextObj.get().is() && pOLV && pOLV->GetFieldUnderMousePointer()) { const SvxFieldItem* pFieldItem = pOLV->GetFieldUnderMousePointer(); if (pFieldItem) { const SvxFieldData* pField = pFieldItem->GetField(); if (auto pURLField = dynamic_cast< const SvxURLField *>( pField )) { eHit = SdrHitKind::MarkedObject; aVEvt.meEvent = SdrEventKind::ExecuteUrl; aVEvt.mpURLField = pURLField; } } } } if (eHit == SdrHitKind::TextEdit) { // hit text -> SdrView handles event if (mpView->MouseButtonDown(rMEvt, mpWindow->GetOutDev())) return true; } if (rMEvt.GetClicks() == 1) { if (mpView->IsTextEdit() && eHit != SdrHitKind::MarkedObject && eHit != SdrHitKind::Handle) { // finish text input if(mpView->SdrEndTextEdit() == SdrEndTextEditKind::Deleted) { /* Bugfix from MBA: during a double click onto the unused? area in text mode, we get with the second click eHit = SdrHitKind::TextEditObj since it goes to the TextObject which was created with the first click. But this is removed by SdrEndTextEdit since it is empty. But it is still in the mark list. The call MarkObj further below accesses then the dead object. As a simple fix, we determine eHit after SdrEndTextEdit again, this returns then SdrHitKind::NONE. */ mxTextObj = nullptr; eHit = mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); } mpView->SetCurrentObj(SdrObjKind::Text); mpView->SetEditMode(SdrViewEditMode::Edit); } if (rMEvt.IsLeft() || rMEvt.IsRight()) { mpWindow->CaptureMouse(); SdrPageView* pPV = mpView->GetSdrPageView(); if (eHit == SdrHitKind::TextEdit) { SetInEditMode(rMEvt, false); } else { // Don't remark table when clicking in it, mark change triggers a lot of updating bool bMarkChanges = true; rtl::Reference< sdr::SelectionController > xSelectionController(mpView->getSelectionController()); if (eHit == SdrHitKind::TextEditObj && xSelectionController.is()) { const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); if (rMarkList.GetMarkCount() == 1 && rMarkList.GetMark(0)->GetMarkedSdrObj() == aVEvt.mpRootObj) bMarkChanges = false; } if (eHit != SdrHitKind::Handle) { // deselect selection if (!rMEvt.IsShift() && eHit == SdrHitKind::TextEditObj) { if(bMarkChanges) { mpView->UnmarkAll(); mpView->SetDragMode(SdrDragMode::Move); } } } if ( aVEvt.meEvent == SdrEventKind::ExecuteUrl || eHit == SdrHitKind::Handle || eHit == SdrHitKind::MarkedObject || eHit == SdrHitKind::TextEditObj || ( eHit == SdrHitKind::UnmarkedObject && bFirstObjCreated && !bPermanent ) ) { // Handle, hit marked or unmarked object if (eHit == SdrHitKind::TextEditObj) { /* hit text of unmarked object: select object and set to EditMode */ if (bMarkChanges) mpView->MarkObj(aVEvt.mpRootObj, pPV); if (auto pSdrTextObj = DynCastSdrTextObj(aVEvt.mpObj)) { mxTextObj = pSdrTextObj; } SetInEditMode(rMEvt, true); } else if (aVEvt.meEvent == SdrEventKind::ExecuteUrl && !rMEvt.IsMod2()) { // execute URL mpWindow->ReleaseMouse(); if (aVEvt.mpURLField) { SfxStringItem aStrItem(SID_FILE_NAME, aVEvt.mpURLField->GetURL()); SfxStringItem aReferer(SID_REFERER, mpDocSh->GetMedium()->GetName()); SfxBoolItem aBrowseItem( SID_BROWSE, true ); SfxViewFrame* pFrame = mpViewShell->GetViewFrame(); if (rMEvt.IsMod1()) { // open in new frame pFrame->GetDispatcher()->ExecuteList(SID_OPENDOC, SfxCallMode::ASYNCHRON | SfxCallMode::RECORD, { &aStrItem, &aBrowseItem, &aReferer }); } else { // open in current frame SfxFrameItem aFrameItem(SID_DOCFRAME, pFrame); pFrame->GetDispatcher()->ExecuteList(SID_OPENDOC, SfxCallMode::ASYNCHRON | SfxCallMode::RECORD, { &aStrItem, &aFrameItem, &aBrowseItem, &aReferer }); } } } else { // drag object or handle // #i78748# // do the EndTextEdit first, it will delete the handles and force a // recreation. This will make aVEvt.mpHdl to point to a deleted handle, // thus it is necessary to reset it and to get it again. // #i112855# // cl: I'm not sure why we checked here also for mxTextObj->GetOutlinerParaObject // this caused SdrEndTextEdit() to be called also when not in text editing and // this does not make sense and caused troubles. (see issue 112855) if( mpView->IsTextEdit() ) { mpView->SdrEndTextEdit(); bJustEndedEdit = true; if(aVEvt.mpHdl) { // force new handle identification, the pointer will be dead here // since SdrEndTextEdit has reset (deleted) the handles. aVEvt.mpHdl = nullptr; mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); } } if (!aVEvt.mpHdl) { if( eHit == SdrHitKind::UnmarkedObject ) { if ( !rMEvt.IsShift() ) mpView->UnmarkAll(); mpView->MarkObj(aVEvt.mpRootObj, pPV); } // Drag object bFirstMouseMove = true; aDragTimer.Start(); } if ( ! rMEvt.IsRight()) { // we need to pick again since SdrEndTextEdit can rebuild the handles list eHit = mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); if( (eHit == SdrHitKind::Handle) || (eHit == SdrHitKind::MarkedObject) ) { sal_uInt16 nDrgLog = sal_uInt16 ( mpWindow->PixelToLogic(Size(mpView->GetDragThresholdPixels(),0)).Width() ); mpView->BegDragObj(aMDPos, nullptr, aVEvt.mpHdl, nDrgLog); } } bReturn = true; } } else if ( nSlotId != SID_TEXTEDIT && (bPermanent || !bFirstObjCreated) ) { // create object mpView->SetCurrentObj(SdrObjKind::Text); mpView->SetEditMode(SdrViewEditMode::Create); sal_uInt16 nDrgLog = sal_uInt16 ( mpWindow->PixelToLogic(Size(mpView->GetDragThresholdPixels(),0)).Width() ); mpView->BegCreateObj(aMDPos, nullptr, nDrgLog); } else { // select if( !rMEvt.IsShift() ) mpView->UnmarkAll(); mpView->BegMarkObj( aMDPos ); } } } } else if ( rMEvt.GetClicks() == 2 && !mpView->IsTextEdit() ) { MouseEvent aMEvt( mpWindow->GetPointerPosPixel() ); SetInEditMode( aMEvt, false ); } if (!bIsInDragMode) { ForcePointer(&rMEvt); mpViewShell->GetViewFrame()->GetBindings().Invalidate(SidArray); } return bReturn; } bool FuText::MouseMove(const MouseEvent& rMEvt) { bool bReturn = FuDraw::MouseMove(rMEvt); if (aDragTimer.IsActive() ) { if( bFirstMouseMove ) bFirstMouseMove = false; else aDragTimer.Stop(); } if (!bReturn && mpView->IsAction() && !mpDocSh->IsReadOnly()) { Point aPix(rMEvt.GetPosPixel()); Point aPnt(mpWindow->PixelToLogic(aPix)); ForceScroll(aPix); mpView->MovAction(aPnt); } ForcePointer(&rMEvt); return bReturn; } void FuText::ImpSetAttributesForNewTextObject(SdrTextObj* pTxtObj) { if( nSlotId == SID_ATTR_CHAR ) { SfxItemSet aSet(mpViewShell->GetPool()); aSet.Put(makeSdrTextAutoGrowWidthItem(false)); aSet.Put(makeSdrTextAutoGrowHeightItem(true)); pTxtObj->SetMergedItemSet(aSet); pTxtObj->AdjustTextFrameWidthAndHeight(); const SfxViewShell* pCurrentViewShell = SfxViewShell::Current(); if (pCurrentViewShell && (pCurrentViewShell->isLOKMobilePhone() || pCurrentViewShell->isLOKTablet())) pTxtObj->SetText(SdResId(STR_PRESOBJ_TEXT_EDIT_MOBILE)); } else if( nSlotId == SID_ATTR_CHAR_VERTICAL ) { // draw text object, needs to be initialized when vertical text is used SfxItemSet aSet(mpViewShell->GetPool()); aSet.Put(makeSdrTextAutoGrowWidthItem(true)); aSet.Put(makeSdrTextAutoGrowHeightItem(false)); // Set defaults for vertical click-n'drag text object, pool defaults are: // SdrTextVertAdjustItem: SDRTEXTVERTADJUST_TOP // SdrTextHorzAdjustItem: SDRTEXTHORZADJUST_BLOCK // Analog to that: aSet.Put(SdrTextVertAdjustItem(SDRTEXTVERTADJUST_BLOCK)); aSet.Put(SdrTextHorzAdjustItem(SDRTEXTHORZADJUST_RIGHT)); pTxtObj->SetMergedItemSet(aSet); pTxtObj->AdjustTextFrameWidthAndHeight(); } } void FuText::ImpSetAttributesFitToSize(SdrTextObj* pTxtObj) { // FitToSize (fit to frame) SfxItemSetFixed aSet(mpViewShell->GetPool()); aSet.Put(SdrTextFitToSizeTypeItem(drawing::TextFitToSizeType_PROPORTIONAL)); aSet.Put(makeSdrTextAutoGrowHeightItem(false)); aSet.Put(makeSdrTextAutoGrowWidthItem(false)); pTxtObj->SetMergedItemSet(aSet); pTxtObj->AdjustTextFrameWidthAndHeight(); } void FuText::ImpSetAttributesFitToSizeVertical(SdrTextObj* pTxtObj) { SfxItemSetFixed aSet(mpViewShell->GetPool()); aSet.Put(SdrTextFitToSizeTypeItem(drawing::TextFitToSizeType_PROPORTIONAL)); aSet.Put(makeSdrTextAutoGrowHeightItem(false)); aSet.Put(makeSdrTextAutoGrowWidthItem(false)); pTxtObj->SetMergedItemSet(aSet); pTxtObj->AdjustTextFrameWidthAndHeight(); } bool FuText::MouseButtonUp(const MouseEvent& rMEvt) { bool bReturn = false; if (aDragTimer.IsActive()) { aDragTimer.Stop(); bIsInDragMode = false; } mpViewShell->GetViewFrame()->GetBindings().Invalidate( SidArray ); Point aPnt( mpWindow->PixelToLogic( rMEvt.GetPosPixel() ) ); if( (mpView && mpView->MouseButtonUp(rMEvt, mpWindow->GetOutDev())) || rMEvt.GetClicks() == 2 ) return true; // handle event from SdrView bool bEmptyTextObj = false; if (mxTextObj.get().is()) { bool bReset = true; if (mpView) { const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); if (rMarkList.GetMarkCount() == 1 && ( rMarkList.GetMark(0)->GetMarkedSdrObj() == mxTextObj.get().get()) ) { if (!GetTextObj()->GetOutlinerParaObject() ) bEmptyTextObj = true; else bFirstObjCreated = true; bReset = false; } } if (bReset) { mxTextObj = nullptr; } } if (rMEvt.IsLeft() && !mxTextObj.get().is() && IsIgnoreUnexpectedMouseButtonUp()) return false; if( mpView && mpView->IsDragObj()) { // object was moved FrameView* pFrameView = mpViewShell->GetFrameView(); bool bDragWithCopy = (rMEvt.IsMod1() && pFrameView->IsDragWithCopy()); if (bDragWithCopy) { bDragWithCopy = !mpView->IsPresObjSelected(false); } mpView->SetDragWithCopy(bDragWithCopy); mpView->EndDragObj( mpView->IsDragWithCopy() ); mpView->ForceMarkedToAnotherPage(); mpView->SetCurrentObj(SdrObjKind::Text); sal_uInt16 nDrgLog = sal_uInt16 ( mpWindow->PixelToLogic(Size(mpView->GetDragThresholdPixels(),0)).Width() ); if (bJustEndedEdit) { bJustEndedEdit = false; FuPoor::cancel(); } if ((rMEvt.GetClicks() != 2) && !rMEvt.IsShift() && !rMEvt.IsMod1() && !rMEvt.IsMod2() && !rMEvt.IsRight() && std::abs(aPnt.X() - aMDPos.X()) < nDrgLog && std::abs(aPnt.Y() - aMDPos.Y()) < nDrgLog) { /************************************************************* * From text mode, you don't want to rotate immediately. **************************************************************/ SdrPageView* pPV; SdrObject* pObj = mpView->PickObj(aMDPos, mpView->getHitTolLog(), pPV, SdrSearchOptions::ALSOONMASTER | SdrSearchOptions::BEFOREMARK); if (pObj && pPV->IsObjMarkable(pObj)) { mpView->UnmarkAllObj(); mpView->MarkObj(pObj,pPV); return bReturn; } } } else if( mpView && mpView->IsCreateObj() && rMEvt.IsLeft()) { // object was created rtl::Reference pTextObj = DynCastSdrTextObj( mpView->GetCreateObj() ); mxTextObj = pTextObj.get(); if( pTextObj ) { //AW outliner needs to be set to vertical when there is no // outliner object up to now; also it needs to be set back to not // vertical when there was a vertical one used last time. OutlinerParaObject* pOPO = GetTextObj()->GetOutlinerParaObject(); SdrOutliner& rOutl(pTextObj->getSdrModelFromSdrObject().GetDrawOutliner(GetTextObj())); bool bVertical((pOPO && pOPO->IsEffectivelyVertical()) || nSlotId == SID_ATTR_CHAR_VERTICAL || nSlotId == SID_TEXT_FITTOSIZE_VERTICAL); rOutl.SetVertical(bVertical); // Before ImpSetAttributesForNewTextObject the vertical writing mode // needs to be set at the object. This is done here at the OutlinerParaObject // directly to not mirror the layout text items involved. These items will be set // from ImpSetAttributesForNewTextObject and below. OutlinerParaObject* pPara = GetTextObj()->GetOutlinerParaObject(); if(!pPara) { GetTextObj()->ForceOutlinerParaObject(); pPara = GetTextObj()->GetOutlinerParaObject(); } if(pPara && bVertical != pPara->IsEffectivelyVertical()) { // set ParaObject orientation accordingly pPara->SetVertical(bVertical); } ImpSetAttributesForNewTextObject(GetTextObj()); } if (!mpView->EndCreateObj(SdrCreateCmd::ForceEnd)) { // it was not possible to create text object mxTextObj = nullptr; pTextObj = nullptr; } else if (nSlotId == SID_TEXT_FITTOSIZE) { ImpSetAttributesFitToSize(GetTextObj()); SetInEditMode(rMEvt, false); } else if ( nSlotId == SID_TEXT_FITTOSIZE_VERTICAL ) { ImpSetAttributesFitToSizeVertical(GetTextObj()); SetInEditMode(rMEvt, false); } else { // thereby the handles and the gray frame are correct mpView->AdjustMarkHdl(); mpView->PickHandle(aPnt); SetInEditMode(rMEvt, false); } } else if ( mpView && mpView->IsAction()) { mpView->EndAction(); } ForcePointer(&rMEvt); mpWindow->ReleaseMouse(); if ( mpView && !mpView->AreObjectsMarked() ) { sal_uInt16 nDrgLog1 = sal_uInt16 ( mpWindow->PixelToLogic(Size(mpView->GetDragThresholdPixels(),0)).Width() ); if ( std::abs(aMDPos.X() - aPnt.X()) < nDrgLog1 && std::abs(aMDPos.Y() - aPnt.Y()) < nDrgLog1 && !rMEvt.IsShift() && !rMEvt.IsMod2() ) { SdrPageView* pPV2 = mpView->GetSdrPageView(); SdrViewEvent aVEvt; mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); mpView->MarkObj(aVEvt.mpRootObj, pPV2); } } if ( !mxTextObj.get().is() && mpView ) { if ( ( (!bEmptyTextObj && bPermanent) || (!bFirstObjCreated && !bPermanent) ) && !mpDocSh->IsReadOnly() && nSlotId != SID_TEXTEDIT ) { // text body (left-justified AutoGrow) mpView->SetCurrentObj(SdrObjKind::Text); mpView->SetEditMode(SdrViewEditMode::Create); sal_uInt16 nDrgLog = sal_uInt16 ( mpWindow->PixelToLogic(Size(mpView->GetDragThresholdPixels(),0)).Width() ); mpView->BegCreateObj(aMDPos, nullptr, nDrgLog); bool bSnapEnabled = mpView->IsSnapEnabled(); if (bSnapEnabled) mpView->SetSnapEnabled(false); aPnt.AdjustX(nDrgLog + nDrgLog ); aPnt.AdjustY(nDrgLog + nDrgLog ); mpView->MovAction(aPnt); mxTextObj = DynCastSdrTextObj( mpView->GetCreateObj() ); if(mxTextObj.get().is()) { GetTextObj()->SetDisableAutoWidthOnDragging(true); } if(!mpView->EndCreateObj(SdrCreateCmd::ForceEnd)) { mxTextObj.clear(); } if(bSnapEnabled) mpView->SetSnapEnabled(bSnapEnabled); if(mxTextObj.get().is()) { SfxItemSet aSet(mpViewShell->GetPool()); aSet.Put(makeSdrTextMinFrameHeightItem(0)); aSet.Put(makeSdrTextMinFrameWidthItem(0)); aSet.Put(makeSdrTextAutoGrowHeightItem(true)); aSet.Put(makeSdrTextAutoGrowWidthItem(true)); if(nSlotId == SID_ATTR_CHAR_VERTICAL) { // Here, all items which need to be different from pool default need to be set // again on the newly created text object. // Since this is a simple click text object, it is first created, then SetVertical() // is used, then ImpSetAttributesForNewTextObject is called and then the object is // deleted again since not the minimum drag distance was travelled. Then, a new // click text object is created and thus all that stuff needs to be set again here. // Before using the new object the vertical writing mode // needs to be set. This is done here at the OutlinerParaObject // directly to not mirror the layout text items involved. These items will be set // below. OutlinerParaObject* pPara = GetTextObj()->GetOutlinerParaObject(); if(!pPara) { GetTextObj()->ForceOutlinerParaObject(); pPara = GetTextObj()->GetOutlinerParaObject(); } if(pPara && !pPara->IsEffectivelyVertical()) { // set ParaObject orientation accordingly pPara->SetVertical(true); } aSet.Put(SdrTextHorzAdjustItem(SDRTEXTHORZADJUST_RIGHT)); // Analog to the else case below, for vertical simple click texts // one of the default set items from ImpSetAttributesForNewTextObject // needs to be adapted to non-block mode. const SfxItemSet& rSet = mpView->GetDefaultAttr(); SvxFrameDirection eDirection = rSet.Get(EE_PARA_WRITINGDIR).GetValue(); if(SvxFrameDirection::Horizontal_RL_TB == eDirection || SvxFrameDirection::Vertical_RL_TB == eDirection) { aSet.Put(SdrTextVertAdjustItem(SDRTEXTVERTADJUST_BOTTOM)); } else { aSet.Put(SdrTextVertAdjustItem(SDRTEXTVERTADJUST_TOP)); } } else { // This is for Format/Page settings. Since this also leads // to the object defaults to be changed, i think this code can be // removed. CL. wanted to take a look before adding this. // Look in the object defaults if left-to-right is wanted. If // yes, set text anchoring to right to let the box grow to left. const SfxItemSet& rSet = mpView->GetDefaultAttr(); SvxFrameDirection eDirection = rSet.Get(EE_PARA_WRITINGDIR).GetValue(); if(SvxFrameDirection::Horizontal_RL_TB == eDirection) { aSet.Put(SdrTextHorzAdjustItem(SDRTEXTHORZADJUST_RIGHT)); } else { aSet.Put(SdrTextHorzAdjustItem(SDRTEXTHORZADJUST_LEFT)); } } GetTextObj()->SetMergedItemSet(aSet); GetTextObj()->SetDisableAutoWidthOnDragging(true); SetInEditMode(rMEvt, false); } bFirstObjCreated = true; } else { // switch to selection if (mpView->SdrEndTextEdit() == SdrEndTextEditKind::Deleted) { mxTextObj = nullptr; } mpViewShell->GetViewFrame()->GetDispatcher()->Execute( SID_OBJECT_SELECT, SfxCallMode::ASYNCHRON | SfxCallMode::RECORD ); } } if (bJustEndedEdit) { bJustEndedEdit = false; FuPoor::cancel(); } bMBDown = false; FuConstruct::MouseButtonUp(rMEvt); return bReturn; } /** * handle keyboard events * @returns sal_True if the event was handled, sal_False otherwise */ bool FuText::KeyInput(const KeyEvent& rKEvt) { bool bReturn = false; vcl::KeyCode nCode = rKEvt.GetKeyCode(); bool bShift = nCode.IsShift(); if(mxTextObj.get().is()) { // maybe object is deleted, test if it's equal to the selected object const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); SdrObject* pSelectedObj = nullptr; if(1 == rMarkList.GetMarkCount()) { SdrMark* pMark = rMarkList.GetMark(0); pSelectedObj = pMark->GetMarkedSdrObj(); } if(mxTextObj.get().get() != pSelectedObj) { mxTextObj = nullptr; } } if (auto pTextObj = mxTextObj.get()) if ( pTextObj->GetObjInventor() == SdrInventor::Default && pTextObj->GetObjIdentifier() == SdrObjKind::TitleText && rKEvt.GetKeyCode().GetCode() == KEY_RETURN ) { // title text object: always soft breaks bShift = true; } sal_uInt16 nKey = nCode.GetCode(); vcl::KeyCode aKeyCode (nKey, bShift, nCode.IsMod1(), nCode.IsMod2(), nCode.IsMod3() ); KeyEvent aKEvt(rKEvt.GetCharCode(), aKeyCode); bool bOK = true; if (mpDocSh->IsReadOnly()) { bOK = !EditEngine::DoesKeyChangeText(aKEvt); } if( aKeyCode.GetCode() == KEY_PAGEUP || aKeyCode.GetCode() == KEY_PAGEDOWN ) { bOK = false; // default handling in base class } if (bOK && mpView->KeyInput(aKEvt, mpWindow) ) { bReturn = true; mpViewShell->GetViewFrame()->GetBindings().Invalidate( SidArray ); } else if (aKeyCode == KEY_ESCAPE) { bReturn = cancel(); } if( bPermanent ) { mpView->SetCurrentObj(SdrObjKind::Text); mpView->SetEditMode(SdrViewEditMode::Create); } if (!bReturn) { bReturn = FuDraw::KeyInput(aKEvt); } return bReturn; } void FuText::Activate() { mpView->SetQuickTextEditMode(mpViewShell->GetFrameView()->IsQuickEdit()); // #i89661# it's no longer necessary to make it so big here, it's fine tuned // for text objects in SdrMarkView::CheckSingleSdrObjectHit mpView->SetHitTolerancePixel( 2 * HITPIX ); OutlinerView* pOLV = mpView->GetTextEditOutlinerView(); if (pOLV) pOLV->ShowCursor(/*bGotoCursor=*/true, /*bActivate=*/true); FuConstruct::Activate(); if( pOLV ) mpView->SetEditMode(SdrViewEditMode::Edit); } void FuText::Deactivate() { OutlinerView* pOLV = mpView->GetTextEditOutlinerView(); if (pOLV) pOLV->HideCursor(/*bDeactivate=*/true); mpView->SetHitTolerancePixel( HITPIX ); FuConstruct::Deactivate(); } /** * Sets the object into the edit mode. */ void FuText::SetInEditMode(const MouseEvent& rMEvt, bool bQuickDrag) { SdrPageView* pPV = mpView->GetSdrPageView(); if( mxTextObj.get().is() && (mxTextObj.get()->getSdrPageFromSdrObject() == pPV->GetPage()) ) { mpView->SetCurrentObj(SdrObjKind::Text); if( bPermanent ) mpView->SetEditMode(SdrViewEditMode::Create); else mpView->SetEditMode(SdrViewEditMode::Edit); bool bEmptyOutliner = false; if (!GetTextObj()->GetOutlinerParaObject() && mpView->GetTextEditOutliner()) { ::Outliner* pOutl = mpView->GetTextEditOutliner(); sal_Int32 nParagraphCnt = pOutl->GetParagraphCount(); Paragraph* p1stPara = pOutl->GetParagraph( 0 ); if (nParagraphCnt==1 && p1stPara) { // with only one paragraph if (pOutl->GetText(p1stPara).isEmpty()) { bEmptyOutliner = true; } } } if (GetTextObj() != mpView->GetTextEditObject() || bEmptyOutliner) { rtl::Reference pTextObj = mxTextObj.get(); SdrInventor nInv = pTextObj->GetObjInventor(); SdrObjKind nSdrObjKind = pTextObj->GetObjIdentifier(); if (nInv == SdrInventor::Default && GetTextObj()->HasTextEdit() && (nSdrObjKind == SdrObjKind::Text || nSdrObjKind == SdrObjKind::TitleText || nSdrObjKind == SdrObjKind::OutlineText || !pTextObj->IsEmptyPresObj() ) ) { // create new outliner (owned by SdrObjEditView) std::unique_ptr pOutl = SdrMakeOutliner(OutlinerMode::OutlineObject, *mpDoc); if (bEmptyOutliner) mpView->SdrEndTextEdit(true); pTextObj = GetTextObj(); if( pTextObj ) { OutlinerParaObject* pOPO = pTextObj->GetOutlinerParaObject(); if( pOPO && pOPO->IsEffectivelyVertical() ) { pOutl->SetVertical(pOPO->GetVertical()); pOutl->SetRotation(pOPO->GetRotation()); } else if (nSlotId == SID_ATTR_CHAR_VERTICAL || nSlotId == SID_TEXT_FITTOSIZE_VERTICAL) pOutl->SetVertical( true ); if( pTextObj->getTextCount() > 1 ) { Point aPix(rMEvt.GetPosPixel()); Point aPnt(mpWindow->PixelToLogic(aPix)); pTextObj->setActiveText( pTextObj->CheckTextHit(aPnt ) ); } if (mpView->SdrBeginTextEdit(pTextObj.get(), pPV, mpWindow, true, pOutl.release()) && mxTextObj.get()->GetObjInventor() == SdrInventor::Default) { //tdf#102293 flush overlay before going on to pass clicks down to //the outline view which will want to paint selections for (sal_uInt32 b = 0; b < pPV->PageWindowCount(); ++b) { const SdrPageWindow& rPageWindow = *pPV->GetPageWindow(b); if (!rPageWindow.GetPaintWindow().OutputToWindow()) continue; const rtl::Reference< sdr::overlay::OverlayManager >& xManager = rPageWindow.GetOverlayManager(); if (!xManager.is()) continue; xManager->flush(); } bFirstObjCreated = true; DeleteDefaultText(); OutlinerView* pOLV = mpView->GetTextEditOutlinerView(); nSdrObjKind = mxTextObj.get()->GetObjIdentifier(); SdrViewEvent aVEvt; SdrHitKind eHit = mpView->PickAnything(rMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); if (eHit == SdrHitKind::TextEdit) { // hit text if (nSdrObjKind == SdrObjKind::Text || nSdrObjKind == SdrObjKind::TitleText || nSdrObjKind == SdrObjKind::OutlineText || nSdrObjKind == SdrObjKind::Table || nSlotId == SID_TEXTEDIT || !bQuickDrag) { pOLV->MouseButtonDown(rMEvt); pOLV->MouseMove(rMEvt); pOLV->MouseButtonUp(rMEvt); } if (mpViewShell->GetFrameView()->IsQuickEdit() && bQuickDrag && GetTextObj()->GetOutlinerParaObject()) { pOLV->MouseButtonDown(rMEvt); } } else { // Move cursor to end of text ESelection aNewSelection(EE_PARA_NOT_FOUND, EE_INDEX_NOT_FOUND, EE_PARA_NOT_FOUND, EE_INDEX_NOT_FOUND); if (pOLV != nullptr) pOLV->SetSelection(aNewSelection); } } else { mpView->RestoreDefaultText( mxTextObj.get().get() ); } } } } } else { mxTextObj = nullptr; } } /** * Text entry is started, if necessary delete the default text. */ void FuText::DeleteDefaultText() { if ( !(mxTextObj.get().is() && mxTextObj.get()->IsEmptyPresObj()) ) return; SdPage* pPage = static_cast( mxTextObj.get()->getSdrPageFromSdrObject() ); if (!pPage) return; PresObjKind ePresObjKind = pPage->GetPresObjKind(mxTextObj.get().get()); if ( !(ePresObjKind == PresObjKind::Title || ePresObjKind == PresObjKind::Outline || ePresObjKind == PresObjKind::Notes || ePresObjKind == PresObjKind::Text ) || pPage->IsMasterPage() ) return; ::Outliner* pOutliner = mpView->GetTextEditOutliner(); SfxStyleSheet* pSheet = pOutliner->GetStyleSheet( 0 ); bool bIsUndoEnabled = pOutliner->IsUndoEnabled(); if( bIsUndoEnabled ) pOutliner->EnableUndo(false); pOutliner->SetText( OUString(), pOutliner->GetParagraph( 0 ) ); if( bIsUndoEnabled ) pOutliner->EnableUndo(true); if (pSheet && (ePresObjKind == PresObjKind::Notes || ePresObjKind == PresObjKind::Text)) pOutliner->SetStyleSheet(0, pSheet); mxTextObj.get()->SetEmptyPresObj(true); } bool FuText::RequestHelp(const HelpEvent& rHEvt) { bool bReturn = false; OutlinerView* pOLV = mpView->GetTextEditOutlinerView(); if ((Help::IsBalloonHelpEnabled() || Help::IsQuickHelpEnabled()) && mxTextObj.get().is() && pOLV && pOLV->GetFieldUnderMousePointer()) { OUString aHelpText; const SvxFieldItem* pFieldItem = pOLV->GetFieldUnderMousePointer(); const SvxFieldData* pField = pFieldItem->GetField(); if (auto pURLField = dynamic_cast< const SvxURLField *>( pField )) { // URL-Field aHelpText = INetURLObject::decode( pURLField->GetURL(), INetURLObject::DecodeMechanism::WithCharset ); } if (!aHelpText.isEmpty()) { ::tools::Rectangle aLogicPix = mpWindow->LogicToPixel(mxTextObj.get()->GetLogicRect()); ::tools::Rectangle aScreenRect(mpWindow->OutputToScreenPixel(aLogicPix.TopLeft()), mpWindow->OutputToScreenPixel(aLogicPix.BottomRight())); if (Help::IsBalloonHelpEnabled()) { Help::ShowBalloon( static_cast(mpWindow), rHEvt.GetMousePosPixel(), aScreenRect, aHelpText); bReturn = true; } else if (Help::IsQuickHelpEnabled()) { Help::ShowQuickHelp( static_cast(mpWindow), aScreenRect, aHelpText); bReturn = true; } } } if (!bReturn) { bReturn = FuConstruct::RequestHelp(rHEvt); } return bReturn; } void FuText::ReceiveRequest(SfxRequest& rReq) { nSlotId = rReq.GetSlot(); // then we call the base class (besides others, nSlotId is NOT set there) FuPoor::ReceiveRequest(rReq); if (!(nSlotId == SID_TEXTEDIT || mpViewShell->GetFrameView()->IsQuickEdit() || SID_ATTR_CHAR == nSlotId)) return; MouseEvent aMEvt(mpWindow->GetPointerPosPixel()); mxTextObj = nullptr; if (nSlotId == SID_TEXTEDIT) { // are we currently editing? mxTextObj = mpView->GetTextEditObject(); if (!mxTextObj.get().is()) { // Try to select an object SdrPageView* pPV = mpView->GetSdrPageView(); SdrViewEvent aVEvt; mpView->PickAnything(aMEvt, SdrMouseEventKind::BUTTONDOWN, aVEvt); mpView->MarkObj(aVEvt.mpRootObj, pPV); if (auto pSdrTextObj = DynCastSdrTextObj(aVEvt.mpObj)) { mxTextObj = pSdrTextObj; } } } else if (mpView->AreObjectsMarked()) { const SdrMarkList& rMarkList = mpView->GetMarkedObjectList(); if (rMarkList.GetMarkCount() == 1) { SdrObject* pObj = rMarkList.GetMark(0)->GetMarkedSdrObj(); if( auto pTextObj = DynCastSdrTextObj( pObj )) { mxTextObj = pTextObj; } } } bool bQuickDrag = true; const SfxItemSet* pArgs = rReq.GetArgs(); if (pArgs // test for type before using && SID_TEXTEDIT == nSlotId && SfxItemState::SET == pArgs->GetItemState(SID_TEXTEDIT) && pArgs->Get(SID_TEXTEDIT).GetValue() == 2) { // selection with double click -> do not allow QuickDrag bQuickDrag = false; } SetInEditMode(aMEvt, bQuickDrag); } void FuText::DoubleClick(const MouseEvent& ) { // Nothing to do } /** Removed the insertion of default text and putting a new text object directly into edit mode. */ rtl::Reference FuText::CreateDefaultObject(const sal_uInt16 nID, const ::tools::Rectangle& rRectangle) { rtl::Reference pObj( SdrObjFactory::MakeNewObject( mpView->getSdrModelFromSdrView(), mpView->GetCurrentObjInventor(), mpView->GetCurrentObjIdentifier(), nullptr) ); if(pObj) { if( auto pText = DynCastSdrTextObj( pObj.get() ) ) { pText->SetLogicRect(rRectangle); bool bVertical = (SID_ATTR_CHAR_VERTICAL == nID || SID_TEXT_FITTOSIZE_VERTICAL == nID); pText->SetVerticalWriting(bVertical); ImpSetAttributesForNewTextObject(pText); if (nSlotId == SID_TEXT_FITTOSIZE) { ImpSetAttributesFitToSize(pText); } else if ( nSlotId == SID_TEXT_FITTOSIZE_VERTICAL ) { ImpSetAttributesFitToSizeVertical(pText); } // Put text object into edit mode. SdrPageView* pPV = mpView->GetSdrPageView(); mpView->SdrBeginTextEdit(pText, pPV); } else { OSL_FAIL("Object is NO text object"); } } return pObj; } /** is called when the current function should be aborted.

This is used when a function gets a KEY_ESCAPE but can also be called directly. @returns true if an active function was aborted */ bool FuText::cancel() { if ( mpView->IsTextEdit() ) { if(mpView->SdrEndTextEdit() == SdrEndTextEditKind::Deleted) mxTextObj = nullptr; mpView->SetCurrentObj(SdrObjKind::Text); mpView->SetEditMode(SdrViewEditMode::Edit); return true; } else { return false; } } void FuText::ChangeFontSize( bool bGrow, OutlinerView* pOLV, const FontList* pFontList, ::sd::View* pView ) { if( !pFontList || !pView ) return; if( pOLV ) { pOLV->GetEditView().ChangeFontSize( bGrow, pFontList ); } else { pView->BegUndo(SdResId(bGrow ? STR_GROW_FONT_SIZE : STR_SHRINK_FONT_SIZE)); const SdrMarkList& rMarkList = pView->GetMarkedObjectList(); for( size_t nMark = 0; nMark < rMarkList.GetMarkCount(); ++nMark ) { SdrTextObj* pTextObj = DynCastSdrTextObj( rMarkList.GetMark(nMark)->GetMarkedSdrObj() ); if( pTextObj ) { rtl::Reference xSelectionController(pView->getSelectionController()); if (xSelectionController.is() && xSelectionController->ChangeFontSize(bGrow, pFontList)) { continue; } for( sal_Int32 nText = 0; nText < pTextObj->getTextCount(); nText++ ) { pTextObj->setActiveText( nText ); // Put text object into edit mode. SdrPageView* pPV = pView->GetSdrPageView(); pView->SdrBeginTextEdit(pTextObj, pPV); pOLV = pView->GetTextEditOutlinerView(); if( pOLV ) { EditEngine& rEditEngine = pOLV->GetEditView().getEditEngine(); ESelection aSel; aSel.nEndPara = rEditEngine.GetParagraphCount() - 1; aSel.nEndPos = rEditEngine.GetTextLen(aSel.nEndPara); pOLV->SetSelection(aSel); ChangeFontSize( bGrow, pOLV, pFontList, pView ); } pView->SdrEndTextEdit(); } SfxItemSet aShapeSet( pTextObj->GetMergedItemSet() ); if( EditView::ChangeFontSize( bGrow, aShapeSet, pFontList ) ) { pTextObj->SetObjectItemNoBroadcast( aShapeSet.Get( EE_CHAR_FONTHEIGHT ) ); pTextObj->SetObjectItemNoBroadcast( aShapeSet.Get( EE_CHAR_FONTHEIGHT_CJK ) ); pTextObj->SetObjectItemNoBroadcast( aShapeSet.Get( EE_CHAR_FONTHEIGHT_CTL ) ); } } } pView->EndUndo(); } } void FuText::InvalidateBindings() { mpViewShell->GetViewFrame()->GetBindings().Invalidate(SidArray); } } // end of namespace sd /* vim:set shiftwidth=4 softtabstop=4 expandtab: */