3917 lines
171 KiB
C++
3917 lines
171 KiB
C++
/* -*- 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 <sal/config.h>
|
|
|
|
#include <config_features.h>
|
|
|
|
#include <com/sun/star/i18n/WordType.hpp>
|
|
#include <com/sun/star/linguistic2/XThesaurus.hpp>
|
|
#include <com/sun/star/text/XContentControlsSupplier.hpp>
|
|
|
|
#include <hintids.hxx>
|
|
#include <cmdid.h>
|
|
#include <comphelper/lok.hxx>
|
|
|
|
#include <i18nutil/unicode.hxx>
|
|
#include <i18nlangtag/languagetag.hxx>
|
|
#include <svtools/langtab.hxx>
|
|
#include <svl/numformat.hxx>
|
|
#include <svl/slstitm.hxx>
|
|
#include <svl/stritem.hxx>
|
|
#include <sfx2/htmlmode.hxx>
|
|
#include <svl/whiter.hxx>
|
|
#include <sfx2/bindings.hxx>
|
|
#include <sfx2/namedcolor.hxx>
|
|
#include <sfx2/viewfrm.hxx>
|
|
#include <vcl/unohelp2.hxx>
|
|
#include <vcl/weld.hxx>
|
|
#include <sfx2/lokhelper.hxx>
|
|
#include <sfx2/request.hxx>
|
|
#include <svl/eitem.hxx>
|
|
#include <editeng/lrspitem.hxx>
|
|
#include <editeng/colritem.hxx>
|
|
#include <editeng/tstpitem.hxx>
|
|
#include <editeng/brushitem.hxx>
|
|
#include <editeng/svxacorr.hxx>
|
|
#include <svl/cjkoptions.hxx>
|
|
#include <svl/ctloptions.hxx>
|
|
#include <IDocumentDrawModelAccess.hxx>
|
|
#include <IDocumentSettingAccess.hxx>
|
|
#include <charfmt.hxx>
|
|
#include <svx/SmartTagItem.hxx>
|
|
#include <svx/xflgrit.hxx>
|
|
#include <svx/xflhtit.hxx>
|
|
#include <svx/xfillit0.hxx>
|
|
#include <fmtinfmt.hxx>
|
|
#include <wrtsh.hxx>
|
|
#include <wview.hxx>
|
|
#include <swmodule.hxx>
|
|
#include <viewopt.hxx>
|
|
#include <uitool.hxx>
|
|
#include <textsh.hxx>
|
|
#include <IMark.hxx>
|
|
#include <swdtflvr.hxx>
|
|
#include <swundo.hxx>
|
|
#include <reffld.hxx>
|
|
#include <textcontentcontrol.hxx>
|
|
#include <txatbase.hxx>
|
|
#include <docsh.hxx>
|
|
#include <inputwin.hxx>
|
|
#include <chrdlgmodes.hxx>
|
|
#include <fmtcol.hxx>
|
|
#include <cellatr.hxx>
|
|
#include <edtwin.hxx>
|
|
#include <fldmgr.hxx>
|
|
#include <ndtxt.hxx>
|
|
#include <strings.hrc>
|
|
#include <paratr.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include <sfx2/app.hxx>
|
|
#include <breakit.hxx>
|
|
#include <SwSmartTagMgr.hxx>
|
|
#include <editeng/acorrcfg.hxx>
|
|
#include <swabstdlg.hxx>
|
|
#include <sfx2/sfxdlg.hxx>
|
|
#include <com/sun/star/container/XNameContainer.hpp>
|
|
#include <com/sun/star/beans/XPropertySet.hpp>
|
|
#include <com/sun/star/style/XStyleFamiliesSupplier.hpp>
|
|
#include <com/sun/star/uno/Any.hxx>
|
|
#include <com/sun/star/linguistic2/ProofreadingResult.hpp>
|
|
#include <com/sun/star/linguistic2/XDictionary.hpp>
|
|
#include <com/sun/star/linguistic2/XSpellAlternatives.hpp>
|
|
#include <editeng/unolingu.hxx>
|
|
#include <doc.hxx>
|
|
#include <drawdoc.hxx>
|
|
#include <view.hxx>
|
|
#include <pam.hxx>
|
|
#include <sfx2/objface.hxx>
|
|
#include <langhelper.hxx>
|
|
#include <uiitems.hxx>
|
|
#include <svx/nbdtmgfact.hxx>
|
|
#include <svx/nbdtmg.hxx>
|
|
#include <SwRewriter.hxx>
|
|
#include <svx/drawitem.hxx>
|
|
#include <numrule.hxx>
|
|
#include <memory>
|
|
#include <xmloff/odffields.hxx>
|
|
#include <bookmark.hxx>
|
|
#include <linguistic/misc.hxx>
|
|
#include <comphelper/sequenceashashmap.hxx>
|
|
#include <comphelper/scopeguard.hxx>
|
|
#include <authfld.hxx>
|
|
#include <config_wasm_strip.h>
|
|
#if HAVE_FEATURE_CURL && !ENABLE_WASM_STRIP_EXTRA
|
|
#include <officecfg/Office/Common.hxx>
|
|
#include <officecfg/Office/Linguistic.hxx>
|
|
#include <svl/visitem.hxx>
|
|
#include <translatelangselect.hxx>
|
|
#endif // HAVE_FEATURE_CURL && ENABLE_WASM_STRIP_EXTRA
|
|
#include <translatehelper.hxx>
|
|
#include <IDocumentContentOperations.hxx>
|
|
#include <IDocumentUndoRedo.hxx>
|
|
#include <fmtcntnt.hxx>
|
|
#include <fmtrfmrk.hxx>
|
|
#include <cntfrm.hxx>
|
|
#include <flyfrm.hxx>
|
|
#include <unoprnms.hxx>
|
|
#include <boost/property_tree/json_parser.hpp>
|
|
#include <formatcontentcontrol.hxx>
|
|
#include <rtl/uri.hxx>
|
|
#include <unotxdoc.hxx>
|
|
|
|
#include <com/sun/star/text/XTextEmbeddedObjectsSupplier.hpp>
|
|
#include <com/sun/star/chart2/XInternalDataProvider.hpp>
|
|
#include <com/sun/star/chart2/XChartDocument.hpp>
|
|
#include <com/sun/star/chart/XChartDocument.hpp>
|
|
#include <com/sun/star/chart/XChartDataArray.hpp>
|
|
#include <com/sun/star/chart2/XTitle.hpp>
|
|
#include <com/sun/star/chart2/XTitled.hpp>
|
|
#include <com/sun/star/chart/ChartDataRowSource.hpp>
|
|
#include <com/sun/star/util/XModifiable.hpp>
|
|
|
|
#include <com/sun/star/chart2/XCoordinateSystemContainer.hpp>
|
|
#include <com/sun/star/chart2/XChartTypeContainer.hpp>
|
|
#include <com/sun/star/chart2/XDataSeriesContainer.hpp>
|
|
#include <com/sun/star/util/XCloneable.hpp>
|
|
|
|
using namespace ::com::sun::star;
|
|
using namespace com::sun::star::beans;
|
|
using namespace ::com::sun::star::container;
|
|
using namespace com::sun::star::style;
|
|
using namespace svx::sidebar;
|
|
|
|
static void sw_CharDialogResult(const SfxItemSet* pSet, SwWrtShell &rWrtSh, std::shared_ptr<SfxItemSet> const & pCoreSet, bool bSel,
|
|
bool bSelectionPut, bool bApplyToParagraph, SfxRequest *pReq);
|
|
|
|
static void sw_CharDialog(SwWrtShell& rWrtSh, bool bUseDialog, bool bApplyToParagraph,
|
|
sal_uInt16 nSlot, const SfxItemSet* pArgs, SfxRequest* pReq)
|
|
{
|
|
FieldUnit eMetric = ::GetDfltMetric(dynamic_cast<SwWebView*>( &rWrtSh.GetView()) != nullptr );
|
|
SwModule::get()->PutItem(SfxUInt16Item(SID_ATTR_METRIC, static_cast<sal_uInt16>(eMetric)));
|
|
auto pCoreSet = std::make_shared<SfxItemSetFixed<
|
|
RES_CHRATR_BEGIN, RES_CHRATR_END - 1,
|
|
RES_TXTATR_INETFMT, RES_TXTATR_INETFMT,
|
|
RES_BACKGROUND, RES_SHADOW,
|
|
SID_ATTR_BORDER_INNER, SID_ATTR_BORDER_INNER,
|
|
SID_HTML_MODE, SID_HTML_MODE,
|
|
SID_ATTR_CHAR_WIDTH_FIT_TO_LINE, SID_ATTR_CHAR_WIDTH_FIT_TO_LINE,
|
|
FN_PARAM_SELECTION, FN_PARAM_SELECTION>> ( rWrtSh.GetView().GetPool() );
|
|
rWrtSh.GetCurAttr(*pCoreSet);
|
|
|
|
bool bSel = rWrtSh.HasSelection();
|
|
bool bSelectionPut = false;
|
|
if(bSel || rWrtSh.IsInWord())
|
|
{
|
|
if(!bSel)
|
|
{
|
|
rWrtSh.StartAction();
|
|
rWrtSh.Push();
|
|
if(!rWrtSh.SelectTextAttr( RES_TXTATR_INETFMT ))
|
|
rWrtSh.SelWrd();
|
|
}
|
|
pCoreSet->Put(SfxStringItem(FN_PARAM_SELECTION, rWrtSh.GetSelText()));
|
|
bSelectionPut = true;
|
|
if(!bSel)
|
|
{
|
|
rWrtSh.Pop(SwCursorShell::PopMode::DeleteCurrent);
|
|
rWrtSh.EndAction();
|
|
}
|
|
}
|
|
pCoreSet->Put(SfxUInt16Item(SID_ATTR_CHAR_WIDTH_FIT_TO_LINE, rWrtSh.GetScalingOfSelectedText()));
|
|
|
|
::ConvertAttrCharToGen(*pCoreSet);
|
|
|
|
// Setting the BoxInfo
|
|
::PrepareBoxInfo(*pCoreSet, rWrtSh);
|
|
|
|
pCoreSet->Put(SfxUInt16Item(SID_HTML_MODE, ::GetHtmlMode(rWrtSh.GetView().GetDocShell())));
|
|
VclPtr<SfxAbstractTabDialog> pDlg;
|
|
if ( bUseDialog && GetActiveView() )
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
pDlg.reset(pFact->CreateSwCharDlg(rWrtSh.GetView().GetFrameWeld(), rWrtSh.GetView(), *pCoreSet, SwCharDlgMode::Std));
|
|
|
|
if (nSlot == SID_CHAR_DLG_EFFECT)
|
|
pDlg->SetCurPageId(u"fonteffects"_ustr);
|
|
else if (nSlot == SID_CHAR_DLG_POSITION)
|
|
pDlg->SetCurPageId(u"position"_ustr);
|
|
else if (nSlot == SID_CHAR_DLG_FOR_PARAGRAPH)
|
|
pDlg->SetCurPageId(u"font"_ustr);
|
|
else if (pReq)
|
|
{
|
|
const SfxStringItem* pItem = (*pReq).GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pItem)
|
|
pDlg->SetCurPageId(pItem->GetValue());
|
|
}
|
|
}
|
|
|
|
if (bUseDialog)
|
|
{
|
|
std::shared_ptr<SfxRequest> pRequest;
|
|
if (pReq)
|
|
{
|
|
pRequest = std::make_shared<SfxRequest>(*pReq);
|
|
pReq->Ignore(); // the 'old' request is not relevant any more
|
|
}
|
|
pDlg->StartExecuteAsync([pDlg, &rWrtSh, pCoreSet=std::move(pCoreSet), bSel,
|
|
bSelectionPut, bApplyToParagraph,
|
|
pRequest=std::move(pRequest)](sal_Int32 nResult){
|
|
if (nResult == RET_OK)
|
|
{
|
|
sw_CharDialogResult(pDlg->GetOutputItemSet(), rWrtSh, pCoreSet, bSel, bSelectionPut,
|
|
bApplyToParagraph, pRequest.get());
|
|
}
|
|
pDlg->disposeOnce();
|
|
});
|
|
}
|
|
else if (pArgs)
|
|
{
|
|
sw_CharDialogResult(pArgs, rWrtSh, pCoreSet, bSel, bSelectionPut, bApplyToParagraph, pReq);
|
|
}
|
|
}
|
|
|
|
static void sw_CharDialogResult(const SfxItemSet* pSet, SwWrtShell& rWrtSh, std::shared_ptr<SfxItemSet> const & pCoreSet, bool bSel,
|
|
bool bSelectionPut, bool bApplyToParagraph, SfxRequest* pReq)
|
|
{
|
|
SfxItemSet aTmpSet( *pSet );
|
|
::ConvertAttrGenToChar(aTmpSet, *pCoreSet);
|
|
|
|
const bool bWasLocked = rWrtSh.IsViewLocked();
|
|
if (bApplyToParagraph)
|
|
{
|
|
rWrtSh.StartAction();
|
|
rWrtSh.LockView(true);
|
|
rWrtSh.Push();
|
|
SwLangHelper::SelectCurrentPara(rWrtSh);
|
|
}
|
|
|
|
const SfxStringItem* pSelectionItem;
|
|
bool bInsert = false;
|
|
sal_Int32 nInsert = 0;
|
|
|
|
// The old item is for unknown reasons back in the set again.
|
|
if( !bSelectionPut && (pSelectionItem = aTmpSet.GetItemIfSet(FN_PARAM_SELECTION, false)) )
|
|
{
|
|
const OUString& sInsert = pSelectionItem->GetValue();
|
|
bInsert = !sInsert.isEmpty();
|
|
if(bInsert)
|
|
{
|
|
nInsert = sInsert.getLength();
|
|
rWrtSh.StartAction();
|
|
rWrtSh.Insert( sInsert );
|
|
rWrtSh.SetMark();
|
|
rWrtSh.ExtendSelection(false, sInsert.getLength());
|
|
SfxRequest aReq(rWrtSh.GetView().GetViewFrame(), FN_INSERT_STRING);
|
|
aReq.AppendItem( SfxStringItem( FN_INSERT_STRING, sInsert ) );
|
|
aReq.Done();
|
|
SfxRequest aReq1(rWrtSh.GetView().GetViewFrame(), FN_CHAR_LEFT);
|
|
aReq1.AppendItem( SfxInt32Item(FN_PARAM_MOVE_COUNT, nInsert) );
|
|
aReq1.AppendItem( SfxBoolItem(FN_PARAM_MOVE_SELECTION, true) );
|
|
aReq1.Done();
|
|
}
|
|
}
|
|
aTmpSet.ClearItem(FN_PARAM_SELECTION);
|
|
|
|
SwTextFormatColl* pColl = rWrtSh.GetCurTextFormatColl();
|
|
if(bSel && rWrtSh.IsSelFullPara() && pColl && pColl->IsAutoUpdateOnDirectFormat())
|
|
{
|
|
rWrtSh.AutoUpdatePara(pColl, aTmpSet);
|
|
}
|
|
else
|
|
rWrtSh.SetAttrSet( aTmpSet );
|
|
if (pReq)
|
|
pReq->Done(aTmpSet);
|
|
if(bInsert)
|
|
{
|
|
SfxRequest aReq1(rWrtSh.GetView().GetViewFrame(), FN_CHAR_RIGHT);
|
|
aReq1.AppendItem( SfxInt32Item(FN_PARAM_MOVE_COUNT, nInsert) );
|
|
aReq1.AppendItem( SfxBoolItem(FN_PARAM_MOVE_SELECTION, false) );
|
|
aReq1.Done();
|
|
rWrtSh.SwapPam();
|
|
rWrtSh.ClearMark();
|
|
rWrtSh.DontExpandFormat();
|
|
rWrtSh.EndAction();
|
|
}
|
|
|
|
if (bApplyToParagraph)
|
|
{
|
|
rWrtSh.Pop(SwCursorShell::PopMode::DeleteCurrent);
|
|
rWrtSh.LockView(bWasLocked);
|
|
rWrtSh.EndAction();
|
|
}
|
|
}
|
|
|
|
|
|
static void sw_ParagraphDialogResult(SfxItemSet* pSet, SwWrtShell &rWrtSh, SfxRequest& rReq, SwPaM* pPaM)
|
|
{
|
|
if (!pSet)
|
|
return;
|
|
|
|
rReq.Done( *pSet );
|
|
::SfxToSwPageDescAttr( rWrtSh, *pSet );
|
|
// #i56253#
|
|
// enclose all undos.
|
|
// Thus, check conditions, if actions will be performed.
|
|
const bool bUndoNeeded( pSet->Count() ||
|
|
SfxItemState::SET == pSet->GetItemState(FN_NUMBER_NEWSTART) ||
|
|
SfxItemState::SET == pSet->GetItemState(FN_NUMBER_NEWSTART_AT) );
|
|
if ( bUndoNeeded )
|
|
{
|
|
rWrtSh.StartUndo( SwUndoId::INSATTR );
|
|
}
|
|
if( pSet->Count() )
|
|
{
|
|
rWrtSh.StartAction();
|
|
if ( const SfxStringItem* pDropTextItem = pSet->GetItemIfSet(FN_DROP_TEXT, false) )
|
|
{
|
|
if ( !pDropTextItem->GetValue().isEmpty() )
|
|
rWrtSh.ReplaceDropText(pDropTextItem->GetValue(), pPaM);
|
|
}
|
|
rWrtSh.SetAttrSet(*pSet, SetAttrMode::DEFAULT, pPaM);
|
|
rWrtSh.EndAction();
|
|
SwTextFormatColl* pColl = rWrtSh.GetPaMTextFormatColl(pPaM);
|
|
if(pColl && pColl->IsAutoUpdateOnDirectFormat())
|
|
{
|
|
rWrtSh.AutoUpdatePara(pColl, *pSet, pPaM);
|
|
}
|
|
}
|
|
|
|
if( SfxItemState::SET == pSet->GetItemState(FN_NUMBER_NEWSTART) )
|
|
{
|
|
//SetNumRuleStart(true) restarts the numbering at the value
|
|
//that is defined at the starting point of the numbering level
|
|
//otherwise the SetNodeNumStart() value determines the start
|
|
//if it's set to something different than USHRT_MAX
|
|
|
|
bool bStart = static_cast<const SfxBoolItem&>(pSet->Get(FN_NUMBER_NEWSTART)).GetValue();
|
|
|
|
// Default value for restart value has to be USHRT_MAX
|
|
// in order to indicate that the restart value of the list
|
|
// style has to be used on restart.
|
|
sal_uInt16 nNumStart = USHRT_MAX;
|
|
if( SfxItemState::SET == pSet->GetItemState(FN_NUMBER_NEWSTART_AT) )
|
|
{
|
|
nNumStart = pSet->Get(FN_NUMBER_NEWSTART_AT).GetValue();
|
|
}
|
|
rWrtSh.SetNumRuleStart(bStart, pPaM);
|
|
rWrtSh.SetNodeNumStart(nNumStart);
|
|
}
|
|
else if( SfxItemState::SET == pSet->GetItemState(FN_NUMBER_NEWSTART_AT) )
|
|
{
|
|
rWrtSh.SetNodeNumStart(pSet->Get(FN_NUMBER_NEWSTART_AT).GetValue());
|
|
rWrtSh.SetNumRuleStart(false, pPaM);
|
|
}
|
|
// #i56253#
|
|
if ( bUndoNeeded )
|
|
{
|
|
rWrtSh.EndUndo( SwUndoId::INSATTR );
|
|
}
|
|
}
|
|
|
|
namespace {
|
|
|
|
void InsertBreak(SwWrtShell& rWrtSh,
|
|
sal_uInt16 nKind,
|
|
::std::optional<sal_uInt16> oPageNumber,
|
|
const OUString& rTemplateName, std::optional<SwLineBreakClear> oClear)
|
|
{
|
|
switch ( nKind )
|
|
{
|
|
case 1 :
|
|
rWrtSh.InsertLineBreak(oClear);
|
|
break;
|
|
case 2 :
|
|
rWrtSh.InsertColumnBreak(); break;
|
|
case 3 :
|
|
{
|
|
rWrtSh.StartAllAction();
|
|
if( !rTemplateName.isEmpty() )
|
|
rWrtSh.InsertPageBreak( &rTemplateName, oPageNumber );
|
|
else
|
|
rWrtSh.InsertPageBreak();
|
|
rWrtSh.EndAllAction();
|
|
}
|
|
}
|
|
}
|
|
|
|
OUString GetLocalURL(const SwWrtShell& rSh)
|
|
{
|
|
SwField* pField = rSh.GetCurField();
|
|
if (!pField)
|
|
{
|
|
return OUString();
|
|
}
|
|
|
|
if (pField->GetTyp()->Which() != SwFieldIds::TableOfAuthorities)
|
|
{
|
|
return OUString();
|
|
}
|
|
|
|
const auto& rAuthorityField = *static_cast<const SwAuthorityField*>(pField);
|
|
SwAuthEntry* pAuthEntry = rAuthorityField.GetAuthEntry();
|
|
if (!pAuthEntry)
|
|
{
|
|
return OUString();
|
|
}
|
|
|
|
const OUString& rLocalURL = pAuthEntry->GetAuthorField(AUTH_FIELD_LOCAL_URL);
|
|
return rLocalURL;
|
|
}
|
|
|
|
void UpdateSections(SfxRequest& rReq, SwWrtShell& rWrtSh)
|
|
{
|
|
OUString aSectionNamePrefix;
|
|
const SfxStringItem* pSectionNamePrefix = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pSectionNamePrefix)
|
|
{
|
|
aSectionNamePrefix = pSectionNamePrefix->GetValue();
|
|
}
|
|
|
|
uno::Sequence<beans::PropertyValues> aSections;
|
|
const SfxUnoAnyItem* pSections = rReq.GetArg<SfxUnoAnyItem>(FN_PARAM_2);
|
|
if (pSections)
|
|
{
|
|
pSections->GetValue() >>= aSections;
|
|
}
|
|
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::UPDATE_SECTIONS, nullptr);
|
|
rWrtSh.StartAction();
|
|
|
|
SwDoc* pDoc = rWrtSh.GetDoc();
|
|
sal_Int32 nSectionIndex = 0;
|
|
const SwSectionFormats& rFormats = pDoc->GetSections();
|
|
IDocumentContentOperations& rIDCO = pDoc->getIDocumentContentOperations();
|
|
for (size_t i = 0; i < rFormats.size(); ++i)
|
|
{
|
|
const SwSectionFormat* pFormat = rFormats[i];
|
|
if (!pFormat->GetName().startsWith(aSectionNamePrefix))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (nSectionIndex >= aSections.getLength())
|
|
{
|
|
break;
|
|
}
|
|
|
|
comphelper::SequenceAsHashMap aMap(aSections[nSectionIndex++]);
|
|
OUString aSectionName = aMap[u"RegionName"_ustr].get<OUString>();
|
|
if (aSectionName != pFormat->GetName())
|
|
{
|
|
const_cast<SwSectionFormat*>(pFormat)->SetFormatName(aSectionName, /*bBroadcast=*/true);
|
|
SwSectionData aSectionData(*pFormat->GetSection());
|
|
aSectionData.SetSectionName(aSectionName);
|
|
pDoc->UpdateSection(i, aSectionData);
|
|
}
|
|
|
|
const SwFormatContent& rContent = pFormat->GetContent();
|
|
const SwNodeIndex* pContentNodeIndex = rContent.GetContentIdx();
|
|
if (pContentNodeIndex)
|
|
{
|
|
SwPaM aSectionStart(SwPosition{*pContentNodeIndex});
|
|
aSectionStart.Move(fnMoveForward, GoInContent);
|
|
SwPaM* pCursorPos = rWrtSh.GetCursor();
|
|
*pCursorPos = aSectionStart;
|
|
rWrtSh.EndOfSection(/*bSelect=*/true);
|
|
rIDCO.DeleteAndJoin(*pCursorPos);
|
|
rWrtSh.EndSelect();
|
|
|
|
OUString aSectionText = aMap[u"Content"_ustr].get<OUString>();
|
|
SwTranslateHelper::PasteHTMLToPaM(rWrtSh, pCursorPos, aSectionText.toUtf8());
|
|
}
|
|
}
|
|
|
|
rWrtSh.EndAction();
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::UPDATE_SECTIONS, nullptr);
|
|
}
|
|
|
|
void DeleteSections(SfxRequest& rReq, SwWrtShell& rWrtSh)
|
|
{
|
|
OUString aSectionNamePrefix;
|
|
const SfxStringItem* pSectionNamePrefix = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pSectionNamePrefix)
|
|
{
|
|
aSectionNamePrefix = pSectionNamePrefix->GetValue();
|
|
}
|
|
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::DELETE_SECTIONS, nullptr);
|
|
rWrtSh.StartAction();
|
|
comphelper::ScopeGuard g(
|
|
[&rWrtSh]
|
|
{
|
|
rWrtSh.EndAction();
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::DELETE_SECTIONS, nullptr);
|
|
});
|
|
|
|
SwDoc* pDoc = rWrtSh.GetDoc();
|
|
std::vector<SwSectionFormat*> aRemovals;
|
|
for (SwSectionFormat* pFormat : pDoc->GetSections())
|
|
if (aSectionNamePrefix.isEmpty() || pFormat->GetName().startsWith(aSectionNamePrefix))
|
|
aRemovals.push_back(pFormat);
|
|
|
|
for (const auto& pFormat : aRemovals)
|
|
{
|
|
// Just delete the format, not the content of the section.
|
|
pDoc->DelSectionFormat(pFormat);
|
|
}
|
|
}
|
|
|
|
void DeleteContentControl( SwWrtShell& rWrtSh )
|
|
{
|
|
SwTextContentControl* pTextContentControl = rWrtSh.CursorInsideContentControl();
|
|
if (pTextContentControl) {
|
|
const SwFormatContentControl& rFormatContentControl = pTextContentControl->GetContentControl();
|
|
const std::shared_ptr<SwContentControl>& pContentControl = rFormatContentControl.GetContentControl();
|
|
pContentControl->SetReadWrite(true);
|
|
pTextContentControl->Delete(true);
|
|
}
|
|
}
|
|
|
|
|
|
void UpdateBookmarks(SfxRequest& rReq, SwWrtShell& rWrtSh)
|
|
{
|
|
if (rWrtSh.getIDocumentSettingAccess().get(DocumentSettingId::PROTECT_BOOKMARKS))
|
|
{
|
|
return;
|
|
}
|
|
|
|
OUString aBookmarkNamePrefix;
|
|
const SfxStringItem* pBookmarkNamePrefix = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pBookmarkNamePrefix)
|
|
{
|
|
aBookmarkNamePrefix = pBookmarkNamePrefix->GetValue();
|
|
}
|
|
|
|
uno::Sequence<beans::PropertyValues> aBookmarks;
|
|
const SfxUnoAnyItem* pBookmarks = rReq.GetArg<SfxUnoAnyItem>(FN_PARAM_2);
|
|
if (pBookmarks)
|
|
{
|
|
pBookmarks->GetValue() >>= aBookmarks;
|
|
}
|
|
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::UPDATE_BOOKMARKS, nullptr);
|
|
rWrtSh.StartAction();
|
|
|
|
IDocumentMarkAccess& rIDMA = *rWrtSh.GetDoc()->getIDocumentMarkAccess();
|
|
sal_Int32 nBookmarkIndex = 0;
|
|
bool bSortMarks = false;
|
|
for (auto it = rIDMA.getBookmarksBegin(); it != rIDMA.getBookmarksEnd(); ++it)
|
|
{
|
|
sw::mark::Bookmark* pMark = *it;
|
|
assert(pMark);
|
|
if (!pMark->GetName().startsWith(aBookmarkNamePrefix))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (aBookmarks.getLength() <= nBookmarkIndex)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
comphelper::SequenceAsHashMap aMap(aBookmarks[nBookmarkIndex++]);
|
|
if (aMap[u"Bookmark"_ustr].get<OUString>() != pMark->GetName())
|
|
{
|
|
rIDMA.renameMark(pMark, aMap[u"Bookmark"_ustr].get<OUString>());
|
|
}
|
|
|
|
OUString aBookmarkText = aMap[u"BookmarkText"_ustr].get<OUString>();
|
|
|
|
// Insert markers to remember where the paste positions are.
|
|
SwPaM aMarkers(pMark->GetMarkEnd());
|
|
IDocumentContentOperations& rIDCO = rWrtSh.GetDoc()->getIDocumentContentOperations();
|
|
bool bSuccess = rIDCO.InsertString(aMarkers, u"XY"_ustr);
|
|
if (bSuccess)
|
|
{
|
|
SwPaM aPasteEnd(pMark->GetMarkEnd());
|
|
aPasteEnd.Move(fnMoveForward, GoInContent);
|
|
|
|
// Paste HTML content.
|
|
SwPaM* pCursorPos = rWrtSh.GetCursor();
|
|
*pCursorPos = aPasteEnd;
|
|
SwTranslateHelper::PasteHTMLToPaM(rWrtSh, pCursorPos, aBookmarkText.toUtf8());
|
|
|
|
// Update the bookmark to point to the new content.
|
|
SwPaM aPasteStart(pMark->GetMarkEnd());
|
|
aPasteStart.Move(fnMoveForward, GoInContent);
|
|
SwPaM aStartMarker(pMark->GetMarkStart(), *aPasteStart.GetPoint());
|
|
SwPaM aEndMarker(*aPasteEnd.GetPoint(), *aPasteEnd.GetPoint());
|
|
aEndMarker.GetMark()->AdjustContent(1);
|
|
pMark->SetMarkPos(*aPasteStart.GetPoint());
|
|
pMark->SetOtherMarkPos(*aPasteEnd.GetPoint());
|
|
bSortMarks = true;
|
|
|
|
// Remove markers. the start marker includes the old content as well.
|
|
rIDCO.DeleteAndJoin(aStartMarker);
|
|
rIDCO.DeleteAndJoin(aEndMarker);
|
|
}
|
|
}
|
|
if (bSortMarks)
|
|
{
|
|
rIDMA.assureSortedMarkContainers();
|
|
}
|
|
|
|
rWrtSh.EndAction();
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::UPDATE_BOOKMARKS, nullptr);
|
|
}
|
|
|
|
void UpdateBookmark(SfxRequest& rReq, SwWrtShell& rWrtSh)
|
|
{
|
|
if (rWrtSh.getIDocumentSettingAccess().get(DocumentSettingId::PROTECT_BOOKMARKS))
|
|
{
|
|
return;
|
|
}
|
|
|
|
OUString aBookmarkNamePrefix;
|
|
const SfxStringItem* pBookmarkNamePrefix = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pBookmarkNamePrefix)
|
|
{
|
|
aBookmarkNamePrefix = pBookmarkNamePrefix->GetValue();
|
|
}
|
|
|
|
uno::Sequence<beans::PropertyValue> aBookmark;
|
|
const SfxUnoAnyItem* pBookmarks = rReq.GetArg<SfxUnoAnyItem>(FN_PARAM_2);
|
|
if (pBookmarks)
|
|
{
|
|
pBookmarks->GetValue() >>= aBookmark;
|
|
}
|
|
|
|
IDocumentMarkAccess& rIDMA = *rWrtSh.GetDoc()->getIDocumentMarkAccess();
|
|
SwPosition& rCursor = *rWrtSh.GetCursor()->GetPoint();
|
|
sw::mark::Bookmark* pBookmark = rIDMA.getOneInnermostBookmarkFor(rCursor);
|
|
if (!pBookmark || !pBookmark->GetName().startsWith(aBookmarkNamePrefix))
|
|
{
|
|
return;
|
|
}
|
|
|
|
SwRewriter aRewriter;
|
|
aRewriter.AddRule(UndoArg1, pBookmark->GetName());
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::UPDATE_BOOKMARK, &aRewriter);
|
|
rWrtSh.StartAction();
|
|
comphelper::ScopeGuard g(
|
|
[&rWrtSh, &aRewriter]
|
|
{
|
|
rWrtSh.EndAction();
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::UPDATE_BOOKMARK, &aRewriter);
|
|
});
|
|
|
|
|
|
comphelper::SequenceAsHashMap aMap(aBookmark);
|
|
if (aMap[u"Bookmark"_ustr].get<OUString>() != pBookmark->GetName())
|
|
{
|
|
rIDMA.renameMark(pBookmark, aMap[u"Bookmark"_ustr].get<OUString>());
|
|
}
|
|
|
|
// Insert markers to remember where the paste positions are.
|
|
SwPaM aMarkers(pBookmark->GetMarkEnd());
|
|
IDocumentContentOperations& rIDCO = rWrtSh.GetDoc()->getIDocumentContentOperations();
|
|
if (!rIDCO.InsertString(aMarkers, u"XY"_ustr))
|
|
{
|
|
return;
|
|
}
|
|
|
|
SwPaM aPasteEnd(pBookmark->GetMarkEnd());
|
|
aPasteEnd.Move(fnMoveForward, GoInContent);
|
|
|
|
OUString aBookmarkText = aMap[u"BookmarkText"_ustr].get<OUString>();
|
|
|
|
// Paste HTML content.
|
|
SwPaM* pCursorPos = rWrtSh.GetCursor();
|
|
*pCursorPos = aPasteEnd;
|
|
SwTranslateHelper::PasteHTMLToPaM(rWrtSh, pCursorPos, aBookmarkText.toUtf8());
|
|
|
|
// Update the bookmark to point to the new content.
|
|
SwPaM aPasteStart(pBookmark->GetMarkEnd());
|
|
aPasteStart.Move(fnMoveForward, GoInContent);
|
|
SwPaM aStartMarker(pBookmark->GetMarkStart(), *aPasteStart.GetPoint());
|
|
SwPaM aEndMarker(*aPasteEnd.GetPoint(), *aPasteEnd.GetPoint());
|
|
aEndMarker.GetMark()->AdjustContent(1);
|
|
pBookmark->SetMarkPos(*aPasteStart.GetPoint());
|
|
pBookmark->SetOtherMarkPos(*aPasteEnd.GetPoint());
|
|
|
|
// Remove markers. the start marker includes the old content as well.
|
|
rIDCO.DeleteAndJoin(aStartMarker);
|
|
rIDCO.DeleteAndJoin(aEndMarker);
|
|
rIDMA.assureSortedMarkContainers();
|
|
}
|
|
|
|
void DeleteBookmarks(SfxRequest& rReq, SwWrtShell& rWrtSh)
|
|
{
|
|
if (rWrtSh.getIDocumentSettingAccess().get(DocumentSettingId::PROTECT_BOOKMARKS))
|
|
{
|
|
return;
|
|
}
|
|
|
|
OUString aBookmarkNamePrefix;
|
|
const SfxStringItem* pBookmarkNamePrefix = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pBookmarkNamePrefix)
|
|
{
|
|
aBookmarkNamePrefix = pBookmarkNamePrefix->GetValue();
|
|
}
|
|
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().StartUndo(SwUndoId::DELETE_BOOKMARKS, nullptr);
|
|
rWrtSh.StartAction();
|
|
comphelper::ScopeGuard g(
|
|
[&rWrtSh]
|
|
{
|
|
rWrtSh.EndAction();
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::DELETE_BOOKMARKS, nullptr);
|
|
});
|
|
|
|
IDocumentMarkAccess* pMarkAccess = rWrtSh.GetDoc()->getIDocumentMarkAccess();
|
|
std::vector<sw::mark::MarkBase*> aRemovals;
|
|
for (auto it = pMarkAccess->getBookmarksBegin(); it != pMarkAccess->getBookmarksEnd(); ++it)
|
|
{
|
|
sw::mark::Bookmark* pBookmark = *it;
|
|
assert(pBookmark);
|
|
|
|
if (!aBookmarkNamePrefix.isEmpty())
|
|
{
|
|
if (!pBookmark->GetName().startsWith(aBookmarkNamePrefix))
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
|
|
aRemovals.push_back(pBookmark);
|
|
}
|
|
|
|
for (const auto& pMark : aRemovals)
|
|
{
|
|
pMarkAccess->deleteMark(pMark);
|
|
}
|
|
}
|
|
|
|
void DeleteFields(SfxRequest& rReq, SwWrtShell& rWrtSh)
|
|
{
|
|
const SfxStringItem* pTypeName = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (!pTypeName || pTypeName->GetValue() != "SetRef")
|
|
{
|
|
// This is implemented so far only for reference marks.
|
|
return;
|
|
}
|
|
|
|
OUString aNamePrefix;
|
|
const SfxStringItem* pNamePrefix = rReq.GetArg<SfxStringItem>(FN_PARAM_2);
|
|
if (pNamePrefix)
|
|
{
|
|
aNamePrefix = pNamePrefix->GetValue();
|
|
}
|
|
|
|
SwDoc* pDoc = rWrtSh.GetDoc();
|
|
pDoc->GetIDocumentUndoRedo().StartUndo(SwUndoId::DELETE_FIELDS, nullptr);
|
|
rWrtSh.StartAction();
|
|
comphelper::ScopeGuard g(
|
|
[&rWrtSh]
|
|
{
|
|
rWrtSh.EndAction();
|
|
rWrtSh.GetDoc()->GetIDocumentUndoRedo().EndUndo(SwUndoId::DELETE_FIELDS, nullptr);
|
|
});
|
|
|
|
std::vector<const SwFormatRefMark*> aRemovals;
|
|
for (sal_uInt16 i = 0; i < pDoc->GetRefMarks(); ++i)
|
|
{
|
|
const SwFormatRefMark* pRefMark = pDoc->GetRefMark(i);
|
|
if (!aNamePrefix.isEmpty())
|
|
{
|
|
if (!pRefMark->GetRefName().startsWith(aNamePrefix))
|
|
{
|
|
continue;
|
|
}
|
|
}
|
|
|
|
aRemovals.push_back(pRefMark);
|
|
}
|
|
|
|
for (const auto& pMark : aRemovals)
|
|
{
|
|
pDoc->DeleteFormatRefMark(pMark);
|
|
}
|
|
}
|
|
|
|
void lcl_LogWarning(std::string sWarning)
|
|
{
|
|
LOK_WARN("sw.transform", sWarning);
|
|
}
|
|
|
|
bool lcl_ChangeChartColumnCount(const uno::Reference<chart2::XChartDocument>& xChartDoc, sal_Int32 nId,
|
|
bool bInsert, bool bResize = false)
|
|
{
|
|
uno::Reference<chart2::XDiagram> xDiagram = xChartDoc->getFirstDiagram();
|
|
if (!xDiagram.is())
|
|
return false;
|
|
uno::Reference<chart2::XCoordinateSystemContainer> xCooSysContainer(xDiagram, uno::UNO_QUERY);
|
|
if (!xCooSysContainer.is())
|
|
return false;
|
|
uno::Sequence<uno::Reference<chart2::XCoordinateSystem>> xCooSysSequence(
|
|
xCooSysContainer->getCoordinateSystems());
|
|
if (xCooSysSequence.getLength() <= 0)
|
|
return false;
|
|
uno::Reference<chart2::XChartTypeContainer> xChartTypeContainer(xCooSysSequence[0],
|
|
uno::UNO_QUERY);
|
|
if (!xChartTypeContainer.is())
|
|
return false;
|
|
uno::Sequence<uno::Reference<chart2::XChartType>> xChartTypeSequence(
|
|
xChartTypeContainer->getChartTypes());
|
|
if (xChartTypeSequence.getLength() <= 0)
|
|
return false;
|
|
uno::Reference<chart2::XDataSeriesContainer> xDSContainer(xChartTypeSequence[0],
|
|
uno::UNO_QUERY);
|
|
if (!xDSContainer.is())
|
|
return false;
|
|
|
|
uno::Reference<chart2::XInternalDataProvider> xIDataProvider(xChartDoc->getDataProvider(),
|
|
uno::UNO_QUERY);
|
|
if (!xIDataProvider.is())
|
|
return false;
|
|
|
|
uno::Sequence<uno::Reference<chart2::XDataSeries>> aSeriesSeq(xDSContainer->getDataSeries());
|
|
|
|
int nSeriesCount = aSeriesSeq.getLength();
|
|
|
|
if (bResize)
|
|
{
|
|
// Resize is actually some inserts, or deletes
|
|
if (nId > nSeriesCount)
|
|
{
|
|
bInsert = true;
|
|
}
|
|
else if (nId < nSeriesCount)
|
|
{
|
|
bInsert = false;
|
|
}
|
|
else
|
|
{
|
|
// Resize to the same size. No change needed
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// insert or delete
|
|
if (bInsert)
|
|
{
|
|
// insert
|
|
if (nId > nSeriesCount && !bResize)
|
|
return false;
|
|
|
|
int nInsertCount = bResize ? nId - nSeriesCount : 1;
|
|
|
|
// call dialog code
|
|
if (bResize)
|
|
{
|
|
for (int i = 0; i < nInsertCount; i++)
|
|
{
|
|
xIDataProvider->insertDataSeries(nSeriesCount);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
xIDataProvider->insertDataSeries(nId);
|
|
}
|
|
else
|
|
{
|
|
// delete 1 or more columns
|
|
if (nId >= nSeriesCount)
|
|
return false;
|
|
int nDeleteCount = bResize ? nSeriesCount - nId : 1;
|
|
for (int i = 0; i < nDeleteCount; i++)
|
|
{
|
|
xDSContainer->removeDataSeries(aSeriesSeq[nId]);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool lcl_ResizeChartColumns(const uno::Reference<chart2::XChartDocument>& xChartDoc, sal_Int32 nSize)
|
|
{
|
|
return lcl_ChangeChartColumnCount(xChartDoc, nSize, false, true);
|
|
}
|
|
|
|
bool lcl_InsertChartColumns(const uno::Reference<chart2::XChartDocument>& xChartDoc, sal_Int32 nId)
|
|
{
|
|
return lcl_ChangeChartColumnCount(xChartDoc, nId, true);
|
|
}
|
|
|
|
bool lcl_DeleteChartColumns(const uno::Reference<chart2::XChartDocument>& xChartDoc, sal_Int32 nId)
|
|
{
|
|
return lcl_ChangeChartColumnCount(xChartDoc, nId, false);
|
|
}
|
|
}
|
|
|
|
void SwTextShell::Execute(SfxRequest &rReq)
|
|
{
|
|
bool bUseDialog = true;
|
|
const SfxItemSet *pArgs = rReq.GetArgs();
|
|
SwWrtShell& rWrtSh = GetShell();
|
|
const SfxPoolItem* pItem = nullptr;
|
|
const sal_uInt16 nSlot = rReq.GetSlot();
|
|
if(pArgs)
|
|
pArgs->GetItemState(GetPool().GetWhichIDFromSlotID(nSlot), false, &pItem);
|
|
switch( nSlot )
|
|
{
|
|
case SID_UNICODE_NOTATION_TOGGLE:
|
|
{
|
|
tools::Long nMaxUnits = 256;
|
|
sal_Int32 nSelLength = rWrtSh.GetSelText().getLength();
|
|
if( rWrtSh.IsSelection() && !rWrtSh.IsMultiSelection() && (nSelLength < nMaxUnits) )
|
|
nMaxUnits = nSelLength;
|
|
|
|
tools::Long index = 0;
|
|
ToggleUnicodeCodepoint aToggle;
|
|
while( nMaxUnits-- && aToggle.AllowMoreInput(rWrtSh.GetChar(true, index-1)) )
|
|
--index;
|
|
|
|
OUString sReplacement = aToggle.ReplacementString();
|
|
if( !sReplacement.isEmpty() )
|
|
{
|
|
if (rWrtSh.HasReadonlySel() && !rWrtSh.CursorInsideInputField())
|
|
{
|
|
// Only break if there's something to do; don't nag with the dialog otherwise
|
|
rWrtSh.InfoReadOnlyDialog(false);
|
|
break;
|
|
}
|
|
OUString stringToReplace = aToggle.StringToReplace();
|
|
SwRewriter aRewriter;
|
|
aRewriter.AddRule( UndoArg1, stringToReplace );
|
|
aRewriter.AddRule( UndoArg2, SwResId(STR_YIELDS) );
|
|
aRewriter.AddRule( UndoArg3, sReplacement );
|
|
rWrtSh.StartUndo(SwUndoId::REPLACE, &aRewriter);
|
|
rWrtSh.GetCursor()->Normalize(false);
|
|
|
|
rWrtSh.ClearMark();
|
|
if( rWrtSh.IsInSelect() ) // cancel any in-progress keyboard selection as well
|
|
rWrtSh.EndSelect();
|
|
// Select exactly what was chosen for replacement
|
|
rWrtSh.GetCursor()->SetMark();
|
|
rWrtSh.GetCursor()->GetPoint()->AdjustContent(-stringToReplace.getLength());
|
|
rWrtSh.DelLeft();
|
|
rWrtSh.Insert2( sReplacement );
|
|
rWrtSh.EndUndo(SwUndoId::REPLACE, &aRewriter);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SID_LANGUAGE_STATUS:
|
|
{
|
|
// get the language
|
|
OUString aNewLangText;
|
|
const SfxStringItem* pItem2 = rReq.GetArg<SfxStringItem>(SID_LANGUAGE_STATUS);
|
|
if (pItem2)
|
|
aNewLangText = pItem2->GetValue();
|
|
|
|
//!! Remember the view frame right now...
|
|
//!! (call to GetView().GetViewFrame() will break if the
|
|
//!! SwTextShell got destroyed meanwhile.)
|
|
SfxViewFrame& rViewFrame = GetView().GetViewFrame();
|
|
|
|
if (aNewLangText == "*")
|
|
{
|
|
// open the dialog "Tools/Options/Languages and Locales - General"
|
|
// to set the documents default language
|
|
SfxAbstractDialogFactory* pFact = SfxAbstractDialogFactory::Create();
|
|
ScopedVclPtr<VclAbstractDialog> pDlg(pFact->CreateVclDialog(GetView().GetFrameWeld(), SID_LANGUAGE_OPTIONS));
|
|
pDlg->Execute();
|
|
}
|
|
else
|
|
{
|
|
//!! We have to use StartAction / EndAction bracketing in
|
|
//!! order to prevent possible destruction of the SwTextShell
|
|
//!! due to the selection changes coming below.
|
|
rWrtSh.StartAction();
|
|
// prevent view from jumping because of (temporary) selection changes
|
|
rWrtSh.LockView( true );
|
|
|
|
// setting the new language...
|
|
if (!aNewLangText.isEmpty())
|
|
{
|
|
static constexpr OUString aSelectionLangPrefix(u"Current_"_ustr);
|
|
static constexpr OUString aParagraphLangPrefix(u"Paragraph_"_ustr);
|
|
static constexpr OUString aDocumentLangPrefix(u"Default_"_ustr);
|
|
|
|
SfxItemSetFixed
|
|
<RES_CHRATR_LANGUAGE, RES_CHRATR_LANGUAGE,
|
|
RES_CHRATR_CJK_LANGUAGE, RES_CHRATR_CJK_LANGUAGE,
|
|
RES_CHRATR_CTL_LANGUAGE, RES_CHRATR_CTL_LANGUAGE> aCoreSet( GetPool() );
|
|
|
|
sal_Int32 nPos = 0;
|
|
bool bForSelection = true;
|
|
bool bForParagraph = false;
|
|
if (-1 != (nPos = aNewLangText.indexOf( aSelectionLangPrefix )))
|
|
{
|
|
// ... for the current selection
|
|
aNewLangText = aNewLangText.replaceAt(nPos, aSelectionLangPrefix.getLength(), u"");
|
|
bForSelection = true;
|
|
}
|
|
else if (-1 != (nPos = aNewLangText.indexOf(aParagraphLangPrefix)))
|
|
{
|
|
// ... for the current paragraph language
|
|
aNewLangText = aNewLangText.replaceAt(nPos, aParagraphLangPrefix.getLength(), u"");
|
|
bForSelection = true;
|
|
bForParagraph = true;
|
|
}
|
|
else if (-1 != (nPos = aNewLangText.indexOf(aDocumentLangPrefix)))
|
|
{
|
|
// ... as default document language
|
|
aNewLangText = aNewLangText.replaceAt(nPos, aDocumentLangPrefix.getLength(), u"");
|
|
bForSelection = false;
|
|
}
|
|
|
|
if (bForParagraph || !bForSelection)
|
|
{
|
|
rWrtSh.Push(); // save selection for later restoration
|
|
rWrtSh.ClearMark(); // fdo#67796: invalidate table crsr
|
|
}
|
|
|
|
if (bForParagraph)
|
|
SwLangHelper::SelectCurrentPara( rWrtSh );
|
|
|
|
if (!bForSelection) // document language to be changed...
|
|
{
|
|
rWrtSh.SelAll();
|
|
rWrtSh.ExtendedSelectAll();
|
|
}
|
|
|
|
rWrtSh.StartUndo( ( !bForParagraph && !bForSelection ) ? SwUndoId::SETDEFTATTR : SwUndoId::EMPTY );
|
|
if (aNewLangText == "LANGUAGE_NONE")
|
|
SwLangHelper::SetLanguage_None( rWrtSh, bForSelection, aCoreSet );
|
|
else if (aNewLangText == "RESET_LANGUAGES")
|
|
SwLangHelper::ResetLanguages( rWrtSh );
|
|
else
|
|
SwLangHelper::SetLanguage( rWrtSh, aNewLangText, bForSelection, aCoreSet );
|
|
rWrtSh.EndUndo();
|
|
|
|
if (bForParagraph || !bForSelection)
|
|
{
|
|
rWrtSh.Pop(SwCursorShell::PopMode::DeleteCurrent); // restore selection...
|
|
}
|
|
}
|
|
|
|
rWrtSh.LockView( false );
|
|
rWrtSh.EndAction();
|
|
}
|
|
|
|
// invalidate slot to get the new language displayed
|
|
rViewFrame.GetBindings().Invalidate( nSlot );
|
|
|
|
rReq.Done();
|
|
break;
|
|
}
|
|
|
|
case SID_THES:
|
|
{
|
|
// replace word/selection with text from selected sub menu entry
|
|
OUString aReplaceText;
|
|
const SfxStringItem* pItem2 = rReq.GetArg(FN_PARAM_THES_WORD_REPLACE);
|
|
if (pItem2)
|
|
aReplaceText = pItem2->GetValue();
|
|
if (!aReplaceText.isEmpty())
|
|
{
|
|
SwView &rView2 = rWrtSh.GetView();
|
|
const bool bSelection = rWrtSh.HasSelection();
|
|
const OUString aLookUpText = rView2.GetThesaurusLookUpText( bSelection );
|
|
rView2.InsertThesaurusSynonym( aReplaceText, aLookUpText, bSelection );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SID_CHARMAP:
|
|
{
|
|
InsertSymbol( rReq );
|
|
}
|
|
break;
|
|
case FN_INSERT_FOOTNOTE:
|
|
case FN_INSERT_ENDNOTE:
|
|
{
|
|
OUString aStr;
|
|
const SfxStringItem* pFont = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
const SfxStringItem* pNameItem = rReq.GetArg<SfxStringItem>(nSlot);
|
|
if ( pNameItem )
|
|
aStr = pNameItem->GetValue();
|
|
bool bFont = pFont && !pFont->GetValue().isEmpty();
|
|
rWrtSh.StartUndo( SwUndoId::UI_INSERT_FOOTNOTE );
|
|
rWrtSh.InsertFootnote( aStr, nSlot == FN_INSERT_ENDNOTE, !bFont );
|
|
if ( bFont )
|
|
{
|
|
rWrtSh.Left( SwCursorSkipMode::Chars, true, 1, false );
|
|
SfxItemSetFixed<RES_CHRATR_FONT, RES_CHRATR_FONT> aSet( rWrtSh.GetAttrPool() );
|
|
rWrtSh.GetCurAttr( aSet );
|
|
rWrtSh.SetAttrSet( aSet, SetAttrMode::DONTEXPAND );
|
|
rWrtSh.ResetSelect(nullptr, false);
|
|
rWrtSh.EndSelect();
|
|
rWrtSh.GotoFootnoteText();
|
|
}
|
|
rWrtSh.EndUndo( SwUndoId::UI_INSERT_FOOTNOTE );
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
case FN_INSERT_FOOTNOTE_DLG:
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
VclPtr<AbstractInsFootNoteDlg> pDlg(pFact->CreateInsFootNoteDlg(
|
|
GetView().GetFrameWeld(), rWrtSh));
|
|
pDlg->SetHelpId(GetStaticInterface()->GetSlot(nSlot)->GetCommand());
|
|
pDlg->StartExecuteAsync(
|
|
[this, pDlg] (sal_Int32 nResult)->void
|
|
{
|
|
if ( nResult == RET_OK )
|
|
{
|
|
pDlg->Apply();
|
|
const sal_uInt16 nId = pDlg->IsEndNote() ? FN_INSERT_ENDNOTE : FN_INSERT_FOOTNOTE;
|
|
SfxRequest aReq(GetView().GetViewFrame(), nId);
|
|
if ( !pDlg->GetStr().isEmpty() )
|
|
aReq.AppendItem( SfxStringItem( nId, pDlg->GetStr() ) );
|
|
if ( !pDlg->GetFontName().isEmpty() )
|
|
aReq.AppendItem( SfxStringItem( FN_PARAM_1, pDlg->GetFontName() ) );
|
|
ExecuteSlot( aReq );
|
|
}
|
|
pDlg->disposeOnce();
|
|
}
|
|
);
|
|
rReq.Ignore();
|
|
}
|
|
break;
|
|
case FN_FORMAT_FOOTNOTE_DLG:
|
|
case FN_FORMAT_CURRENT_FOOTNOTE_DLG:
|
|
{
|
|
GetView().ExecFormatFootnote();
|
|
break;
|
|
}
|
|
case SID_INSERTDOC:
|
|
{
|
|
GetView().ExecuteInsertDoc( rReq, pItem );
|
|
break;
|
|
}
|
|
case FN_FORMAT_RESET:
|
|
{
|
|
// #i78856, reset all attributes but not the language attributes
|
|
// (for this build an array of all relevant attributes and
|
|
// remove the languages from that)
|
|
o3tl::sorted_vector<sal_uInt16> aAttribs;
|
|
|
|
static constexpr std::pair<sal_uInt16, sal_uInt16> aResetableSetRange[] = {
|
|
// tdf#40496: we don't want to change writing direction, so exclude RES_FRAMEDIR:
|
|
{ RES_FRMATR_BEGIN, RES_FRAMEDIR - 1 },
|
|
{ RES_FRAMEDIR + 1, RES_FRMATR_END - 1 },
|
|
{ RES_CHRATR_BEGIN, RES_CHRATR_LANGUAGE - 1 },
|
|
{ RES_CHRATR_LANGUAGE + 1, RES_CHRATR_CJK_LANGUAGE - 1 },
|
|
{ RES_CHRATR_CJK_LANGUAGE + 1, RES_CHRATR_CTL_LANGUAGE - 1 },
|
|
{ RES_CHRATR_CTL_LANGUAGE + 1, RES_CHRATR_END - 1 },
|
|
{ RES_PARATR_BEGIN, RES_PARATR_END - 1 },
|
|
{ RES_PARATR_LIST_AUTOFMT, RES_PARATR_LIST_AUTOFMT },
|
|
{ RES_TXTATR_UNKNOWN_CONTAINER, RES_TXTATR_UNKNOWN_CONTAINER },
|
|
{ RES_UNKNOWNATR_BEGIN, RES_UNKNOWNATR_END - 1 },
|
|
};
|
|
for (const auto& [nBegin, nEnd] : aResetableSetRange)
|
|
{
|
|
for (sal_uInt16 i = nBegin; i <= nEnd; ++i)
|
|
aAttribs.insert( i );
|
|
}
|
|
|
|
// also clear the direct formatting flag inside SwTableBox(es)
|
|
if (SwFEShell* pFEShell = GetView().GetDocShell()->GetFEShell())
|
|
pFEShell->UpdateTableStyleFormatting(nullptr, true);
|
|
|
|
// tdf#160801 fix crash by delaying resetting of attributes
|
|
// Calling SwWrtShell::ResetAttr() will sometimes delete the
|
|
// current SwTextShell instance so call it after clearing the
|
|
// direct formatting flag.
|
|
rWrtSh.ResetAttr( aAttribs );
|
|
|
|
rReq.Done();
|
|
break;
|
|
}
|
|
case FN_INSERT_BREAK_DLG:
|
|
{
|
|
if ( pItem )
|
|
{
|
|
::std::optional<sal_uInt16> oPageNumber;
|
|
std::optional<SwLineBreakClear> oClear;
|
|
OUString aTemplateName;
|
|
sal_uInt16 nKind = static_cast<const SfxInt16Item*>(pItem)->GetValue();
|
|
const SfxStringItem* pTemplate = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
const SfxUInt16Item* pNumber = rReq.GetArg<SfxUInt16Item>(FN_PARAM_2);
|
|
const SfxBoolItem* pIsNumberFilled = rReq.GetArg<SfxBoolItem>(FN_PARAM_3);
|
|
if ( pTemplate )
|
|
aTemplateName = pTemplate->GetValue();
|
|
if ( pNumber && pIsNumberFilled && pIsNumberFilled->GetValue() )
|
|
oPageNumber = pNumber->GetValue();
|
|
|
|
InsertBreak(rWrtSh, nKind, oPageNumber, aTemplateName, oClear);
|
|
}
|
|
else
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
|
|
std::shared_ptr<AbstractSwBreakDlg> pAbstractDialog(pFact->CreateSwBreakDlg(GetView().GetFrameWeld(), rWrtSh));
|
|
std::shared_ptr<weld::DialogController> pDialogController(pAbstractDialog->getDialogController());
|
|
|
|
weld::DialogController::runAsync(pDialogController,
|
|
[pAbstractDialog=std::move(pAbstractDialog), &rWrtSh] (sal_Int32 nResult) {
|
|
if( RET_OK == nResult )
|
|
{
|
|
sal_uInt16 nKind = pAbstractDialog->GetKind();
|
|
OUString aTemplateName = pAbstractDialog->GetTemplateName();
|
|
::std::optional<sal_uInt16> oPageNumber = pAbstractDialog->GetPageNumber();
|
|
std::optional<SwLineBreakClear> oClear = pAbstractDialog->GetClear();
|
|
|
|
InsertBreak(rWrtSh, nKind, oPageNumber, aTemplateName, oClear);
|
|
}
|
|
});
|
|
}
|
|
|
|
break;
|
|
}
|
|
case FN_INSERT_BOOKMARK:
|
|
{
|
|
const SfxStringItem* pBookmarkText = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
SwPaM* pCursorPos = rWrtSh.GetCursor();
|
|
if ( pItem )
|
|
{
|
|
rWrtSh.StartAction();
|
|
OUString sName = static_cast<const SfxStringItem*>(pItem)->GetValue();
|
|
|
|
if (pBookmarkText)
|
|
{
|
|
OUString aBookmarkText = pBookmarkText->GetValue();
|
|
// Split node to remember where the start position is.
|
|
bool bSuccess = rWrtSh.GetDoc()->getIDocumentContentOperations().SplitNode(
|
|
*pCursorPos->GetPoint(), /*bChkTableStart=*/false);
|
|
if (bSuccess)
|
|
{
|
|
SwPaM aBookmarkPam(*pCursorPos->GetPoint());
|
|
aBookmarkPam.Move(fnMoveBackward, GoInContent);
|
|
|
|
// Paste HTML content.
|
|
SwTranslateHelper::PasteHTMLToPaM(
|
|
rWrtSh, pCursorPos, aBookmarkText.toUtf8());
|
|
if (pCursorPos->GetPoint()->GetContentIndex() == 0)
|
|
{
|
|
// The paste created a last empty text node, remove it.
|
|
SwPaM aPam(*pCursorPos->GetPoint());
|
|
aPam.SetMark();
|
|
aPam.Move(fnMoveBackward, GoInContent);
|
|
rWrtSh.GetDoc()->getIDocumentContentOperations().DeleteAndJoin(aPam);
|
|
}
|
|
|
|
// Undo the above SplitNode().
|
|
aBookmarkPam.SetMark();
|
|
aBookmarkPam.Move(fnMoveForward, GoInContent);
|
|
rWrtSh.GetDoc()->getIDocumentContentOperations().DeleteAndJoin(
|
|
aBookmarkPam);
|
|
*aBookmarkPam.GetMark() = *pCursorPos->GetPoint();
|
|
*pCursorPos = aBookmarkPam;
|
|
}
|
|
}
|
|
|
|
rWrtSh.SetBookmark( vcl::KeyCode(), sName );
|
|
if (pBookmarkText)
|
|
{
|
|
pCursorPos->DeleteMark();
|
|
}
|
|
rWrtSh.EndAction();
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
}
|
|
case FN_EDIT_BOOKMARK:
|
|
{
|
|
::std::optional<OUString> oName;
|
|
if (pItem)
|
|
{
|
|
oName.emplace(static_cast<const SfxStringItem*>(pItem)->GetValue());
|
|
}
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
ScopedVclPtr<VclAbstractDialog> pDlg(pFact->CreateSwInsertBookmarkDlg(GetView().GetFrameWeld(), rWrtSh, oName ? &*oName : nullptr));
|
|
VclAbstractDialog::AsyncContext aContext;
|
|
aContext.maEndDialogFn = [](sal_Int32){};
|
|
pDlg->StartExecuteAsync(aContext);
|
|
}
|
|
|
|
break;
|
|
}
|
|
case FN_UPDATE_BOOKMARKS:
|
|
{
|
|
// This updates all bookmarks in the document that match the conditions specified in
|
|
// rReq.
|
|
UpdateBookmarks(rReq, rWrtSh);
|
|
break;
|
|
}
|
|
case FN_UPDATE_BOOKMARK:
|
|
{
|
|
// This updates the bookmark under the cursor.
|
|
UpdateBookmark(rReq, rWrtSh);
|
|
break;
|
|
}
|
|
case FN_DELETE_BOOKMARK:
|
|
{
|
|
// This deletes a bookmark with the specified name.
|
|
if (pItem && !rWrtSh.getIDocumentSettingAccess().get(DocumentSettingId::PROTECT_BOOKMARKS))
|
|
{
|
|
IDocumentMarkAccess* const pMarkAccess = rWrtSh.getIDocumentMarkAccess();
|
|
pMarkAccess->deleteMark(pMarkAccess->findMark(static_cast<const SfxStringItem*>(pItem)->GetValue()), false);
|
|
}
|
|
break;
|
|
}
|
|
case FN_DELETE_BOOKMARKS:
|
|
{
|
|
// This deletes all bookmarks in the document matching a specified prefix.
|
|
DeleteBookmarks(rReq, rWrtSh);
|
|
break;
|
|
}
|
|
case FN_DELETE_FIELDS:
|
|
{
|
|
// This deletes all fields in the document matching a specified type & prefix.
|
|
DeleteFields(rReq, rWrtSh);
|
|
break;
|
|
}
|
|
case FN_UPDATE_SECTIONS:
|
|
{
|
|
UpdateSections(rReq, rWrtSh);
|
|
break;
|
|
}
|
|
case FN_DELETE_SECTIONS:
|
|
{
|
|
// This deletes all sections in the document matching a specified prefix. Note that the
|
|
// section is deleted, but not its contents.
|
|
DeleteSections(rReq, rWrtSh);
|
|
break;
|
|
}
|
|
case FN_DELETE_CONTENT_CONTROL:
|
|
{
|
|
DeleteContentControl( rWrtSh );
|
|
break;
|
|
}
|
|
case FN_SET_REMINDER:
|
|
{
|
|
// collect and sort navigator reminder names
|
|
IDocumentMarkAccess* const pMarkAccess = rWrtSh.getIDocumentMarkAccess();
|
|
std::vector< OUString > vNavMarkNames;
|
|
for(auto ppMark = pMarkAccess->getAllMarksBegin();
|
|
ppMark != pMarkAccess->getAllMarksEnd();
|
|
++ppMark)
|
|
{
|
|
if( IDocumentMarkAccess::GetType(**ppMark) == IDocumentMarkAccess::MarkType::NAVIGATOR_REMINDER )
|
|
vNavMarkNames.push_back((*ppMark)->GetName());
|
|
}
|
|
std::sort(vNavMarkNames.begin(), vNavMarkNames.end());
|
|
|
|
// we are maxed out so delete the first one
|
|
// this assumes that IDocumentMarkAccess generates Names in ascending order
|
|
if(vNavMarkNames.size() == MAX_MARKS)
|
|
pMarkAccess->deleteMark(pMarkAccess->findMark(vNavMarkNames[0]), false);
|
|
|
|
rWrtSh.SetBookmark(vcl::KeyCode(), OUString(), IDocumentMarkAccess::MarkType::NAVIGATOR_REMINDER);
|
|
SwView::SetActMark(vNavMarkNames.size() < MAX_MARKS ? vNavMarkNames.size() : MAX_MARKS-1);
|
|
|
|
break;
|
|
}
|
|
case FN_AUTOFORMAT_REDLINE_APPLY:
|
|
{
|
|
SvxSwAutoFormatFlags aFlags(SvxAutoCorrCfg::Get().GetAutoCorrect()->GetSwFlags());
|
|
// This must always be false for the postprocessing.
|
|
aFlags.bAFormatByInput = false;
|
|
aFlags.bWithRedlining = true;
|
|
rWrtSh.AutoFormat( &aFlags, false );
|
|
aFlags.bWithRedlining = false;
|
|
|
|
SfxViewFrame& rVFrame = GetView().GetViewFrame();
|
|
if (rVFrame.HasChildWindow(FN_REDLINE_ACCEPT))
|
|
rVFrame.ToggleChildWindow(FN_REDLINE_ACCEPT);
|
|
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
auto xRequest = std::make_shared<SfxRequest>(rReq);
|
|
rReq.Ignore(); // the 'old' request is not relevant any more
|
|
VclPtr<AbstractSwModalRedlineAcceptDlg> pDlg(pFact->CreateSwModalRedlineAcceptDlg(GetView().GetEditWin().GetFrameWeld()));
|
|
pDlg->StartExecuteAsync(
|
|
[pDlg, xRequest=std::move(xRequest)] (sal_Int32 /*nResult*/)->void
|
|
{
|
|
pDlg->disposeOnce();
|
|
xRequest->Done();
|
|
}
|
|
);
|
|
}
|
|
break;
|
|
|
|
case FN_AUTOFORMAT_APPLY:
|
|
{
|
|
SvxSwAutoFormatFlags aFlags(SvxAutoCorrCfg::Get().GetAutoCorrect()->GetSwFlags());
|
|
// This must always be false for the postprocessing.
|
|
aFlags.bAFormatByInput = false;
|
|
rWrtSh.AutoFormat( &aFlags, false );
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
case FN_AUTOFORMAT_AUTO:
|
|
{
|
|
SvxAutoCorrCfg& rACfg = SvxAutoCorrCfg::Get();
|
|
bool bSet = pItem ? static_cast<const SfxBoolItem*>(pItem)->GetValue() : !rACfg.IsAutoFormatByInput();
|
|
if( bSet != rACfg.IsAutoFormatByInput() )
|
|
{
|
|
rACfg.SetAutoFormatByInput( bSet );
|
|
rACfg.Commit();
|
|
GetView().GetViewFrame().GetBindings().Invalidate( nSlot );
|
|
if ( !pItem )
|
|
rReq.AppendItem( SfxBoolItem( GetPool().GetWhichIDFromSlotID(nSlot), bSet ) );
|
|
rReq.Done();
|
|
}
|
|
}
|
|
break;
|
|
case FN_AUTO_CORRECT:
|
|
{
|
|
// At first set to blank as default.
|
|
rWrtSh.AutoCorrect( *SvxAutoCorrCfg::Get().GetAutoCorrect(), ' ' );
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
case FN_TABLE_SORT_DIALOG:
|
|
case FN_SORTING_DLG:
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
VclPtr<AbstractSwSortDlg> pDlg(pFact->CreateSwSortingDialog(GetView().GetFrameWeld(), rWrtSh));
|
|
auto xRequest = std::make_shared<SfxRequest>(rReq);
|
|
rReq.Ignore(); // the 'old' request is not relevant any more
|
|
pDlg->StartExecuteAsync(
|
|
[pDlg, xRequest=std::move(xRequest)] (sal_Int32 nResult)->void
|
|
{
|
|
if (nResult == RET_OK)
|
|
pDlg->Apply();
|
|
pDlg->disposeOnce();
|
|
xRequest->Done();
|
|
}
|
|
);
|
|
}
|
|
break;
|
|
case FN_NUMBERING_OUTLINE_DLG:
|
|
{
|
|
GetView().ExecNumberingOutline(GetPool());
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
case FN_CALCULATE:
|
|
{
|
|
rtl::Reference<SwTransferable> pTransfer = new SwTransferable( rWrtSh );
|
|
pTransfer->CalculateAndCopy();
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
case FN_GOTO_REFERENCE:
|
|
{
|
|
SwField *pField = rWrtSh.GetCurField();
|
|
if(pField && pField->GetTypeId() == SwFieldTypesEnum::GetRef)
|
|
{
|
|
rWrtSh.StartAllAction();
|
|
rWrtSh.SwCursorShell::GotoRefMark( static_cast<SwGetRefField*>(pField)->GetSetRefName(),
|
|
static_cast<SwGetRefField*>(pField)->GetSubType(),
|
|
static_cast<SwGetRefField*>(pField)->GetSeqNo(),
|
|
static_cast<SwGetRefField*>(pField)->GetFlags() );
|
|
rWrtSh.EndAllAction();
|
|
rReq.Done();
|
|
}
|
|
}
|
|
break;
|
|
case FN_EDIT_FORMULA:
|
|
{
|
|
const sal_uInt16 nId = SwInputChild::GetChildWindowId();
|
|
SfxViewFrame& rVFrame = GetView().GetViewFrame();
|
|
if(pItem)
|
|
{
|
|
//if the ChildWindow is active it has to be removed
|
|
if( rVFrame.HasChildWindow( nId ) )
|
|
{
|
|
rVFrame.ToggleChildWindow( nId );
|
|
rVFrame.GetBindings().InvalidateAll( true );
|
|
}
|
|
|
|
OUString sFormula(static_cast<const SfxStringItem*>(pItem)->GetValue());
|
|
SwFieldMgr aFieldMgr;
|
|
rWrtSh.StartAllAction();
|
|
bool bDelSel = rWrtSh.HasSelection();
|
|
if( bDelSel )
|
|
{
|
|
rWrtSh.StartUndo( SwUndoId::START );
|
|
rWrtSh.DelRight();
|
|
}
|
|
else
|
|
{
|
|
rWrtSh.EnterStdMode();
|
|
}
|
|
|
|
if( !bDelSel && aFieldMgr.GetCurField() && SwFieldTypesEnum::Formel == aFieldMgr.GetCurTypeId() )
|
|
aFieldMgr.UpdateCurField( aFieldMgr.GetCurField()->GetFormat(), OUString(), sFormula );
|
|
else if( !sFormula.isEmpty() )
|
|
{
|
|
if( rWrtSh.IsCursorInTable() )
|
|
{
|
|
SfxItemSetFixed<RES_BOXATR_FORMULA, RES_BOXATR_FORMULA> aSet( rWrtSh.GetAttrPool() );
|
|
aSet.Put( SwTableBoxFormula( sFormula ));
|
|
rWrtSh.SetTableBoxFormulaAttrs( aSet );
|
|
rWrtSh.UpdateTable();
|
|
}
|
|
else
|
|
{
|
|
SvNumberFormatter* pFormatter = rWrtSh.GetNumberFormatter();
|
|
const sal_uInt32 nSysNumFormat = pFormatter->GetFormatIndex( NF_NUMBER_STANDARD, LANGUAGE_SYSTEM);
|
|
SwInsertField_Data aData(SwFieldTypesEnum::Formel, nsSwGetSetExpType::GSE_FORMULA, OUString(), sFormula, nSysNumFormat);
|
|
aFieldMgr.InsertField(aData);
|
|
}
|
|
}
|
|
|
|
if( bDelSel )
|
|
rWrtSh.EndUndo( SwUndoId::END );
|
|
rWrtSh.EndAllAction();
|
|
rReq.Done();
|
|
}
|
|
else
|
|
{
|
|
rWrtSh.EndAllTableBoxEdit();
|
|
rVFrame.ToggleChildWindow( nId );
|
|
if( !rVFrame.HasChildWindow( nId ) )
|
|
rVFrame.GetBindings().InvalidateAll( true );
|
|
rReq.Ignore();
|
|
}
|
|
}
|
|
|
|
break;
|
|
case FN_TABLE_UNSET_READ_ONLY:
|
|
{
|
|
rWrtSh.UnProtectTables();
|
|
}
|
|
break;
|
|
case SID_EDIT_HYPERLINK:
|
|
{
|
|
if (!rWrtSh.HasSelection())
|
|
{
|
|
SfxItemSetFixed<RES_TXTATR_INETFMT, RES_TXTATR_INETFMT> aSet(GetPool());
|
|
rWrtSh.GetCurAttr(aSet);
|
|
if (SfxItemState::SET > aSet.GetItemState(RES_TXTATR_INETFMT))
|
|
{
|
|
// Didn't find a hyperlink to edit yet.
|
|
|
|
// If the cursor is just before an unselected hyperlink,
|
|
// the dialog will not know that it should edit that hyperlink,
|
|
// so in this case, first select it so the dialog will find the hyperlink.
|
|
// The dialog would leave the hyperlink selected anyway after a successful edit
|
|
// (although it isn't normally selected after a cancel, but oh well).
|
|
if (!rWrtSh.SelectTextAttr(RES_TXTATR_INETFMT))
|
|
break;
|
|
}
|
|
}
|
|
|
|
GetView().GetViewFrame().SetChildWindow(SID_HYPERLINK_DIALOG, true);
|
|
}
|
|
break;
|
|
case SID_REMOVE_HYPERLINK:
|
|
{
|
|
bool bSel = rWrtSh.HasSelection();
|
|
if(!bSel)
|
|
{
|
|
rWrtSh.StartAction();
|
|
rWrtSh.Push();
|
|
if(!rWrtSh.SelectTextAttr( RES_TXTATR_INETFMT ))
|
|
rWrtSh.SelWrd();
|
|
}
|
|
//now remove the attribute
|
|
rWrtSh.ResetAttr({ RES_TXTATR_INETFMT });
|
|
if(!bSel)
|
|
{
|
|
rWrtSh.Pop(SwCursorShell::PopMode::DeleteCurrent);
|
|
rWrtSh.EndAction();
|
|
}
|
|
}
|
|
break;
|
|
case SID_ATTR_BRUSH_CHAR :
|
|
case SID_ATTR_CHAR_SCALEWIDTH :
|
|
case SID_ATTR_CHAR_ROTATED :
|
|
case FN_TXTATR_INET :
|
|
{
|
|
const sal_uInt16 nWhich = GetPool().GetWhichIDFromSlotID( nSlot );
|
|
if ( pArgs && pArgs->GetItemState( nWhich ) == SfxItemState::SET )
|
|
bUseDialog = false;
|
|
[[fallthrough]];
|
|
}
|
|
case SID_CHAR_DLG:
|
|
case SID_CHAR_DLG_EFFECT:
|
|
case SID_CHAR_DLG_POSITION:
|
|
{
|
|
sw_CharDialog(rWrtSh, bUseDialog, /*ApplyToParagraph*/false, nSlot, pArgs, &rReq);
|
|
}
|
|
break;
|
|
case SID_CHAR_DLG_FOR_PARAGRAPH:
|
|
{
|
|
sw_CharDialog(rWrtSh, /*UseDialog*/true, /*ApplyToParagraph*/true, nSlot, pArgs, &rReq);
|
|
}
|
|
break;
|
|
case SID_ATTR_LRSPACE :
|
|
case SID_ATTR_ULSPACE :
|
|
case SID_ATTR_BRUSH :
|
|
case SID_PARA_VERTALIGN :
|
|
case SID_ATTR_PARA_NUMRULE :
|
|
case SID_ATTR_PARA_REGISTER :
|
|
case SID_ATTR_PARA_PAGENUM :
|
|
case FN_FORMAT_LINENUMBER :
|
|
case FN_NUMBER_NEWSTART :
|
|
case FN_NUMBER_NEWSTART_AT :
|
|
case FN_FORMAT_DROPCAPS :
|
|
case FN_DROP_TEXT:
|
|
case SID_ATTR_PARA_LRSPACE:
|
|
{
|
|
const sal_uInt16 nWhich = GetPool().GetWhichIDFromSlotID( nSlot );
|
|
if ( pArgs && pArgs->GetItemState( nWhich ) == SfxItemState::SET )
|
|
bUseDialog = false;
|
|
[[fallthrough]];
|
|
}
|
|
case SID_PARA_DLG:
|
|
{
|
|
SwPaM* pPaM = nullptr;
|
|
|
|
if ( pArgs )
|
|
{
|
|
const SwPaMItem* pPaMItem = pArgs->GetItemIfSet( GetPool().GetWhichIDFromSlotID( FN_PARAM_PAM ), false );
|
|
if ( pPaMItem )
|
|
pPaM = pPaMItem->GetValue( );
|
|
}
|
|
|
|
if ( !pPaM )
|
|
pPaM = rWrtSh.GetCursor();
|
|
|
|
FieldUnit eMetric = ::GetDfltMetric( dynamic_cast<SwWebView*>( &GetView()) != nullptr );
|
|
SwModule* mod = SwModule::get();
|
|
mod->PutItem(SfxUInt16Item(SID_ATTR_METRIC, static_cast<sal_uInt16>(eMetric)));
|
|
|
|
bool bApplyCharUnit = ::HasCharUnit( dynamic_cast<SwWebView*>( &GetView()) != nullptr );
|
|
mod->PutItem(SfxBoolItem(SID_ATTR_APPLYCHARUNIT, bApplyCharUnit));
|
|
|
|
SfxItemSetFixed<
|
|
RES_PARATR_BEGIN, RES_FRMATR_END - 1,
|
|
// FillAttribute support:
|
|
XATTR_FILL_FIRST, XATTR_FILL_LAST,
|
|
// Includes SID_ATTR_TABSTOP_POS:
|
|
SID_ATTR_TABSTOP_DEFAULTS, SID_ATTR_TABSTOP_OFFSET,
|
|
SID_ATTR_BORDER_INNER, SID_ATTR_BORDER_INNER,
|
|
SID_ATTR_PARA_MODEL, SID_ATTR_PARA_KEEP,
|
|
// Items to hand over XPropertyList things like XColorList,
|
|
// XHatchList, XGradientList, and XBitmapList to the Area
|
|
// TabPage:
|
|
SID_COLOR_TABLE, SID_PATTERN_LIST,
|
|
SID_HTML_MODE, SID_HTML_MODE,
|
|
SID_ATTR_PARA_PAGENUM, SID_ATTR_PARA_PAGENUM,
|
|
FN_PARAM_1, FN_PARAM_1,
|
|
FN_NUMBER_NEWSTART, FN_NUMBER_NEWSTART_AT,
|
|
FN_DROP_TEXT, FN_DROP_CHAR_STYLE_NAME> aCoreSet( GetPool() );
|
|
|
|
// get also the list level indent values merged as LR-SPACE item, if needed.
|
|
rWrtSh.GetPaMAttr( pPaM, aCoreSet, true );
|
|
|
|
// create needed items for XPropertyList entries from the DrawModel so that
|
|
// the Area TabPage can access them
|
|
// Do this after GetCurAttr, this resets the ItemSet content again
|
|
SwDrawModel* pDrawModel = GetView().GetDocShell()->GetDoc()->getIDocumentDrawModelAccess().GetDrawModel();
|
|
|
|
aCoreSet.Put(SvxColorListItem(pDrawModel->GetColorList(), SID_COLOR_TABLE));
|
|
aCoreSet.Put(SvxGradientListItem(pDrawModel->GetGradientList(), SID_GRADIENT_LIST));
|
|
aCoreSet.Put(SvxHatchListItem(pDrawModel->GetHatchList(), SID_HATCH_LIST));
|
|
aCoreSet.Put(SvxBitmapListItem(pDrawModel->GetBitmapList(), SID_BITMAP_LIST));
|
|
aCoreSet.Put(SvxPatternListItem(pDrawModel->GetPatternList(), SID_PATTERN_LIST));
|
|
aCoreSet.Put(SfxUInt16Item(SID_HTML_MODE,
|
|
::GetHtmlMode(GetView().GetDocShell())));
|
|
|
|
// Tabulators: Put DefaultTabs into ItemSet
|
|
const SvxTabStopItem& rDefTabs =
|
|
GetPool().GetUserOrPoolDefaultItem(RES_PARATR_TABSTOP);
|
|
|
|
const sal_uInt16 nDefDist = o3tl::narrowing<sal_uInt16>(::GetTabDist( rDefTabs ));
|
|
SfxUInt16Item aDefDistItem( SID_ATTR_TABSTOP_DEFAULTS, nDefDist );
|
|
aCoreSet.Put( aDefDistItem );
|
|
|
|
// Current tabulator
|
|
SfxUInt16Item aTabPos( SID_ATTR_TABSTOP_POS, 0 );
|
|
aCoreSet.Put( aTabPos );
|
|
|
|
// Left border as offset
|
|
//#i24363# tab stops relative to indent
|
|
const tools::Long nOff
|
|
= rWrtSh.getIDocumentSettingAccess().get(DocumentSettingId::TABS_RELATIVE_TO_INDENT)
|
|
? aCoreSet.Get(RES_MARGIN_TEXTLEFT).ResolveTextLeft({})
|
|
: 0;
|
|
SfxInt32Item aOff( SID_ATTR_TABSTOP_OFFSET, nOff );
|
|
aCoreSet.Put( aOff );
|
|
|
|
// Setting the BoxInfo
|
|
::PrepareBoxInfo( aCoreSet, rWrtSh );
|
|
|
|
// Current page format
|
|
::SwToSfxPageDescAttr( aCoreSet );
|
|
|
|
// Properties of numbering
|
|
if (rWrtSh.GetNumRuleAtCurrCursorPos())
|
|
{
|
|
SfxBoolItem aStart( FN_NUMBER_NEWSTART, rWrtSh.IsNumRuleStart( pPaM ) );
|
|
aCoreSet.Put(aStart);
|
|
SfxUInt16Item aStartAt( FN_NUMBER_NEWSTART_AT,
|
|
rWrtSh.GetNodeNumStart( pPaM ) );
|
|
aCoreSet.Put(aStartAt);
|
|
}
|
|
VclPtr<SfxAbstractTabDialog> pDlg;
|
|
|
|
if ( bUseDialog && GetActiveView() )
|
|
{
|
|
OUString sDefPage;
|
|
if (pItem)
|
|
sDefPage = static_cast<const SfxStringItem*>(pItem)->GetValue();
|
|
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
pDlg.reset(pFact->CreateSwParaDlg(GetView().GetFrameWeld(), GetView(), aCoreSet, false, sDefPage));
|
|
}
|
|
|
|
if ( !bUseDialog )
|
|
{
|
|
if ( nSlot == SID_ATTR_PARA_LRSPACE)
|
|
{
|
|
SvxLRSpaceItem aParaMargin(static_cast<const SvxLRSpaceItem&>(pArgs->Get(nSlot)));
|
|
SvxFirstLineIndentItem firstLine(RES_MARGIN_FIRSTLINE);
|
|
SvxTextLeftMarginItem leftMargin(RES_MARGIN_TEXTLEFT);
|
|
SvxRightMarginItem rightMargin(RES_MARGIN_RIGHT);
|
|
firstLine.SetTextFirstLineOffset(aParaMargin.GetTextFirstLineOffset(),
|
|
aParaMargin.GetPropTextFirstLineOffset());
|
|
firstLine.SetAutoFirst(aParaMargin.IsAutoFirst());
|
|
leftMargin.SetTextLeft(aParaMargin.GetTextLeft(), aParaMargin.GetPropLeft());
|
|
rightMargin.SetRight(aParaMargin.GetRight(), aParaMargin.GetPropRight());
|
|
aCoreSet.Put(firstLine);
|
|
aCoreSet.Put(leftMargin);
|
|
aCoreSet.Put(rightMargin);
|
|
|
|
sw_ParagraphDialogResult(&aCoreSet, rWrtSh, rReq, pPaM);
|
|
}
|
|
else
|
|
sw_ParagraphDialogResult(const_cast<SfxItemSet*>(pArgs), rWrtSh, rReq, pPaM);
|
|
}
|
|
else if (pDlg)
|
|
{
|
|
auto pRequest = std::make_shared<SfxRequest>(rReq);
|
|
rReq.Ignore(); // the 'old' request is not relevant any more
|
|
|
|
auto vCursors = CopyPaMRing(*pPaM); // tdf#134439 make a copy to use at later apply
|
|
pDlg->StartExecuteAsync([pDlg, &rWrtSh, pDrawModel, pRequest=std::move(pRequest), nDefDist, vCursors=std::move(vCursors)](sal_Int32 nResult){
|
|
if (nResult == RET_OK)
|
|
{
|
|
// Apply defaults if necessary.
|
|
SfxItemSet* pSet = const_cast<SfxItemSet*>(pDlg->GetOutputItemSet());
|
|
sal_uInt16 nNewDist;
|
|
const SfxUInt16Item* pDefaultsItem = pSet->GetItemIfSet(SID_ATTR_TABSTOP_DEFAULTS, false);
|
|
if (pDefaultsItem && nDefDist != (nNewDist = pDefaultsItem->GetValue()) )
|
|
{
|
|
SvxTabStopItem aDefTabs( 0, 0, SvxTabAdjust::Default, RES_PARATR_TABSTOP );
|
|
MakeDefTabs( nNewDist, aDefTabs );
|
|
rWrtSh.SetDefault( aDefTabs );
|
|
pSet->ClearItem( SID_ATTR_TABSTOP_DEFAULTS );
|
|
}
|
|
|
|
const SfxPoolItem* pItem2 = nullptr;
|
|
if (SfxItemState::SET == pSet->GetItemState(FN_PARAM_1, false, &pItem2))
|
|
{
|
|
pSet->Put(SfxStringItem(FN_DROP_TEXT, static_cast<const SfxStringItem*>(pItem2)->GetValue()));
|
|
pSet->ClearItem(FN_PARAM_1);
|
|
}
|
|
|
|
if (const SwFormatDrop* pDropItem = pSet->GetItemIfSet(RES_PARATR_DROP, false))
|
|
{
|
|
OUString sCharStyleName;
|
|
if (pDropItem->GetCharFormat())
|
|
sCharStyleName = pDropItem->GetCharFormat()->GetName();
|
|
pSet->Put(SfxStringItem(FN_DROP_CHAR_STYLE_NAME, sCharStyleName));
|
|
}
|
|
|
|
const XFillStyleItem* pFS = pSet->GetItem<XFillStyleItem>(XATTR_FILLSTYLE);
|
|
bool bSet = pFS && pFS->GetValue() == drawing::FillStyle_GRADIENT;
|
|
const XFillGradientItem* pTempGradItem
|
|
= bSet ? pSet->GetItem<XFillGradientItem>(XATTR_FILLGRADIENT) : nullptr;
|
|
if (pTempGradItem && pTempGradItem->GetName().isEmpty())
|
|
{
|
|
// MigrateItemSet guarantees unique gradient names
|
|
SfxItemSetFixed<XATTR_FILLGRADIENT, XATTR_FILLGRADIENT> aMigrateSet(rWrtSh.GetView().GetPool());
|
|
aMigrateSet.Put(XFillGradientItem(u"gradient"_ustr, pTempGradItem->GetGradientValue()));
|
|
SdrModel::MigrateItemSet(&aMigrateSet, pSet, *pDrawModel);
|
|
}
|
|
|
|
bSet = pFS && pFS->GetValue() == drawing::FillStyle_HATCH;
|
|
const XFillHatchItem* pTempHatchItem
|
|
= bSet ? pSet->GetItem<XFillHatchItem>(XATTR_FILLHATCH) : nullptr;
|
|
if (pTempHatchItem && pTempHatchItem->GetName().isEmpty())
|
|
{
|
|
SfxItemSetFixed<XATTR_FILLHATCH, XATTR_FILLHATCH> aMigrateSet(rWrtSh.GetView().GetPool());
|
|
aMigrateSet.Put(XFillHatchItem(u"hatch"_ustr, pTempHatchItem->GetHatchValue()));
|
|
SdrModel::MigrateItemSet(&aMigrateSet, pSet, *pDrawModel);
|
|
}
|
|
|
|
sw_ParagraphDialogResult(pSet, rWrtSh, *pRequest, vCursors->front().get());
|
|
}
|
|
pDlg->disposeOnce();
|
|
});
|
|
}
|
|
}
|
|
break;
|
|
case FN_NUM_CONTINUE:
|
|
{
|
|
OUString sContinuedListId;
|
|
const SwNumRule* pRule =
|
|
rWrtSh.SearchNumRule( true, sContinuedListId );
|
|
// #i86492#
|
|
// Search also for bullet list
|
|
if ( !pRule )
|
|
{
|
|
pRule = rWrtSh.SearchNumRule( false, sContinuedListId );
|
|
}
|
|
if ( pRule )
|
|
{
|
|
rWrtSh.SetCurNumRule( *pRule, false, sContinuedListId );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case FN_SELECT_PARA:
|
|
{
|
|
if ( !rWrtSh.IsSttOfPara() )
|
|
rWrtSh.SttPara();
|
|
else
|
|
rWrtSh.EnterStdMode();
|
|
rWrtSh.EndPara( true );
|
|
}
|
|
break;
|
|
|
|
case SID_DEC_INDENT:
|
|
case SID_INC_INDENT:
|
|
//According to the requirement, modified the behavior when user
|
|
//using the indent button on the toolbar. Now if we increase/decrease indent for a
|
|
//paragraph which has bullet style it will increase/decrease the bullet level.
|
|
{
|
|
//If the current paragraph has bullet call the function to
|
|
//increase or decrease the bullet level.
|
|
//Why could I know whether a paragraph has bullet or not by checking the below conditions?
|
|
//Please refer to the "case KEY_TAB:" section in SwEditWin::KeyInput(..) :
|
|
// if( rSh.GetCurNumRule() && rSh.IsSttOfPara() &&
|
|
// !rSh.HasReadonlySel() )
|
|
// eKeyState = KS_NumDown;
|
|
//Above code demonstrates that when the cursor is at the start of a paragraph which has bullet,
|
|
//press TAB will increase the bullet level.
|
|
//So I copied from that ^^
|
|
if ( rWrtSh.GetNumRuleAtCurrCursorPos() && !rWrtSh.HasReadonlySel() )
|
|
{
|
|
rWrtSh.NumUpDown( SID_INC_INDENT == nSlot );
|
|
}
|
|
else //execute the original processing functions
|
|
{
|
|
//below is copied of the old codes
|
|
rWrtSh.MoveLeftMargin( SID_INC_INDENT == nSlot, rReq.GetModifier() != KEY_MOD1 );
|
|
}
|
|
}
|
|
rReq.Done();
|
|
break;
|
|
|
|
case FN_DEC_INDENT_OFFSET:
|
|
case FN_INC_INDENT_OFFSET:
|
|
rWrtSh.MoveLeftMargin( FN_INC_INDENT_OFFSET == nSlot, rReq.GetModifier() == KEY_MOD1 );
|
|
rReq.Done();
|
|
break;
|
|
|
|
case SID_ATTR_CHAR_COLOR2:
|
|
{
|
|
std::unique_ptr<const SvxColorItem> pRecentColor; // manage lifecycle scope
|
|
if (!pItem)
|
|
{
|
|
// no color provided: use the pre-selected color shown in the toolbar/sidebar
|
|
const std::optional<NamedColor> oColor
|
|
= GetView().GetDocShell()->GetRecentColor(SID_ATTR_CHAR_COLOR);
|
|
if (oColor.has_value())
|
|
{
|
|
const model::ComplexColor aCol = (*oColor).getComplexColor();
|
|
pRecentColor = std::make_unique<const SvxColorItem>(
|
|
aCol.getFinalColor(), aCol, RES_CHRATR_COLOR);
|
|
pItem = pRecentColor.get();
|
|
}
|
|
}
|
|
|
|
if (pItem)
|
|
{
|
|
auto* pColorItem = static_cast<const SvxColorItem*>(pItem);
|
|
SwEditWin& rEditWin = GetView().GetEditWin();
|
|
rEditWin.SetWaterCanTextColor(pColorItem->GetValue());
|
|
SwApplyTemplate* pApply = rEditWin.GetApplyTemplate();
|
|
|
|
// If there is a selection, then set the color on it
|
|
// otherwise, it'll be the color for the next text to be typed
|
|
if (!pApply || pApply->nColor != SID_ATTR_CHAR_COLOR_EXT)
|
|
{
|
|
rWrtSh.SetAttrItem(SvxColorItem(pColorItem->GetValue(), pColorItem->getComplexColor(), RES_CHRATR_COLOR));
|
|
}
|
|
|
|
rReq.Done();
|
|
}
|
|
}
|
|
break;
|
|
case SID_ATTR_CHAR_BACK_COLOR:
|
|
case SID_ATTR_CHAR_COLOR_BACKGROUND: // deprecated
|
|
case SID_ATTR_CHAR_COLOR_EXT:
|
|
{
|
|
Color aColor = COL_TRANSPARENT;
|
|
model::ComplexColor aComplexColor;
|
|
|
|
if (pItem)
|
|
{
|
|
auto* pColorItem = static_cast<const SvxColorItem*>(pItem);
|
|
aColor = pColorItem->GetValue();
|
|
aComplexColor = pColorItem->getComplexColor();
|
|
}
|
|
else if (nSlot == SID_ATTR_CHAR_BACK_COLOR)
|
|
{
|
|
// no color provided: use the pre-selected color shown in the toolbar/sidebar
|
|
const std::optional<NamedColor> oColor
|
|
= GetView().GetDocShell()->GetRecentColor(nSlot);
|
|
if (oColor.has_value())
|
|
{
|
|
aComplexColor = (*oColor).getComplexColor();
|
|
aColor = aComplexColor.getFinalColor();
|
|
}
|
|
}
|
|
|
|
SwEditWin& rEdtWin = GetView().GetEditWin();
|
|
if (nSlot != SID_ATTR_CHAR_COLOR_EXT)
|
|
rEdtWin.SetWaterCanTextBackColor(aColor);
|
|
else if (pItem)
|
|
rEdtWin.SetWaterCanTextColor(aColor);
|
|
|
|
SwApplyTemplate* pApply = rEdtWin.GetApplyTemplate();
|
|
SwApplyTemplate aTempl;
|
|
if (!pApply && (rWrtSh.HasSelection() || rReq.IsAPI()))
|
|
{
|
|
if (nSlot != SID_ATTR_CHAR_COLOR_EXT)
|
|
{
|
|
SfxItemSetFixed<RES_CHRATR_BACKGROUND, RES_CHRATR_BACKGROUND> aCoreSet( rWrtSh.GetView().GetPool() );
|
|
|
|
rWrtSh.GetCurAttr(aCoreSet);
|
|
|
|
// Remove highlight if already set of the same color
|
|
const SvxBrushItem& rBrushItem = aCoreSet.Get(RES_CHRATR_BACKGROUND);
|
|
if (aColor == rBrushItem.GetColor())
|
|
{
|
|
aComplexColor = model::ComplexColor();
|
|
aColor = COL_TRANSPARENT;
|
|
}
|
|
ApplyCharBackground(aColor, aComplexColor, rWrtSh);
|
|
}
|
|
else
|
|
rWrtSh.SetAttrItem(SvxColorItem(aColor, aComplexColor, RES_CHRATR_COLOR));
|
|
}
|
|
else
|
|
{
|
|
if(!pApply || pApply->nColor != nSlot)
|
|
aTempl.nColor = nSlot;
|
|
rEdtWin.SetApplyTemplate(aTempl);
|
|
}
|
|
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
|
|
case FN_NUM_BULLET_MOVEDOWN:
|
|
if (!rWrtSh.IsAddMode())
|
|
rWrtSh.MoveParagraph();
|
|
rReq.Done();
|
|
break;
|
|
|
|
case FN_NUM_BULLET_MOVEUP:
|
|
if (!rWrtSh.IsAddMode())
|
|
rWrtSh.MoveParagraph(SwNodeOffset(-1));
|
|
rReq.Done();
|
|
break;
|
|
case SID_RUBY_DIALOG:
|
|
case SID_HYPERLINK_DIALOG:
|
|
{
|
|
SfxRequest aReq(nSlot, SfxCallMode::SLOT, SfxGetpApp()->GetPool());
|
|
GetView().GetViewFrame().ExecuteSlot( aReq);
|
|
rReq.Ignore();
|
|
}
|
|
break;
|
|
case FN_INSERT_PAGEHEADER:
|
|
case FN_INSERT_PAGEFOOTER:
|
|
if(pArgs && pArgs->Count())
|
|
{
|
|
OUString sStyleName;
|
|
if(pItem)
|
|
sStyleName = static_cast<const SfxStringItem*>(pItem)->GetValue();
|
|
bool bOn = true;
|
|
if( SfxItemState::SET == pArgs->GetItemState(FN_PARAM_1, false, &pItem))
|
|
bOn = static_cast<const SfxBoolItem*>(pItem)->GetValue();
|
|
rWrtSh.ChangeHeaderOrFooter(sStyleName, FN_INSERT_PAGEHEADER == nSlot, bOn, !rReq.IsAPI());
|
|
rReq.Done();
|
|
}
|
|
break;
|
|
case FN_READONLY_SELECTION_MODE :
|
|
if(GetView().GetDocShell()->IsReadOnly())
|
|
{
|
|
rWrtSh.SetReadonlySelectionOption(
|
|
!rWrtSh.GetViewOptions()->IsSelectionInReadonly());
|
|
rWrtSh.ShowCursor();
|
|
}
|
|
break;
|
|
case FN_SELECTION_MODE_DEFAULT:
|
|
case FN_SELECTION_MODE_BLOCK :
|
|
{
|
|
bool bSetBlockMode = !rWrtSh.IsBlockMode();
|
|
if( pArgs && SfxItemState::SET == pArgs->GetItemState(nSlot, false, &pItem))
|
|
bSetBlockMode = static_cast<const SfxBoolItem*>(pItem)->GetValue();
|
|
if( ( nSlot == FN_SELECTION_MODE_DEFAULT ) != bSetBlockMode )
|
|
rWrtSh.EnterBlockMode();
|
|
else
|
|
rWrtSh.EnterStdMode();
|
|
SfxBindings &rBnd = GetView().GetViewFrame().GetBindings();
|
|
rBnd.Invalidate(FN_STAT_SELMODE);
|
|
rBnd.Update(FN_STAT_SELMODE);
|
|
}
|
|
break;
|
|
case SID_OPEN_HYPERLINK:
|
|
case SID_COPY_HYPERLINK_LOCATION:
|
|
{
|
|
SfxItemSetFixed<RES_TXTATR_INETFMT, RES_TXTATR_INETFMT> aSet(GetPool());
|
|
rWrtSh.GetCurAttr(aSet);
|
|
|
|
const SwFormatINetFormat* pINetFormat = nullptr;
|
|
if(SfxItemState::SET <= aSet.GetItemState( RES_TXTATR_INETFMT ))
|
|
pINetFormat = &aSet.Get(RES_TXTATR_INETFMT);
|
|
else if (!rWrtSh.HasSelection())
|
|
{
|
|
// is the cursor at the beginning of a hyperlink?
|
|
const SwTextNode* pTextNd = rWrtSh.GetCursor()->GetPointNode().GetTextNode();
|
|
if (pTextNd)
|
|
{
|
|
const sal_Int32 nIndex = rWrtSh.GetCursor()->Start()->GetContentIndex();
|
|
const SwTextAttr* pINetFmt = pTextNd->GetTextAttrAt(nIndex, RES_TXTATR_INETFMT);
|
|
if (pINetFmt && !pINetFmt->GetINetFormat().GetValue().isEmpty())
|
|
pINetFormat = &pINetFmt->GetINetFormat();
|
|
}
|
|
}
|
|
|
|
if (pINetFormat)
|
|
{
|
|
if (nSlot == SID_OPEN_HYPERLINK)
|
|
{
|
|
rWrtSh.ClickToINetAttr(*pINetFormat);
|
|
}
|
|
else if (nSlot == SID_COPY_HYPERLINK_LOCATION)
|
|
{
|
|
OUString hyperlinkLocation = pINetFormat->GetValue();
|
|
::uno::Reference< datatransfer::clipboard::XClipboard > xClipboard = GetView().GetEditWin().GetClipboard();
|
|
vcl::unohelper::TextDataObject::CopyStringTo(hyperlinkLocation, xClipboard, SfxViewShell::Current());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SwField* pField = rWrtSh.GetCurField();
|
|
if (pField && pField->GetTyp()->Which() == SwFieldIds::TableOfAuthorities)
|
|
{
|
|
const auto& rAuthorityField = *static_cast<const SwAuthorityField*>(pField);
|
|
OUString targetURL = u""_ustr;
|
|
|
|
if (auto targetType = rAuthorityField.GetTargetType();
|
|
targetType == SwAuthorityField::TargetType::UseDisplayURL
|
|
|| targetType == SwAuthorityField::TargetType::UseTargetURL)
|
|
{
|
|
// Bibliography entry with URL also provides a hyperlink.
|
|
targetURL = rAuthorityField.GetAbsoluteURL();
|
|
}
|
|
|
|
if (targetURL.getLength() > 0)
|
|
{
|
|
if (nSlot == SID_OPEN_HYPERLINK)
|
|
{
|
|
::LoadURL(rWrtSh, targetURL, LoadUrlFlags::NewView, /*rTargetFrameName=*/OUString());
|
|
}
|
|
else if (nSlot == SID_COPY_HYPERLINK_LOCATION)
|
|
{
|
|
::uno::Reference< datatransfer::clipboard::XClipboard > xClipboard = GetView().GetEditWin().GetClipboard();
|
|
vcl::unohelper::TextDataObject::CopyStringTo(targetURL, xClipboard, SfxViewShell::Current());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case FN_OPEN_LOCAL_URL:
|
|
{
|
|
OUString aLocalURL = GetLocalURL(rWrtSh);
|
|
if (!aLocalURL.isEmpty())
|
|
{
|
|
::LoadURL(rWrtSh, aLocalURL, LoadUrlFlags::NewView, /*rTargetFrameName=*/OUString());
|
|
}
|
|
}
|
|
break;
|
|
case SID_OPEN_XML_FILTERSETTINGS:
|
|
{
|
|
HandleOpenXmlFilterSettings(rReq);
|
|
}
|
|
break;
|
|
case FN_FORMAT_APPLY_HEAD1:
|
|
{
|
|
}
|
|
break;
|
|
case FN_FORMAT_APPLY_HEAD2:
|
|
{
|
|
}
|
|
break;
|
|
case FN_FORMAT_APPLY_HEAD3:
|
|
{
|
|
}
|
|
break;
|
|
case FN_FORMAT_APPLY_DEFAULT:
|
|
{
|
|
}
|
|
break;
|
|
case FN_FORMAT_APPLY_TEXTBODY:
|
|
{
|
|
}
|
|
break;
|
|
case FN_WORDCOUNT_DIALOG:
|
|
{
|
|
GetView().UpdateWordCount(this, nSlot);
|
|
}
|
|
break;
|
|
case FN_PROTECT_FIELDS:
|
|
case FN_PROTECT_BOOKMARKS:
|
|
{
|
|
IDocumentSettingAccess& rIDSA = rWrtSh.getIDocumentSettingAccess();
|
|
DocumentSettingId aSettingId = nSlot == FN_PROTECT_FIELDS
|
|
? DocumentSettingId::PROTECT_FIELDS
|
|
: DocumentSettingId::PROTECT_BOOKMARKS;
|
|
rIDSA.set(aSettingId, !rIDSA.get(aSettingId));
|
|
// Invalidate so that toggle state gets updated
|
|
SfxViewFrame& rViewFrame = GetView().GetViewFrame();
|
|
rViewFrame.GetBindings().Invalidate(nSlot);
|
|
rViewFrame.GetBindings().Update(nSlot);
|
|
}
|
|
break;
|
|
case SID_FM_CTL_PROPERTIES:
|
|
{
|
|
SwPosition aPos(*GetShell().GetCursor()->GetPoint());
|
|
sw::mark::Fieldmark* pFieldBM = GetShell().getIDocumentMarkAccess()->getInnerFieldmarkFor(aPos);
|
|
if ( !pFieldBM )
|
|
{
|
|
aPos.AdjustContent(-1);
|
|
pFieldBM = GetShell().getIDocumentMarkAccess()->getInnerFieldmarkFor(aPos);
|
|
}
|
|
|
|
if ( pFieldBM && pFieldBM->GetFieldname() == ODF_FORMDROPDOWN
|
|
&& !(rWrtSh.GetCurrSection() && rWrtSh.GetCurrSection()->IsProtect()) )
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
VclPtr<AbstractDropDownFormFieldDialog> pDlg(pFact->CreateDropDownFormFieldDialog(rWrtSh.GetView().GetFrameWeld(), pFieldBM));
|
|
auto xRequest = std::make_shared<SfxRequest>(rReq);
|
|
rReq.Ignore(); // the 'old' request is not relevant any more
|
|
auto pWrtSh = &rWrtSh;
|
|
pDlg->StartExecuteAsync(
|
|
[pDlg, pFieldBM, pWrtSh, xRequest=std::move(xRequest)] (sal_Int32 nResult)->void
|
|
{
|
|
if (nResult == RET_OK)
|
|
{
|
|
pDlg->Apply();
|
|
pFieldBM->Invalidate();
|
|
pWrtSh->InvalidateWindows( SwRect(pWrtSh->GetView().GetVisArea()) );
|
|
pWrtSh->UpdateCursor(); // cursor position might be invalid
|
|
}
|
|
pDlg->disposeOnce();
|
|
xRequest->Done();
|
|
}
|
|
);
|
|
}
|
|
else if ( pFieldBM && pFieldBM->GetFieldname() == ODF_FORMDATE )
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
sw::mark::DateFieldmark* pDateField = &dynamic_cast<sw::mark::DateFieldmark&>(*pFieldBM);
|
|
VclPtr<AbstractDateFormFieldDialog> pDlg(pFact->CreateDateFormFieldDialog(rWrtSh.GetView().GetFrameWeld(), pDateField, *GetView().GetDocShell()->GetDoc()));
|
|
auto pWrtSh = &rWrtSh;
|
|
auto xRequest = std::make_shared<SfxRequest>(rReq);
|
|
rReq.Ignore(); // the 'old' request is not relevant any more
|
|
pDlg->StartExecuteAsync(
|
|
[pDlg, pWrtSh, pDateField, xRequest=std::move(xRequest)] (sal_Int32 nResult)->void
|
|
{
|
|
if (nResult == RET_OK)
|
|
{
|
|
pDlg->Apply();
|
|
pDateField->Invalidate();
|
|
pWrtSh->InvalidateWindows( SwRect(pWrtSh->GetView().GetVisArea()) );
|
|
pWrtSh->UpdateCursor(); // cursor position might be invalid
|
|
}
|
|
pDlg->disposeOnce();
|
|
xRequest->Done();
|
|
}
|
|
);
|
|
}
|
|
else
|
|
{
|
|
SfxRequest aReq(GetView().GetViewFrame(), SID_FM_CTL_PROPERTIES);
|
|
aReq.AppendItem( SfxBoolItem( SID_FM_CTL_PROPERTIES, true ) );
|
|
rWrtSh.GetView().GetFormShell()->Execute( aReq );
|
|
}
|
|
}
|
|
break;
|
|
case SID_FM_TRANSLATE:
|
|
{
|
|
#if HAVE_FEATURE_CURL && !ENABLE_WASM_STRIP_EXTRA
|
|
const SfxPoolItem* pTargetLangStringItem = nullptr;
|
|
if (pArgs && SfxItemState::SET == pArgs->GetItemState(SID_ATTR_TARGETLANG_STR, false, &pTargetLangStringItem))
|
|
{
|
|
std::optional<OUString> oDeeplAPIUrl = officecfg::Office::Linguistic::Translation::Deepl::ApiURL::get();
|
|
std::optional<OUString> oDeeplKey = officecfg::Office::Linguistic::Translation::Deepl::AuthKey::get();
|
|
if (!oDeeplAPIUrl || oDeeplAPIUrl->isEmpty() || !oDeeplKey || oDeeplKey->isEmpty())
|
|
{
|
|
SAL_WARN("sw.ui", "SID_FM_TRANSLATE: API options are not set");
|
|
break;
|
|
}
|
|
const OString aAPIUrl = OUStringToOString(rtl::Concat2View(*oDeeplAPIUrl + "?tag_handling=html"), RTL_TEXTENCODING_UTF8).trim();
|
|
const OString aAuthKey = OUStringToOString(*oDeeplKey, RTL_TEXTENCODING_UTF8).trim();
|
|
OString aTargetLang = OUStringToOString(static_cast<const SfxStringItem*>(pTargetLangStringItem)->GetValue(), RTL_TEXTENCODING_UTF8);
|
|
SwTranslateHelper::TranslateAPIConfig aConfig({aAPIUrl, aAuthKey, aTargetLang});
|
|
SwTranslateHelper::TranslateDocument(rWrtSh, aConfig);
|
|
}
|
|
else
|
|
{
|
|
SwAbstractDialogFactory* pFact = SwAbstractDialogFactory::Create();
|
|
std::shared_ptr<AbstractSwTranslateLangSelectDlg> pAbstractDialog(pFact->CreateSwTranslateLangSelectDlg(GetView().GetFrameWeld(), rWrtSh));
|
|
std::shared_ptr<weld::DialogController> pDialogController(pAbstractDialog->getDialogController());
|
|
weld::DialogController::runAsync(pDialogController, [] (sal_Int32 /*nResult*/) { });
|
|
}
|
|
#endif // HAVE_FEATURE_CURL && ENABLE_WASM_STRIP_EXTRA
|
|
}
|
|
break;
|
|
case SID_SPELLCHECK_IGNORE:
|
|
{
|
|
SwPaM *pPaM = rWrtSh.GetCursor();
|
|
if (pPaM)
|
|
SwEditShell::IgnoreGrammarErrorAt( *pPaM );
|
|
}
|
|
break;
|
|
case SID_SPELLCHECK_IGNORE_ALL:
|
|
{
|
|
OUString sApplyText;
|
|
const SfxStringItem* pItem2 = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pItem2)
|
|
sApplyText = pItem2->GetValue();
|
|
|
|
if(sApplyText == "Grammar")
|
|
{
|
|
linguistic2::ProofreadingResult aGrammarCheckRes;
|
|
sal_Int32 nErrorInResult = -1;
|
|
uno::Sequence< OUString > aSuggestions;
|
|
sal_Int32 nErrorPosInText = -1;
|
|
SwRect aToFill;
|
|
bool bCorrectionRes = rWrtSh.GetGrammarCorrection( aGrammarCheckRes, nErrorPosInText, nErrorInResult, aSuggestions, nullptr, aToFill );
|
|
if(bCorrectionRes)
|
|
{
|
|
try {
|
|
uno::Reference< linguistic2::XDictionary > xDictionary = LinguMgr::GetIgnoreAllList();
|
|
aGrammarCheckRes.xProofreader->ignoreRule(
|
|
aGrammarCheckRes.aErrors[ nErrorInResult ].aRuleIdentifier,
|
|
aGrammarCheckRes.aLocale );
|
|
// refresh the layout of the actual paragraph (faster)
|
|
SwPaM *pPaM = rWrtSh.GetCursor();
|
|
if (pPaM)
|
|
SwEditShell::IgnoreGrammarErrorAt( *pPaM );
|
|
if (xDictionary.is() && pPaM)
|
|
{
|
|
linguistic::AddEntryToDic( xDictionary, pPaM->GetText(), false, OUString() );
|
|
// refresh the layout of all paragraphs (workaround to launch a dictionary event)
|
|
xDictionary->setActive(false);
|
|
xDictionary->setActive(true);
|
|
}
|
|
}
|
|
catch( const uno::Exception& )
|
|
{
|
|
}
|
|
}
|
|
}
|
|
else if (sApplyText == "Spelling")
|
|
{
|
|
SwRect aToFill;
|
|
uno::Reference<linguistic2::XSpellAlternatives> xSpellAlt(rWrtSh.GetCorrection(nullptr, aToFill));
|
|
if (!xSpellAlt.is())
|
|
return;
|
|
uno::Reference< linguistic2::XDictionary > xDictionary = LinguMgr::GetIgnoreAllList();
|
|
OUString sWord(xSpellAlt->getWord());
|
|
linguistic::DictionaryError nAddRes = linguistic::AddEntryToDic( xDictionary,
|
|
sWord, false, OUString() );
|
|
if (linguistic::DictionaryError::NONE != nAddRes && xDictionary.is() && !xDictionary->getEntry(sWord).is())
|
|
{
|
|
SvxDicError(rWrtSh.GetView().GetFrameWeld(), nAddRes);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case SID_SPELLCHECK_APPLY_SUGGESTION:
|
|
{
|
|
OUString sApplyText;
|
|
const SfxStringItem* pItem2 = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pItem2)
|
|
sApplyText = pItem2->GetValue();
|
|
|
|
static constexpr OUString sSpellingRule(u"Spelling_"_ustr);
|
|
static constexpr OUString sGrammarRule(u"Grammar_"_ustr);
|
|
|
|
bool bGrammar = false;
|
|
sal_Int32 nPos = 0;
|
|
uno::Reference< linguistic2::XSpellAlternatives > xSpellAlt;
|
|
if(-1 != (nPos = sApplyText.indexOf( sGrammarRule )))
|
|
{
|
|
sApplyText = sApplyText.replaceAt(nPos, sGrammarRule.getLength(), u"");
|
|
bGrammar = true;
|
|
}
|
|
else if (-1 != (nPos = sApplyText.indexOf( sSpellingRule )))
|
|
{
|
|
sApplyText = sApplyText.replaceAt(nPos, sSpellingRule.getLength(), u"");
|
|
SwRect aToFill;
|
|
xSpellAlt.set(rWrtSh.GetCorrection(nullptr, aToFill));
|
|
bGrammar = false;
|
|
}
|
|
|
|
if (!bGrammar && !xSpellAlt.is())
|
|
return;
|
|
|
|
bool bOldIns = rWrtSh.IsInsMode();
|
|
rWrtSh.SetInsMode();
|
|
|
|
OUString aTmp( sApplyText );
|
|
OUString aOrig( bGrammar ? OUString() : xSpellAlt->getWord() );
|
|
|
|
// if original word has a trailing . (likely the end of a sentence)
|
|
// and the replacement text hasn't, then add it to the replacement
|
|
if (!aTmp.isEmpty() && !aOrig.isEmpty() &&
|
|
aOrig.endsWith(".") && /* !IsAlphaNumeric ??*/
|
|
!aTmp.endsWith("."))
|
|
{
|
|
aTmp += ".";
|
|
}
|
|
|
|
SwRewriter aRewriter;
|
|
|
|
aRewriter.AddRule(UndoArg1, rWrtSh.GetCursorDescr()
|
|
// don't show the hidden control character of the comment
|
|
.replaceAll(OUStringChar(CH_TXTATR_INWORD), "") );
|
|
aRewriter.AddRule(UndoArg2, SwResId(STR_YIELDS));
|
|
|
|
OUString aTmpStr = SwResId(STR_START_QUOTE) +
|
|
aTmp + SwResId(STR_END_QUOTE);
|
|
aRewriter.AddRule(UndoArg3, aTmpStr);
|
|
|
|
rWrtSh.StartUndo(SwUndoId::UI_REPLACE, &aRewriter);
|
|
rWrtSh.StartAction();
|
|
|
|
// keep comments at the end of the replacement in case spelling correction is
|
|
// invoked via the context menu. The spell check dialog does the correction in edlingu.cxx.
|
|
rWrtSh.ReplaceKeepComments(aTmp);
|
|
|
|
rWrtSh.EndAction();
|
|
rWrtSh.EndUndo();
|
|
|
|
rWrtSh.SetInsMode( bOldIns );
|
|
}
|
|
break;
|
|
case FN_TRANSFORM_DOCUMENT_STRUCTURE:
|
|
{
|
|
// get the parameter, what to transform
|
|
OUString aDataJson;
|
|
const SfxStringItem* pDataJson = rReq.GetArg<SfxStringItem>(FN_PARAM_1);
|
|
if (pDataJson)
|
|
{
|
|
aDataJson = pDataJson->GetValue();
|
|
aDataJson = rtl::Uri::decode(aDataJson, rtl_UriDecodeStrict, RTL_TEXTENCODING_UTF8);
|
|
}
|
|
|
|
// parse the JSON got prom parameter
|
|
boost::property_tree::ptree aTree;
|
|
std::stringstream aStream(
|
|
(std::string(OUStringToOString(aDataJson, RTL_TEXTENCODING_UTF8))));
|
|
try
|
|
{
|
|
boost::property_tree::read_json(aStream, aTree);
|
|
}
|
|
catch (...)
|
|
{
|
|
lcl_LogWarning("FillApi Transform parameter, Wrong JSON format. ");
|
|
throw;
|
|
}
|
|
|
|
// get the loaded content controls
|
|
uno::Reference<text::XContentControlsSupplier> xCCSupplier(
|
|
GetView().GetDocShell()->GetModel(), uno::UNO_QUERY);
|
|
if (!xCCSupplier.is())
|
|
break;
|
|
|
|
uno::Reference<container::XIndexAccess> xContentControls
|
|
= xCCSupplier->getContentControls();
|
|
int iCCcount = xContentControls->getCount();
|
|
|
|
enum class ContentFilterType
|
|
{
|
|
ERROR = -1,
|
|
INDEX,
|
|
TAG,
|
|
ALIAS,
|
|
ID
|
|
};
|
|
std::vector<std::string> aIdTexts = { ".ByIndex.", ".ByTag.", ".ByAlias.", ".ById." };
|
|
|
|
// get charts
|
|
uno::Reference<text::XTextEmbeddedObjectsSupplier> xEOS(
|
|
GetView().GetDocShell()->GetModel(), uno::UNO_QUERY);
|
|
if (!xEOS.is())
|
|
break;
|
|
uno::Reference<container::XIndexAccess> xEmbeddeds(xEOS->getEmbeddedObjects(),
|
|
uno::UNO_QUERY);
|
|
if (!xEmbeddeds.is())
|
|
break;
|
|
|
|
sal_Int32 nEOcount = xEmbeddeds->getCount();
|
|
|
|
enum class ChartFilterType
|
|
{
|
|
ERROR = -1,
|
|
INDEX,
|
|
NAME,
|
|
TITLE,
|
|
SUBTITLE
|
|
};
|
|
std::vector<std::string> aIdChartTexts
|
|
= { ".ByEmbedIndex.", ".ByEmbedName.", ".ByTitle.", ".BySubTitle." };
|
|
|
|
// Iterate through the JSON data loaded into a tree structure
|
|
for (const auto& aItem : aTree)
|
|
{
|
|
if (aItem.first == "Transforms")
|
|
{
|
|
// Handle all transformations
|
|
for (const auto& aItem2 : aItem.second)
|
|
{
|
|
if (aItem2.first.starts_with("Charts"))
|
|
{
|
|
std::string aTextEnd = aItem2.first.substr(6);
|
|
std::string aValue = "";
|
|
ChartFilterType iKeyId = ChartFilterType::ERROR;
|
|
// Find how the chart is identified: ByIndex, ByTitle...
|
|
for (size_t i = 0; i < aIdChartTexts.size(); i++)
|
|
{
|
|
if (aTextEnd.starts_with(aIdChartTexts[i]))
|
|
{
|
|
iKeyId = static_cast<ChartFilterType>(i);
|
|
aValue = aTextEnd.substr(aIdChartTexts[i].length());
|
|
break;
|
|
}
|
|
}
|
|
if (iKeyId != ChartFilterType::ERROR)
|
|
{
|
|
// A chart transformation filter can match multiple charts
|
|
// In that case every matching charts will be transformed
|
|
// If no chart match to the filter, then we show warning
|
|
bool bChartFound = false;
|
|
for (int i = 0; i < nEOcount; ++i)
|
|
{
|
|
uno::Reference<beans::XPropertySet> xShapeProps(
|
|
xEmbeddeds->getByIndex(i), uno::UNO_QUERY);
|
|
if (!xShapeProps.is())
|
|
continue;
|
|
|
|
uno::Reference<frame::XModel> xDocModel;
|
|
xShapeProps->getPropertyValue(u"Model"_ustr) >>= xDocModel;
|
|
if (!xDocModel.is())
|
|
continue;
|
|
|
|
uno::Reference<chart2::XChartDocument> xChartDoc(
|
|
xDocModel, uno::UNO_QUERY);
|
|
if (!xChartDoc.is())
|
|
continue;
|
|
|
|
uno::Reference<chart2::data::XDataProvider> xDataProvider(
|
|
xChartDoc->getDataProvider());
|
|
if (!xDataProvider.is())
|
|
continue;
|
|
|
|
uno::Reference<chart::XChartDataArray> xDataArray(
|
|
xChartDoc->getDataProvider(), uno::UNO_QUERY);
|
|
if (!xDataArray.is())
|
|
continue;
|
|
|
|
uno::Reference<chart2::XInternalDataProvider> xIDataProvider(
|
|
xChartDoc->getDataProvider(), uno::UNO_QUERY);
|
|
if (!xIDataProvider.is())
|
|
continue;
|
|
|
|
uno::Reference<util::XModifiable> xModi(xDocModel,
|
|
uno::UNO_QUERY);
|
|
if (!xModi.is())
|
|
continue;
|
|
|
|
switch (iKeyId)
|
|
{
|
|
case ChartFilterType::INDEX:
|
|
{
|
|
if (stoi(aValue) != i)
|
|
continue;
|
|
}
|
|
break;
|
|
case ChartFilterType::NAME:
|
|
{
|
|
uno::Reference<container::XNamed> xNamedShape(
|
|
xEmbeddeds->getByIndex(i), uno::UNO_QUERY);
|
|
if (xNamedShape.is())
|
|
{
|
|
OUString aName;
|
|
aName = xNamedShape->getName();
|
|
if (OStringToOUString(aValue, RTL_TEXTENCODING_UTF8)
|
|
!= aName)
|
|
continue;
|
|
}
|
|
}
|
|
break;
|
|
case ChartFilterType::TITLE:
|
|
{
|
|
uno::Reference<chart2::XTitled> xTitled(
|
|
xChartDoc, uno::UNO_QUERY_THROW);
|
|
if (!xTitled.is())
|
|
continue;
|
|
uno::Reference<chart2::XTitle> xTitle
|
|
= xTitled->getTitleObject();
|
|
if (!xTitle.is())
|
|
continue;
|
|
|
|
OUString aTitle;
|
|
const uno::Sequence<
|
|
uno::Reference<chart2::XFormattedString>>
|
|
aFSSeq = xTitle->getText();
|
|
for (auto const& fs : aFSSeq)
|
|
aTitle += fs->getString();
|
|
if (OStringToOUString(aValue, RTL_TEXTENCODING_UTF8)
|
|
!= aTitle)
|
|
continue;
|
|
}
|
|
break;
|
|
case ChartFilterType::SUBTITLE:
|
|
{
|
|
uno::Reference<chart2::XDiagram> xDiagram
|
|
= xChartDoc->getFirstDiagram();
|
|
if (!xDiagram.is())
|
|
continue;
|
|
|
|
uno::Reference<chart2::XTitled> xTitled(
|
|
xDiagram, uno::UNO_QUERY_THROW);
|
|
if (!xTitled.is())
|
|
continue;
|
|
|
|
uno::Reference<chart2::XTitle> xSubTitle(
|
|
xTitled->getTitleObject());
|
|
if (!xSubTitle.is())
|
|
continue;
|
|
|
|
OUString aSubTitle;
|
|
const uno::Sequence<
|
|
uno::Reference<chart2::XFormattedString>>
|
|
aFSSeq = xSubTitle->getText();
|
|
for (auto const& fs : aFSSeq)
|
|
aSubTitle += fs->getString();
|
|
if (OStringToOUString(aValue, RTL_TEXTENCODING_UTF8)
|
|
!= aSubTitle)
|
|
continue;
|
|
}
|
|
break;
|
|
default:
|
|
continue;
|
|
}
|
|
|
|
// We have a match, this chart need to be transformed
|
|
// Set all the values (of the chart) what is needed
|
|
bChartFound = true;
|
|
|
|
// Check if the InternalDataProvider is row or column based.
|
|
bool bChartUseColumns = false;
|
|
uno::Sequence<beans::PropertyValue> aArguments(
|
|
xDataProvider->detectArguments(nullptr));
|
|
for (sal_Int32 j = 0; j < aArguments.getLength(); ++j)
|
|
{
|
|
if (aArguments[j].Name == "DataRowSource")
|
|
{
|
|
css::chart::ChartDataRowSource eRowSource;
|
|
if (aArguments[j].Value >>= eRowSource)
|
|
bChartUseColumns
|
|
= (eRowSource
|
|
== css::chart::ChartDataRowSource_COLUMNS);
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (const auto& aItem3 : aItem2.second)
|
|
{
|
|
if (aItem3.first.starts_with("deletecolumn.")
|
|
|| aItem3.first.starts_with("deleterow.")
|
|
|| aItem3.first.starts_with("insertcolumn.")
|
|
|| aItem3.first.starts_with("insertrow.")
|
|
|| aItem3.first.starts_with("modifycolumn.")
|
|
|| aItem3.first.starts_with("modifyrow."))
|
|
{
|
|
// delete insert, or modify a row, or column
|
|
// column, or row?
|
|
bool bSetColumn = (aItem3.first[6] == 'c');
|
|
int nId = stoi(aItem3.first.substr(bSetColumn ? 13 : 10));
|
|
bool bDelete = aItem3.first.starts_with("delete");
|
|
// delete/insert a row/column if needed
|
|
if (!aItem3.first.starts_with("modify"))
|
|
{
|
|
if (bChartUseColumns == bSetColumn)
|
|
{
|
|
if (bDelete)
|
|
{
|
|
if (!lcl_DeleteChartColumns(xChartDoc, nId))
|
|
continue;
|
|
xIDataProvider->deleteSequence(nId);
|
|
}
|
|
else
|
|
{
|
|
if (!lcl_InsertChartColumns(xChartDoc, nId))
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (bDelete)
|
|
{
|
|
xIDataProvider
|
|
->deleteDataPointForAllSequences(nId);
|
|
}
|
|
else
|
|
{
|
|
xIDataProvider
|
|
->insertDataPointForAllSequences(nId
|
|
- 1);
|
|
}
|
|
}
|
|
}
|
|
// set values also, if needed
|
|
if (!bDelete && aItem3.second.size() > 0)
|
|
{
|
|
uno::Sequence<uno::Sequence<double>> aData
|
|
= xDataArray->getData();
|
|
uno::Sequence<double>* pRows = aData.getArray();
|
|
|
|
int nIndex = 0;
|
|
int nX = nId;
|
|
int nY = nId;
|
|
bool bIndexWarning = false;
|
|
for (const auto& aItem4 : aItem3.second)
|
|
{
|
|
if (bSetColumn)
|
|
{
|
|
nY = nIndex;
|
|
}
|
|
else
|
|
{
|
|
nX = nIndex;
|
|
}
|
|
if (nY < aData.getLength() && nY >= 0
|
|
&& nX < pRows[nY].getLength() && nX >= 0)
|
|
{
|
|
double* pCols = pRows[nY].getArray();
|
|
pCols[nX]
|
|
= aItem4.second.get_value<double>();
|
|
}
|
|
else
|
|
{
|
|
bIndexWarning = true;
|
|
}
|
|
|
|
nIndex++;
|
|
}
|
|
if (bIndexWarning)
|
|
{
|
|
std::string sValues = "";
|
|
for (const auto& atemp : aItem3.second)
|
|
{
|
|
if (sValues != "")
|
|
{
|
|
sValues += ", ";
|
|
}
|
|
sValues += atemp.second
|
|
.get_value<std::string>();
|
|
}
|
|
lcl_LogWarning(
|
|
"FillApi chart: Invalid Cell Index at: '"
|
|
+ aItem3.first + ": " + sValues
|
|
+ "' (probably too many parameters)");
|
|
}
|
|
|
|
xDataArray->setData(aData);
|
|
}
|
|
}
|
|
else if (aItem3.first.starts_with("setrowdesc"))
|
|
{
|
|
// set row descriptions
|
|
uno::Sequence<OUString> aRowDesc
|
|
= xDataArray->getRowDescriptions();
|
|
OUString* aRowdata = aRowDesc.getArray();
|
|
|
|
if (aItem3.first.starts_with("setrowdesc."))
|
|
{
|
|
// set only 1 description
|
|
int nValue = stoi(aItem3.first.substr(11));
|
|
if (nValue >= 0 && nValue < aRowDesc.getLength())
|
|
{
|
|
aRowdata[nValue] = OStringToOUString(
|
|
aItem3.second.get_value<std::string>(),
|
|
RTL_TEXTENCODING_UTF8);
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning("FillApi chart setrowdesc: "
|
|
"invalid Index at: '"
|
|
+ aItem3.first + "'");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// set an array of description at once
|
|
int nIndex = 0;
|
|
for (const auto& aItem4 : aItem3.second)
|
|
{
|
|
if (nIndex >= aRowDesc.getLength())
|
|
{
|
|
lcl_LogWarning("FillApi chart setrowdesc: "
|
|
"too many params");
|
|
break;
|
|
}
|
|
aRowdata[nIndex] = OStringToOUString(
|
|
aItem4.second.get_value<std::string>(),
|
|
RTL_TEXTENCODING_UTF8);
|
|
nIndex++;
|
|
}
|
|
}
|
|
xDataArray->setRowDescriptions(aRowDesc);
|
|
}
|
|
else if (aItem3.first.starts_with("setcolumndesc"))
|
|
{
|
|
// set column descriptions
|
|
uno::Sequence<OUString> aColDesc
|
|
= xDataArray->getColumnDescriptions();
|
|
OUString* aColdata = aColDesc.getArray();
|
|
|
|
if (aItem3.first.starts_with("setcolumndesc."))
|
|
{
|
|
int nValue = stoi(aItem3.first.substr(14));
|
|
if (nValue >= 0 && nValue < aColDesc.getLength())
|
|
{
|
|
aColdata[nValue] = OStringToOUString(
|
|
aItem3.second.get_value<std::string>(),
|
|
RTL_TEXTENCODING_UTF8);
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning("FillApi chart setcolumndesc: "
|
|
"invalid Index at: '"
|
|
+ aItem3.first + "'");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int nIndex = 0;
|
|
for (const auto& aItem4 : aItem3.second)
|
|
{
|
|
if (nIndex >= aColDesc.getLength())
|
|
{
|
|
lcl_LogWarning(
|
|
"FillApi chart setcolumndesc:"
|
|
" too many parameters");
|
|
break;
|
|
}
|
|
aColdata[nIndex] = OStringToOUString(
|
|
aItem4.second.get_value<std::string>(),
|
|
RTL_TEXTENCODING_UTF8);
|
|
nIndex++;
|
|
}
|
|
}
|
|
xDataArray->setColumnDescriptions(aColDesc);
|
|
}
|
|
else if (aItem3.first.starts_with("resize"))
|
|
{
|
|
if (aItem3.second.size() >= 2)
|
|
{
|
|
auto aItem4 = aItem3.second.begin();
|
|
int nY = aItem4->second.get_value<int>();
|
|
int nX = (++aItem4)->second.get_value<int>();
|
|
|
|
if (nX < 1 || nY < 1)
|
|
{
|
|
lcl_LogWarning(
|
|
"FillApi chart resize: wrong param"
|
|
" (Needed: x,y >= 1)");
|
|
continue;
|
|
}
|
|
// here we need to use the new insert column thing
|
|
if (!lcl_ResizeChartColumns(xChartDoc, nX))
|
|
continue;
|
|
|
|
uno::Sequence<uno::Sequence<double>> aData
|
|
= xDataArray->getData();
|
|
if (aData.getLength() != nY)
|
|
aData.realloc(nY);
|
|
|
|
for (sal_Int32 j = 0; j < nY; ++j)
|
|
{
|
|
uno::Sequence<double>* pRows = aData.getArray();
|
|
// resize row if needed
|
|
if (pRows[j].getLength() != nX)
|
|
{
|
|
pRows[j].realloc(nX);
|
|
}
|
|
}
|
|
xDataArray->setData(aData);
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning(
|
|
"FillApi chart resize: not enough parameters"
|
|
" (x,y is needed)");
|
|
}
|
|
}
|
|
else if (aItem3.first.starts_with("data"))
|
|
{
|
|
// set table data values
|
|
uno::Sequence<uno::Sequence<double>> aData
|
|
= xDataArray->getData();
|
|
|
|
// set only 1 cell data
|
|
if (aItem3.first.starts_with("datayx."))
|
|
{
|
|
int nPoint = aItem3.first.find('.', 7);
|
|
int nY = stoi(aItem3.first.substr(7, nPoint - 7));
|
|
int nX = stoi(aItem3.first.substr(nPoint + 1));
|
|
bool bValidIndex = false;
|
|
if (nY < aData.getLength() && nY >= 0)
|
|
{
|
|
uno::Sequence<double>* pRows = aData.getArray();
|
|
if (nX < pRows[nY].getLength() && nX >= 0)
|
|
{
|
|
double* pCols = pRows[nY].getArray();
|
|
pCols[nX]
|
|
= aItem3.second.get_value<double>();
|
|
bValidIndex = true;
|
|
}
|
|
}
|
|
if (!bValidIndex)
|
|
{
|
|
lcl_LogWarning(
|
|
"FillApi chart datayx: invalid Index at: '"
|
|
+ aItem3.first + "'");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// set the whole data table
|
|
// resize if needed
|
|
int nRowsCount = aItem3.second.size();
|
|
int nColsCount = 0;
|
|
|
|
for (const auto& aItem4 : aItem3.second)
|
|
{
|
|
if (nColsCount
|
|
< static_cast<int>(aItem4.second.size()))
|
|
{
|
|
nColsCount = aItem4.second.size();
|
|
}
|
|
}
|
|
|
|
if (nColsCount > 0)
|
|
{
|
|
// here we need to use the new insert column thing
|
|
if(!lcl_ResizeChartColumns(xChartDoc, nColsCount))
|
|
continue;
|
|
|
|
if (aData.getLength() != nRowsCount)
|
|
aData.realloc(nRowsCount);
|
|
|
|
// set all the rows
|
|
sal_Int32 nY = 0;
|
|
for (const auto& aItem4 : aItem3.second)
|
|
{
|
|
uno::Sequence<double>* pRows
|
|
= aData.getArray();
|
|
// resize row if needed
|
|
if (pRows[nY].getLength() != nColsCount)
|
|
{
|
|
pRows[nY].realloc(nColsCount);
|
|
}
|
|
double* pCols = pRows[nY].getArray();
|
|
// set all values in the row
|
|
sal_Int32 nX = 0;
|
|
for (const auto& aItem5 : aItem4.second)
|
|
{
|
|
if (nX >= nColsCount)
|
|
{
|
|
// This should never happen
|
|
break;
|
|
}
|
|
pCols[nX]
|
|
= aItem5.second.get_value<double>();
|
|
nX++;
|
|
}
|
|
nY++;
|
|
}
|
|
}
|
|
}
|
|
xDataArray->setData(aData);
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning("FillApi chart command not recognised: '"
|
|
+ aItem3.first + "'");
|
|
}
|
|
xModi->setModified(true);
|
|
}
|
|
}
|
|
if (!bChartFound)
|
|
{
|
|
lcl_LogWarning("FillApi: No chart match the filter: '"
|
|
+ aItem2.first + "'");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning("FillApi chart filter type not recognised: '"
|
|
+ aItem2.first + "'");
|
|
}
|
|
}
|
|
|
|
if (aItem2.first.starts_with("ContentControls"))
|
|
{
|
|
std::string aTextEnd = aItem2.first.substr(15);
|
|
std::string aValue = "";
|
|
ContentFilterType iKeyId = ContentFilterType::ERROR;
|
|
// Find how the content control is identified: ByIndex, ByAlias...
|
|
for (size_t i = 0; i < aIdTexts.size(); i++)
|
|
{
|
|
if (aTextEnd.starts_with(aIdTexts[i]))
|
|
{
|
|
iKeyId = static_cast<ContentFilterType>(i);
|
|
aValue = aTextEnd.substr(aIdTexts[i].length());
|
|
break;
|
|
}
|
|
}
|
|
if (iKeyId != ContentFilterType::ERROR)
|
|
{
|
|
// Check all the content controls, if they match
|
|
bool bCCFound = false;
|
|
for (int i = 0; i < iCCcount; ++i)
|
|
{
|
|
uno::Reference<text::XTextContent> xContentControl;
|
|
xContentControls->getByIndex(i) >>= xContentControl;
|
|
|
|
uno::Reference<beans::XPropertySet> xContentControlProps(
|
|
xContentControl, uno::UNO_QUERY);
|
|
if (!xContentControlProps.is())
|
|
continue;
|
|
|
|
// Compare the loaded and the actual identifier
|
|
switch (iKeyId)
|
|
{
|
|
case ContentFilterType::INDEX:
|
|
{
|
|
if (stoi(aValue) != i)
|
|
continue;
|
|
}
|
|
break;
|
|
case ContentFilterType::ID:
|
|
{
|
|
sal_Int32 iID = -1;
|
|
xContentControlProps->getPropertyValue(UNO_NAME_ID)
|
|
>>= iID;
|
|
if (stoi(aValue) != iID)
|
|
continue;
|
|
}
|
|
break;
|
|
case ContentFilterType::ALIAS:
|
|
{
|
|
OUString aAlias;
|
|
xContentControlProps->getPropertyValue(UNO_NAME_ALIAS)
|
|
>>= aAlias;
|
|
if (OStringToOUString(aValue, RTL_TEXTENCODING_UTF8)
|
|
!= aAlias)
|
|
continue;
|
|
}
|
|
break;
|
|
case ContentFilterType::TAG:
|
|
{
|
|
OUString aTag;
|
|
xContentControlProps->getPropertyValue(UNO_NAME_TAG)
|
|
>>= aTag;
|
|
if (OStringToOUString(aValue, RTL_TEXTENCODING_UTF8)
|
|
!= aTag)
|
|
continue;
|
|
}
|
|
break;
|
|
default:
|
|
continue;
|
|
}
|
|
|
|
// We have a match, this content control need to be transformed
|
|
// Set all the values (of the content control) what is needed
|
|
bCCFound = true;
|
|
for (const auto& aItem3 : aItem2.second)
|
|
{
|
|
if (aItem3.first == "content")
|
|
{
|
|
std::string aContent
|
|
= aItem3.second.get_value<std::string>();
|
|
|
|
uno::Reference<text::XText> xContentControlText(
|
|
xContentControl, uno::UNO_QUERY);
|
|
if (!xContentControlText.is())
|
|
continue;
|
|
|
|
xContentControlText->setString(
|
|
OStringToOUString(aContent, RTL_TEXTENCODING_UTF8));
|
|
|
|
sal_Int32 iType = 0;
|
|
xContentControlProps->getPropertyValue(
|
|
UNO_NAME_CONTENT_CONTROL_TYPE)
|
|
>>= iType;
|
|
SwContentControlType aType
|
|
= static_cast<SwContentControlType>(iType);
|
|
|
|
// if we set the content of a checkbox, then we
|
|
// also set the checked state based on the content
|
|
if (aType == SwContentControlType::CHECKBOX)
|
|
{
|
|
OUString aCheckedContent;
|
|
xContentControlProps->getPropertyValue(
|
|
UNO_NAME_CHECKED_STATE)
|
|
>>= aCheckedContent;
|
|
bool bChecked = false;
|
|
if (aCheckedContent
|
|
== OStringToOUString(
|
|
aItem3.second.get_value<std::string>(),
|
|
RTL_TEXTENCODING_UTF8))
|
|
bChecked = true;
|
|
xContentControlProps->setPropertyValue(
|
|
UNO_NAME_CHECKED, uno::Any(bChecked));
|
|
}
|
|
else if (aType == SwContentControlType::PLAIN_TEXT
|
|
|| aType == SwContentControlType::RICH_TEXT
|
|
|| aType == SwContentControlType::DATE
|
|
|| aType == SwContentControlType::COMBO_BOX
|
|
|| aType
|
|
== SwContentControlType::DROP_DOWN_LIST)
|
|
{
|
|
// Set the placeholder
|
|
bool bPlaceHolder = aContent == "" ? true : false;
|
|
xContentControlProps->setPropertyValue(
|
|
UNO_NAME_SHOWING_PLACE_HOLDER,
|
|
uno::Any(bPlaceHolder));
|
|
if (bPlaceHolder)
|
|
{
|
|
OUString aPlaceHolderText;
|
|
switch (aType)
|
|
{
|
|
case SwContentControlType::PLAIN_TEXT:
|
|
case SwContentControlType::RICH_TEXT:
|
|
{
|
|
aPlaceHolderText = SwResId(
|
|
STR_CONTENT_CONTROL_PLACEHOLDER);
|
|
}
|
|
break;
|
|
case SwContentControlType::COMBO_BOX:
|
|
case SwContentControlType::DROP_DOWN_LIST:
|
|
{
|
|
aPlaceHolderText = SwResId(
|
|
STR_DROPDOWN_CONTENT_CONTROL_PLACEHOLDER);
|
|
}
|
|
break;
|
|
case SwContentControlType::DATE:
|
|
{
|
|
aPlaceHolderText = SwResId(
|
|
STR_DATE_CONTENT_CONTROL_PLACEHOLDER);
|
|
}
|
|
break;
|
|
default: // do nothing for picture and checkbox
|
|
break;
|
|
}
|
|
if (!aPlaceHolderText.isEmpty())
|
|
xContentControlText->setString(
|
|
aPlaceHolderText);
|
|
}
|
|
}
|
|
}
|
|
else if (aItem3.first == "checked")
|
|
{
|
|
bool bChecked
|
|
= (aItem3.second.get_value<std::string>() == "true")
|
|
? true
|
|
: false;
|
|
xContentControlProps->setPropertyValue(
|
|
UNO_NAME_CHECKED,
|
|
uno::Any(bChecked));
|
|
|
|
OUString aCheckContent;
|
|
xContentControlProps->getPropertyValue(
|
|
bChecked ? UNO_NAME_CHECKED_STATE
|
|
: UNO_NAME_UNCHECKED_STATE)
|
|
>>= aCheckContent;
|
|
uno::Reference<text::XText> xContentControlText(
|
|
xContentControl, uno::UNO_QUERY);
|
|
if (!xContentControlText.is())
|
|
continue;
|
|
xContentControlText->setString(aCheckContent);
|
|
}
|
|
else if (aItem3.first == "date")
|
|
{
|
|
std::string aDate
|
|
= aItem3.second.get_value<std::string>();
|
|
xContentControlProps->setPropertyValue(
|
|
UNO_NAME_CURRENT_DATE,
|
|
uno::Any(OStringToOUString(aDate,
|
|
RTL_TEXTENCODING_UTF8)));
|
|
}
|
|
else if (aItem3.first == "alias")
|
|
{
|
|
xContentControlProps->setPropertyValue(
|
|
UNO_NAME_ALIAS,
|
|
uno::Any(OStringToOUString(
|
|
aItem3.second.get_value<std::string>(),
|
|
RTL_TEXTENCODING_UTF8)));
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning(
|
|
"FillApi contentControl command not recognised: '"
|
|
+ aItem3.first + "'");
|
|
}
|
|
}
|
|
}
|
|
if (!bCCFound)
|
|
{
|
|
lcl_LogWarning("FillApi: No contentControl match the filter: '"
|
|
+ aItem2.first + "'");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
lcl_LogWarning(
|
|
"FillApi contentControl filter type not recognised: '"
|
|
+ aItem2.first + "'");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
OSL_ENSURE(false, "wrong dispatcher");
|
|
return;
|
|
}
|
|
}
|
|
|
|
void SwTextShell::GetState( SfxItemSet &rSet )
|
|
{
|
|
SwWrtShell &rSh = GetShell();
|
|
SfxWhichIter aIter( rSet );
|
|
sal_uInt16 nWhich = aIter.FirstWhich();
|
|
while ( nWhich )
|
|
{
|
|
const sal_uInt16 nSlotId = GetPool().GetSlotId(nWhich);
|
|
switch (nSlotId)
|
|
{
|
|
case FN_FORMAT_CURRENT_FOOTNOTE_DLG:
|
|
if( !rSh.IsCursorInFootnote() )
|
|
rSet.DisableItem( nWhich );
|
|
break;
|
|
|
|
case SID_LANGUAGE_STATUS:
|
|
{
|
|
// the value of used script types
|
|
OUString aScriptTypesInUse( OUString::number( static_cast<int>(rSh.GetScriptType()) ) );
|
|
|
|
// get keyboard language
|
|
OUString aKeyboardLang;
|
|
SwEditWin& rEditWin = GetView().GetEditWin();
|
|
LanguageType nLang = rEditWin.GetInputLanguage();
|
|
if (nLang != LANGUAGE_DONTKNOW && nLang != LANGUAGE_SYSTEM)
|
|
aKeyboardLang = SvtLanguageTable::GetLanguageString( nLang );
|
|
|
|
// get the language that is in use
|
|
OUString aCurrentLang = u"*"_ustr;
|
|
nLang = SwLangHelper::GetCurrentLanguage( rSh );
|
|
if (nLang != LANGUAGE_DONTKNOW)
|
|
{
|
|
aCurrentLang = SvtLanguageTable::GetLanguageString( nLang );
|
|
if (comphelper::LibreOfficeKit::isActive())
|
|
{
|
|
if (nLang == LANGUAGE_NONE)
|
|
{
|
|
aCurrentLang += ";-";
|
|
}
|
|
else
|
|
{
|
|
aCurrentLang += ";" + LanguageTag(nLang).getBcp47(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
// build sequence for status value
|
|
uno::Sequence< OUString > aSeq{ aCurrentLang,
|
|
aScriptTypesInUse,
|
|
aKeyboardLang,
|
|
SwLangHelper::GetTextForLanguageGuessing( rSh ) };
|
|
|
|
// set sequence as status value
|
|
SfxStringListItem aItem( SID_LANGUAGE_STATUS );
|
|
aItem.SetStringList( aSeq );
|
|
rSet.Put( aItem );
|
|
}
|
|
break;
|
|
|
|
case SID_THES:
|
|
{
|
|
// is there a valid selection to get text from?
|
|
OUString aText;
|
|
bool bValid = !rSh.HasSelection() ||
|
|
(rSh.IsSelOnePara() && !rSh.IsMultiSelection());
|
|
// prevent context menu from showing when cursor is not in or at the end of a word
|
|
// (GetCurWord will return the next word if there is none at the current position...)
|
|
const sal_Int16 nWordType = ::i18n::WordType::DICTIONARY_WORD;
|
|
bool bWord = rSh.IsInWord( nWordType ) || rSh.IsStartWord( nWordType ) || rSh.IsEndWord( nWordType );
|
|
if (bValid && bWord)
|
|
aText = rSh.HasSelection()? rSh.GetSelText() : rSh.GetCurWord();
|
|
|
|
LanguageType nLang = rSh.GetCurLang();
|
|
LanguageTag aLanguageTag( nLang);
|
|
const lang::Locale& aLocale( aLanguageTag.getLocale());
|
|
|
|
// disable "Thesaurus" context menu entry if there is nothing to look up
|
|
uno::Reference< linguistic2::XThesaurus > xThes( ::GetThesaurus() );
|
|
if (aText.isEmpty() ||
|
|
!xThes.is() || nLang == LANGUAGE_NONE || !xThes->hasLocale( aLocale ))
|
|
rSet.DisableItem( SID_THES );
|
|
else
|
|
{
|
|
// set word and locale to look up as status value
|
|
OUString aStatusVal = aText + "#" + aLanguageTag.getBcp47();
|
|
rSet.Put( SfxStringItem( SID_THES, aStatusVal ) );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case FN_NUMBER_NEWSTART :
|
|
if(!rSh.GetNumRuleAtCurrCursorPos())
|
|
rSet.DisableItem(nWhich);
|
|
else
|
|
rSet.Put(SfxBoolItem(FN_NUMBER_NEWSTART,
|
|
rSh.IsNumRuleStart()));
|
|
break;
|
|
|
|
case FN_EDIT_FORMULA:
|
|
case SID_CHARMAP:
|
|
case SID_CHARMAP_CONTROL:
|
|
{
|
|
const SelectionType nType = rSh.GetSelectionType();
|
|
if (!(nType & SelectionType::Text) &&
|
|
!(nType & SelectionType::Table) &&
|
|
!(nType & SelectionType::NumberList))
|
|
{
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
else if ( nWhich == FN_EDIT_FORMULA
|
|
&& rSh.CursorInsideInputField() )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case FN_INSERT_ENDNOTE:
|
|
case FN_INSERT_FOOTNOTE:
|
|
case FN_INSERT_FOOTNOTE_DLG:
|
|
{
|
|
const FrameTypeFlags nNoType =
|
|
FrameTypeFlags::FLY_ANY | FrameTypeFlags::HEADER | FrameTypeFlags::FOOTER | FrameTypeFlags::FOOTNOTE;
|
|
FrameTypeFlags eType = rSh.GetFrameType(nullptr, true);
|
|
bool bSplitFly = false;
|
|
if (eType & FrameTypeFlags::FLY_ATCNT)
|
|
{
|
|
SwContentFrame* pContentFrame = rSh.GetCurrFrame(/*bCalcFrame=*/false);
|
|
if (pContentFrame)
|
|
{
|
|
SwFlyFrame* pFlyFrame = pContentFrame->FindFlyFrame();
|
|
bSplitFly = pFlyFrame && pFlyFrame->IsFlySplitAllowed();
|
|
}
|
|
}
|
|
if (eType & nNoType && !bSplitFly)
|
|
rSet.DisableItem(nWhich);
|
|
|
|
if ( rSh.CursorInsideInputField() )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case SID_INSERTDOC:
|
|
case FN_INSERT_GLOSSARY:
|
|
case FN_EXPAND_GLOSSARY:
|
|
if ( rSh.CursorInsideInputField() )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
break;
|
|
|
|
case FN_INSERT_TABLE:
|
|
if ( rSh.CursorInsideInputField()
|
|
|| rSh.GetTableFormat()
|
|
|| (rSh.GetFrameType(nullptr,true) & FrameTypeFlags::FOOTNOTE) )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
break;
|
|
|
|
case FN_CALCULATE:
|
|
if ( !rSh.IsSelection() )
|
|
rSet.DisableItem(nWhich);
|
|
break;
|
|
case FN_GOTO_REFERENCE:
|
|
{
|
|
SwField *pField = rSh.GetCurField();
|
|
if ( !pField || (pField->GetTypeId() != SwFieldTypesEnum::GetRef) )
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case FN_AUTOFORMAT_AUTO:
|
|
{
|
|
rSet.Put( SfxBoolItem( nWhich, SvxAutoCorrCfg::Get().IsAutoFormatByInput() ));
|
|
}
|
|
break;
|
|
|
|
case SID_DEC_INDENT:
|
|
case SID_INC_INDENT:
|
|
{
|
|
//if the paragraph has bullet we'll do the following things:
|
|
//1: if the bullet level is the first level, disable the decrease-indent button
|
|
//2: if the bullet level is the last level, disable the increase-indent button
|
|
if ( rSh.GetNumRuleAtCurrCursorPos() && !rSh.HasReadonlySel() )
|
|
{
|
|
const sal_uInt8 nLevel = rSh.GetNumLevel();
|
|
if ( ( nLevel == ( MAXLEVEL - 1 ) && nWhich == SID_INC_INDENT )
|
|
|| ( nLevel == 0 && nWhich == SID_DEC_INDENT ) )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
sal_uInt16 nHtmlMode = ::GetHtmlMode( GetView().GetDocShell() );
|
|
nHtmlMode &= HTMLMODE_ON | HTMLMODE_SOME_STYLES;
|
|
if ( ( nHtmlMode == HTMLMODE_ON )
|
|
|| !rSh.IsMoveLeftMargin( SID_INC_INDENT == nWhich ) )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case FN_DEC_INDENT_OFFSET:
|
|
case FN_INC_INDENT_OFFSET:
|
|
{
|
|
sal_uInt16 nHtmlMode = ::GetHtmlMode(GetView().GetDocShell());
|
|
nHtmlMode &= HTMLMODE_ON|HTMLMODE_SOME_STYLES;
|
|
if( (nHtmlMode == HTMLMODE_ON) ||
|
|
!rSh.IsMoveLeftMargin( FN_INC_INDENT_OFFSET == nWhich,
|
|
false ))
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
break;
|
|
|
|
case SID_ATTR_CHAR_COLOR2:
|
|
{
|
|
SfxItemSet aSet( GetPool() );
|
|
rSh.GetCurAttr( aSet );
|
|
const SvxColorItem& aColorItem = aSet.Get(RES_CHRATR_COLOR);
|
|
rSet.Put( aColorItem.CloneSetWhich(SID_ATTR_CHAR_COLOR2) );
|
|
}
|
|
break;
|
|
case SID_ATTR_CHAR_BACK_COLOR:
|
|
case SID_ATTR_CHAR_COLOR_BACKGROUND:
|
|
{
|
|
// Always use the visible background
|
|
SfxItemSet aSet( GetPool() );
|
|
rSh.GetCurAttr( aSet );
|
|
const SvxBrushItem& aBrushItem = aSet.Get(RES_CHRATR_HIGHLIGHT);
|
|
if( aBrushItem.GetColor() != COL_TRANSPARENT )
|
|
{
|
|
rSet.Put(SvxColorItem(aBrushItem.GetColor(), aBrushItem.getComplexColor(), nWhich));
|
|
}
|
|
else
|
|
{
|
|
const SvxBrushItem& aBrushItem2 = aSet.Get(RES_CHRATR_BACKGROUND);
|
|
rSet.Put(SvxColorItem(aBrushItem2.GetColor(), aBrushItem2.getComplexColor(), nWhich));
|
|
}
|
|
}
|
|
break;
|
|
case SID_ATTR_CHAR_COLOR_BACKGROUND_EXT:
|
|
{
|
|
SwEditWin& rEdtWin = GetView().GetEditWin();
|
|
SwApplyTemplate* pApply = rEdtWin.GetApplyTemplate();
|
|
const sal_uInt32 nColWhich = pApply ? pApply->nColor : 0;
|
|
const bool bUseTemplate = nColWhich == SID_ATTR_CHAR_BACK_COLOR
|
|
|| nColWhich == SID_ATTR_CHAR_COLOR_BACKGROUND;
|
|
rSet.Put(SfxBoolItem(nWhich, bUseTemplate));
|
|
}
|
|
break;
|
|
case SID_ATTR_CHAR_COLOR_EXT:
|
|
{
|
|
SwEditWin& rEdtWin = GetView().GetEditWin();
|
|
SwApplyTemplate* pApply = rEdtWin.GetApplyTemplate();
|
|
rSet.Put(SfxBoolItem(nWhich, pApply && pApply->nColor == nWhich));
|
|
}
|
|
break;
|
|
case FN_SET_REMINDER:
|
|
case FN_INSERT_BOOKMARK:
|
|
if( rSh.IsTableMode()
|
|
|| rSh.CursorInsideInputField() )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
break;
|
|
|
|
case FN_INSERT_BREAK:
|
|
if ( rSh.HasReadonlySel()
|
|
&& !rSh.CursorInsideInputField() )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
break;
|
|
|
|
case FN_INSERT_BREAK_DLG:
|
|
case FN_INSERT_COLUMN_BREAK:
|
|
case FN_INSERT_PAGEBREAK:
|
|
if( rSh.CursorInsideInputField() || rSh.CursorInsideContentControl() )
|
|
{
|
|
rSet.DisableItem( nWhich );
|
|
}
|
|
break;
|
|
|
|
case FN_INSERT_PAGEHEADER:
|
|
case FN_INSERT_PAGEFOOTER:
|
|
if (comphelper::LibreOfficeKit::isActive())
|
|
{
|
|
bool bState = false;
|
|
bool bAllState = true;
|
|
bool bIsPhysical = false;
|
|
|
|
OUString aStyleName;
|
|
std::vector<OUString> aList;
|
|
static constexpr OUStringLiteral sPhysical(u"IsPhysical");
|
|
static constexpr OUStringLiteral sDisplay(u"DisplayName");
|
|
const OUString sHeaderOn(nWhich == FN_INSERT_PAGEHEADER ? u"HeaderIsOn"_ustr : u"FooterIsOn"_ustr);
|
|
|
|
rtl::Reference< SwXTextDocument > xSupplier(GetView().GetDocShell()->GetBaseModel());
|
|
if (xSupplier.is())
|
|
{
|
|
uno::Reference< XNameContainer > xContainer;
|
|
uno::Reference< XNameAccess > xFamilies = xSupplier->getStyleFamilies();
|
|
if (xFamilies->getByName(u"PageStyles"_ustr) >>= xContainer)
|
|
{
|
|
const uno::Sequence< OUString > aSeqNames = xContainer->getElementNames();
|
|
for (const auto& rName : aSeqNames)
|
|
{
|
|
aStyleName = rName;
|
|
uno::Reference<XPropertySet> xPropSet(xContainer->getByName(aStyleName), uno::UNO_QUERY);
|
|
if (xPropSet.is() && (xPropSet->getPropertyValue(sPhysical) >>= bIsPhysical) && bIsPhysical)
|
|
{
|
|
xPropSet->getPropertyValue(sDisplay) >>= aStyleName;
|
|
if ((xPropSet->getPropertyValue(sHeaderOn)>>= bState) && bState)
|
|
aList.push_back(aStyleName);
|
|
else
|
|
bState = false;
|
|
|
|
// Check if all entries have the same state
|
|
bAllState &= bState;
|
|
}
|
|
else
|
|
bIsPhysical = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bAllState && aList.size() > 1)
|
|
aList.push_back(u"_ALL_"_ustr);
|
|
|
|
rSet.Put(SfxStringListItem(nWhich, &aList));
|
|
}
|
|
else
|
|
{
|
|
rSet.Put( SfxObjectShellItem( nWhich, GetView().GetDocShell() ));
|
|
}
|
|
break;
|
|
case FN_TABLE_SORT_DIALOG:
|
|
case FN_SORTING_DLG:
|
|
if(!rSh.HasSelection() ||
|
|
(FN_TABLE_SORT_DIALOG == nWhich && !rSh.GetTableFormat()))
|
|
rSet.DisableItem( nWhich );
|
|
break;
|
|
|
|
case SID_RUBY_DIALOG:
|
|
{
|
|
if( !SvtCJKOptions::IsRubyEnabled()
|
|
|| rSh.CursorInsideInputField() )
|
|
{
|
|
GetView().GetViewFrame().GetBindings().SetVisibleState( nWhich, false );
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
else
|
|
GetView().GetViewFrame().GetBindings().SetVisibleState( nWhich, true );
|
|
}
|
|
break;
|
|
|
|
case SID_FM_TRANSLATE:
|
|
{
|
|
#if HAVE_FEATURE_CURL && !ENABLE_WASM_STRIP_EXTRA
|
|
if (!officecfg::Office::Common::Misc::ExperimentalMode::get()
|
|
&& !comphelper::LibreOfficeKit::isActive())
|
|
{
|
|
rSet.Put(SfxVisibilityItem(nWhich, false));
|
|
break;
|
|
}
|
|
std::optional<OUString> oDeeplAPIUrl = officecfg::Office::Linguistic::Translation::Deepl::ApiURL::get();
|
|
std::optional<OUString> oDeeplKey = officecfg::Office::Linguistic::Translation::Deepl::AuthKey::get();
|
|
if (!oDeeplAPIUrl || oDeeplAPIUrl->isEmpty() || !oDeeplKey || oDeeplKey->isEmpty())
|
|
{
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
#endif
|
|
}
|
|
break;
|
|
|
|
case SID_HYPERLINK_DIALOG:
|
|
if( GetView().GetDocShell()->IsReadOnly()
|
|
|| ( !GetView().GetViewFrame().HasChildWindow(nWhich)
|
|
&& rSh.HasReadonlySel() )
|
|
|| rSh.CursorInsideInputField() )
|
|
{
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
else
|
|
{
|
|
rSet.Put(SfxBoolItem( nWhich, nullptr != GetView().GetViewFrame().GetChildWindow( nWhich ) ));
|
|
}
|
|
break;
|
|
|
|
case SID_EDIT_HYPERLINK:
|
|
{
|
|
if (!rSh.HasReadonlySel())
|
|
{
|
|
SfxItemSetFixed<RES_TXTATR_INETFMT, RES_TXTATR_INETFMT> aSet(GetPool());
|
|
rSh.GetCurAttr(aSet);
|
|
if (SfxItemState::SET <= aSet.GetItemState(RES_TXTATR_INETFMT))
|
|
break;
|
|
|
|
// is the cursor at the beginning of a hyperlink?
|
|
const SwTextNode* pTextNd = rSh.GetCursor()->GetPointNode().GetTextNode();
|
|
if (pTextNd && !rSh.HasSelection())
|
|
{
|
|
const sal_Int32 nIndex = rSh.GetCursor()->Start()->GetContentIndex();
|
|
const SwTextAttr* pINetFmt
|
|
= pTextNd->GetTextAttrAt(nIndex, RES_TXTATR_INETFMT);
|
|
if (pINetFmt && !pINetFmt->GetINetFormat().GetValue().isEmpty())
|
|
break;
|
|
}
|
|
}
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case SID_REMOVE_HYPERLINK:
|
|
{
|
|
if (!rSh.HasReadonlySel())
|
|
{
|
|
SfxItemSetFixed<RES_TXTATR_INETFMT, RES_TXTATR_INETFMT> aSet(GetPool());
|
|
rSh.GetCurAttr(aSet);
|
|
|
|
// If a hyperlink is selected, either alone or along with other text...
|
|
if (SfxItemState::SET <= aSet.GetItemState(RES_TXTATR_INETFMT)
|
|
|| aSet.GetItemState(RES_TXTATR_INETFMT) == SfxItemState::INVALID)
|
|
{
|
|
break;
|
|
}
|
|
|
|
// is the cursor at the beginning of a hyperlink?
|
|
const SwTextNode* pTextNd = rSh.GetCursor()->GetPointNode().GetTextNode();
|
|
if (pTextNd && !rSh.HasSelection())
|
|
{
|
|
const sal_Int32 nIndex = rSh.GetCursor()->Start()->GetContentIndex();
|
|
const SwTextAttr* pINetFmt
|
|
= pTextNd->GetTextAttrAt(nIndex, RES_TXTATR_INETFMT);
|
|
if (pINetFmt && !pINetFmt->GetINetFormat().GetValue().isEmpty())
|
|
break;
|
|
}
|
|
}
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case SID_TRANSLITERATE_HALFWIDTH:
|
|
case SID_TRANSLITERATE_FULLWIDTH:
|
|
case SID_TRANSLITERATE_HIRAGANA:
|
|
case SID_TRANSLITERATE_KATAKANA:
|
|
{
|
|
if(!SvtCJKOptions::IsChangeCaseMapEnabled())
|
|
{
|
|
GetView().GetViewFrame().GetBindings().SetVisibleState( nWhich, false );
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
else
|
|
GetView().GetViewFrame().GetBindings().SetVisibleState( nWhich, true );
|
|
}
|
|
break;
|
|
case FN_READONLY_SELECTION_MODE :
|
|
if(!GetView().GetDocShell()->IsReadOnly())
|
|
rSet.DisableItem( nWhich );
|
|
else
|
|
{
|
|
rSet.Put(SfxBoolItem(nWhich, rSh.GetViewOptions()->IsSelectionInReadonly()));
|
|
}
|
|
break;
|
|
case FN_SELECTION_MODE_DEFAULT:
|
|
case FN_SELECTION_MODE_BLOCK :
|
|
rSet.Put(SfxBoolItem(nWhich, (nWhich == FN_SELECTION_MODE_DEFAULT) != rSh.IsBlockMode()));
|
|
break;
|
|
case SID_COPY_HYPERLINK_LOCATION:
|
|
case SID_OPEN_HYPERLINK:
|
|
{
|
|
SfxItemSetFixed<RES_TXTATR_INETFMT, RES_TXTATR_INETFMT> aSet(GetPool());
|
|
rSh.GetCurAttr(aSet);
|
|
if (SfxItemState::SET <= aSet.GetItemState(RES_TXTATR_INETFMT, false))
|
|
break;
|
|
|
|
// is the cursor at the beginning of a hyperlink?
|
|
const SwTextNode* pTextNd = rSh.GetCursor()->GetPointNode().GetTextNode();
|
|
if (pTextNd && !rSh.HasSelection())
|
|
{
|
|
const sal_Int32 nIndex = rSh.GetCursor()->Start()->GetContentIndex();
|
|
const SwTextAttr* pINetFmt = pTextNd->GetTextAttrAt(nIndex, RES_TXTATR_INETFMT);
|
|
if (pINetFmt && !pINetFmt->GetINetFormat().GetValue().isEmpty())
|
|
break;
|
|
}
|
|
|
|
SwField* pField = rSh.GetCurField();
|
|
if (pField && pField->GetTyp()->Which() == SwFieldIds::TableOfAuthorities)
|
|
{
|
|
const auto& rAuthorityField = *static_cast<const SwAuthorityField*>(pField);
|
|
if (auto targetType = rAuthorityField.GetTargetType();
|
|
targetType == SwAuthorityField::TargetType::UseDisplayURL
|
|
|| targetType == SwAuthorityField::TargetType::UseTargetURL)
|
|
{
|
|
// Check if the Bibliography entry has a target URL
|
|
if (rAuthorityField.GetAbsoluteURL().getLength() > 0)
|
|
break;
|
|
}
|
|
}
|
|
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case FN_OPEN_LOCAL_URL:
|
|
{
|
|
if (GetLocalURL(rSh).isEmpty())
|
|
{
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
}
|
|
break;
|
|
case SID_OPEN_SMARTTAGMENU:
|
|
{
|
|
std::vector< OUString > aSmartTagTypes;
|
|
uno::Sequence< uno::Reference< container::XStringKeyMap > > aStringKeyMaps;
|
|
uno::Reference<text::XTextRange> xRange;
|
|
|
|
rSh.GetSmartTagTerm( aSmartTagTypes, aStringKeyMaps, xRange );
|
|
|
|
if ( xRange.is() && !aSmartTagTypes.empty() )
|
|
{
|
|
uno::Sequence < uno::Sequence< uno::Reference< smarttags::XSmartTagAction > > > aActionComponentsSequence;
|
|
uno::Sequence < uno::Sequence< sal_Int32 > > aActionIndicesSequence;
|
|
|
|
const SmartTagMgr& rSmartTagMgr = SwSmartTagMgr::Get();
|
|
rSmartTagMgr.GetActionSequences( aSmartTagTypes,
|
|
aActionComponentsSequence,
|
|
aActionIndicesSequence );
|
|
|
|
uno::Reference <frame::XController> xController = GetView().GetController();
|
|
lang::Locale aLocale( SW_BREAKITER()->GetLocale( GetAppLanguageTag() ) );
|
|
const OUString& aApplicationName( rSmartTagMgr.GetApplicationName() );
|
|
const OUString aRangeText = xRange->getString();
|
|
|
|
const SvxSmartTagItem aItem( SID_OPEN_SMARTTAGMENU,
|
|
aActionComponentsSequence,
|
|
aActionIndicesSequence,
|
|
aStringKeyMaps,
|
|
xRange,
|
|
std::move(xController),
|
|
std::move(aLocale),
|
|
aApplicationName,
|
|
aRangeText );
|
|
|
|
rSet.Put( aItem );
|
|
}
|
|
else
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
|
|
case FN_NUM_NUMBERING_ON:
|
|
rSet.Put(SfxBoolItem(FN_NUM_NUMBERING_ON,rSh.SelectionHasNumber()));
|
|
break;
|
|
|
|
case FN_NUM_BULLET_ON:
|
|
rSet.Put(SfxBoolItem(FN_NUM_BULLET_ON,rSh.SelectionHasBullet()));
|
|
break;
|
|
|
|
case FN_NUM_BULLET_OFF:
|
|
rSet.Put(SfxBoolItem(FN_NUM_BULLET_OFF, !rSh.GetNumRuleAtCurrCursorPos() &&
|
|
!rSh.GetNumRuleAtCurrentSelection()));
|
|
break;
|
|
|
|
case FN_SVX_SET_OUTLINE:
|
|
{
|
|
NBOTypeMgrBase* pOutline = NBOutlineTypeMgrFact::CreateInstance(NBOType::Outline);
|
|
auto pCurRule = const_cast<SwNumRule*>(rSh.GetNumRuleAtCurrCursorPos());
|
|
if (pOutline && pCurRule)
|
|
{
|
|
SvxNumRule aSvxRule = pCurRule->MakeSvxNumRule();
|
|
const sal_uInt16 nIndex = pOutline->GetNBOIndexForNumRule(aSvxRule, 0);
|
|
rSet.Put(SfxBoolItem(FN_SVX_SET_OUTLINE, nIndex < USHRT_MAX));
|
|
}
|
|
break;
|
|
}
|
|
case FN_BUL_NUM_RULE_INDEX:
|
|
case FN_NUM_NUM_RULE_INDEX:
|
|
case FN_OUTLINE_RULE_INDEX:
|
|
{
|
|
SwNumRule* pCurRule = const_cast<SwNumRule*>(GetShell().GetNumRuleAtCurrCursorPos());
|
|
if( pCurRule )
|
|
{
|
|
sal_uInt16 nActNumLvl = GetShell().GetNumLevel();
|
|
if( nActNumLvl < MAXLEVEL )
|
|
{
|
|
nActNumLvl = 1<<nActNumLvl;
|
|
}
|
|
SvxNumRule aSvxRule = pCurRule->MakeSvxNumRule();
|
|
if ( GetShell().HasBullet())
|
|
{
|
|
rSet.Put(SfxUInt16Item(FN_BUL_NUM_RULE_INDEX, USHRT_MAX));
|
|
rSet.Put(SfxUInt16Item(FN_NUM_NUM_RULE_INDEX, USHRT_MAX));
|
|
NBOTypeMgrBase* pBullets = NBOutlineTypeMgrFact::CreateInstance(NBOType::Bullets);
|
|
if ( pBullets )
|
|
{
|
|
const sal_uInt16 nBulIndex = pBullets->GetNBOIndexForNumRule(aSvxRule,nActNumLvl);
|
|
rSet.Put(SfxUInt16Item(FN_BUL_NUM_RULE_INDEX,nBulIndex));
|
|
}
|
|
}else if ( GetShell().HasNumber() )
|
|
{
|
|
rSet.Put(SfxUInt16Item(FN_BUL_NUM_RULE_INDEX, USHRT_MAX));
|
|
rSet.Put(SfxUInt16Item(FN_NUM_NUM_RULE_INDEX, USHRT_MAX));
|
|
NBOTypeMgrBase* pNumbering = NBOutlineTypeMgrFact::CreateInstance(NBOType::Numbering);
|
|
if ( pNumbering )
|
|
{
|
|
const sal_uInt16 nBulIndex = pNumbering->GetNBOIndexForNumRule(aSvxRule,nActNumLvl);
|
|
rSet.Put(SfxUInt16Item(FN_NUM_NUM_RULE_INDEX,nBulIndex));
|
|
}
|
|
}
|
|
|
|
if ( nWhich == FN_OUTLINE_RULE_INDEX )
|
|
{
|
|
rSet.Put(SfxUInt16Item(FN_OUTLINE_RULE_INDEX, USHRT_MAX));
|
|
NBOTypeMgrBase* pOutline = NBOutlineTypeMgrFact::CreateInstance(NBOType::Outline);
|
|
if ( pOutline )
|
|
{
|
|
const sal_uInt16 nIndex = pOutline->GetNBOIndexForNumRule(aSvxRule,nActNumLvl);
|
|
rSet.Put(SfxUInt16Item(FN_OUTLINE_RULE_INDEX,nIndex));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case FN_BUL_GET_DOC_BULLETS:
|
|
{
|
|
std::set<OUString> aBulletsSet = rSh.GetUsedBullets();
|
|
std::vector<OUString> aBullets;
|
|
std::copy(aBulletsSet.begin(), aBulletsSet.end(), std::back_inserter(aBullets));
|
|
SfxStringListItem aItem(FN_BUL_GET_DOC_BULLETS);
|
|
uno::Sequence<OUString> aSeq(aBullets.data(),
|
|
static_cast<sal_Int32>(aBullets.size()));
|
|
aItem.SetStringList(aSeq);
|
|
rSet.Put(aItem);
|
|
}
|
|
break;
|
|
case FN_NUM_CONTINUE:
|
|
{
|
|
// #i86492#
|
|
// Search also for bullet list
|
|
OUString aDummy;
|
|
const SwNumRule* pRule =
|
|
rSh.SearchNumRule( true, aDummy );
|
|
if ( !pRule )
|
|
{
|
|
pRule = rSh.SearchNumRule( false, aDummy );
|
|
}
|
|
if ( !pRule )
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case SID_INSERT_RLM :
|
|
case SID_INSERT_LRM :
|
|
{
|
|
bool bEnabled = SvtCTLOptions::IsCTLFontEnabled();
|
|
GetView().GetViewFrame().GetBindings().SetVisibleState( nWhich, bEnabled );
|
|
if(!bEnabled)
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case SID_FM_CTL_PROPERTIES:
|
|
{
|
|
bool bDisable = false;
|
|
|
|
// First get the state from the form shell
|
|
SfxItemSetFixed<SID_FM_CTL_PROPERTIES, SID_FM_CTL_PROPERTIES> aSet(GetShell().GetAttrPool());
|
|
aSet.Put(SfxBoolItem( SID_FM_CTL_PROPERTIES, true ));
|
|
GetShell().GetView().GetFormShell()->GetState( aSet );
|
|
|
|
if(SfxItemState::DISABLED == aSet.GetItemState(SID_FM_CTL_PROPERTIES))
|
|
{
|
|
bDisable = true;
|
|
}
|
|
|
|
// Enable it if we have a valid object other than what form shell knows
|
|
SwPosition aPos(*GetShell().GetCursor()->GetPoint());
|
|
sw::mark::Fieldmark* pFieldBM = GetShell().getIDocumentMarkAccess()->getInnerFieldmarkFor(aPos);
|
|
if ( !pFieldBM && aPos.GetContentIndex() > 0)
|
|
{
|
|
aPos.AdjustContent(-1);
|
|
pFieldBM = GetShell().getIDocumentMarkAccess()->getInnerFieldmarkFor(aPos);
|
|
}
|
|
if ( pFieldBM && (pFieldBM->GetFieldname() == ODF_FORMDROPDOWN || pFieldBM->GetFieldname() == ODF_FORMDATE) )
|
|
{
|
|
bDisable = false;
|
|
}
|
|
|
|
if(bDisable)
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
break;
|
|
case SID_COPY:
|
|
case SID_CUT:
|
|
{
|
|
if (GetObjectShell()->isContentExtractionLocked())
|
|
rSet.DisableItem(nWhich);
|
|
break;
|
|
}
|
|
case FN_PROTECT_FIELDS:
|
|
case FN_PROTECT_BOOKMARKS:
|
|
{
|
|
DocumentSettingId aSettingId = nWhich == FN_PROTECT_FIELDS
|
|
? DocumentSettingId::PROTECT_FIELDS
|
|
: DocumentSettingId::PROTECT_BOOKMARKS;
|
|
bool bProtected = rSh.getIDocumentSettingAccess().get(aSettingId);
|
|
rSet.Put(SfxBoolItem(nWhich, bProtected));
|
|
}
|
|
break;
|
|
case FN_DELETE_CONTENT_CONTROL:
|
|
case FN_CONTENT_CONTROL_PROPERTIES:
|
|
{
|
|
if (!GetShell().CursorInsideContentControl())
|
|
{
|
|
rSet.DisableItem(nWhich);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
nWhich = aIter.NextWhich();
|
|
}
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|