1
0
Fork 0
libreoffice/sc/source/ui/view/viewutil.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

431 lines
14 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 <scitems.hxx>
#include <sfx2/bindings.hxx>
#include <sfx2/viewsh.hxx>
#include <sfx2/dispatch.hxx>
#include <editeng/fontitem.hxx>
#include <editeng/langitem.hxx>
#include <editeng/scripttypeitem.hxx>
#include <i18nutil/transliteration.hxx>
#include <svl/itempool.hxx>
#include <svl/itemset.hxx>
#include <svl/cjkoptions.hxx>
#include <svl/ctloptions.hxx>
#include <vcl/svapp.hxx>
#include <vcl/settings.hxx>
#include <sfx2/objsh.hxx>
#include <sfx2/viewfrm.hxx>
#include <svl/eitem.hxx>
#include <osl/diagnose.h>
#include <viewutil.hxx>
#include <chgtrack.hxx>
#include <chgviset.hxx>
#include <markdata.hxx>
#include <document.hxx>
#include <tabvwsh.hxx>
#include <svx/svxdlg.hxx>
#include <svx/svxids.hrc>
#include <memory>
void ScViewUtil::PutItemScript( SfxItemSet& rShellSet, const SfxItemSet& rCoreSet,
sal_uInt16 nWhichId, SvtScriptType nScript )
{
// take the effective item from rCoreSet according to nScript
// and put in rShellSet under the (base) nWhichId
SfxItemPool& rPool = *rShellSet.GetPool();
SvxScriptSetItem aSetItem( rPool.GetSlotId(nWhichId), rPool );
// use PutExtended with eDefaultAs = SfxItemState::SET, so defaults from rCoreSet
// (document pool) are read and put into rShellSet (MessagePool)
aSetItem.GetItemSet().PutExtended( rCoreSet, SfxItemState::INVALID, SfxItemState::SET );
const SfxPoolItem* pI = aSetItem.GetItemOfScript( nScript );
if (pI)
{
rShellSet.Put( pI->CloneSetWhich(nWhichId) );
}
else
rShellSet.InvalidateItem( nWhichId );
}
LanguageType ScViewUtil::GetEffLanguage( ScDocument& rDoc, const ScAddress& rPos )
{
// used for thesaurus
SvtScriptType nScript = rDoc.GetScriptType(rPos.Col(), rPos.Row(), rPos.Tab());
sal_uInt16 nWhich = ( nScript == SvtScriptType::ASIAN ) ? ATTR_CJK_FONT_LANGUAGE :
( ( nScript == SvtScriptType::COMPLEX ) ? ATTR_CTL_FONT_LANGUAGE : ATTR_FONT_LANGUAGE );
const SfxPoolItem* pItem = rDoc.GetAttr( rPos.Col(), rPos.Row(), rPos.Tab(), nWhich);
const SvxLanguageItem* pLangIt = dynamic_cast<const SvxLanguageItem*>( pItem );
LanguageType eLnge;
if (pLangIt)
{
eLnge = pLangIt->GetValue();
if (eLnge == LANGUAGE_DONTKNOW) //! can this happen?
{
LanguageType eLatin, eCjk, eCtl;
rDoc.GetLanguage( eLatin, eCjk, eCtl );
eLnge = ( nScript == SvtScriptType::ASIAN ) ? eCjk :
( ( nScript == SvtScriptType::COMPLEX ) ? eCtl : eLatin );
}
}
else
eLnge = LANGUAGE_ENGLISH_US;
if ( eLnge == LANGUAGE_SYSTEM )
eLnge = Application::GetSettings().GetLanguageTag().getLanguageType(); // never use SYSTEM for spelling
return eLnge;
}
TransliterationFlags ScViewUtil::GetTransliterationType( sal_uInt16 nSlotID )
{
TransliterationFlags nType = TransliterationFlags::NONE;
switch ( nSlotID )
{
case SID_TRANSLITERATE_SENTENCE_CASE:
nType = TransliterationFlags::SENTENCE_CASE;
break;
case SID_TRANSLITERATE_TITLE_CASE:
nType = TransliterationFlags::TITLE_CASE;
break;
case SID_TRANSLITERATE_TOGGLE_CASE:
nType = TransliterationFlags::TOGGLE_CASE;
break;
case SID_TRANSLITERATE_UPPER:
nType = TransliterationFlags::LOWERCASE_UPPERCASE;
break;
case SID_TRANSLITERATE_LOWER:
nType = TransliterationFlags::UPPERCASE_LOWERCASE;
break;
case SID_TRANSLITERATE_HALFWIDTH:
nType = TransliterationFlags::FULLWIDTH_HALFWIDTH;
break;
case SID_TRANSLITERATE_FULLWIDTH:
nType = TransliterationFlags::HALFWIDTH_FULLWIDTH;
break;
case SID_TRANSLITERATE_HIRAGANA:
nType = TransliterationFlags::KATAKANA_HIRAGANA;
break;
case SID_TRANSLITERATE_KATAKANA:
nType = TransliterationFlags::HIRAGANA_KATAKANA;
break;
}
return nType;
}
bool ScViewUtil::IsActionShown( const ScChangeAction& rAction,
const ScChangeViewSettings& rSettings,
ScDocument& rDocument )
{
// discarded are displayed as inverted accepted action, because of this
// order of ShowRejected/ShowAccepted is important
if ( !rSettings.IsShowRejected() && rAction.IsRejecting() )
return false;
if ( !rSettings.IsShowAccepted() && rAction.IsAccepted() && !rAction.IsRejecting() )
return false;
if ( rSettings.HasAuthor() && rAction.GetUser() != rSettings.GetTheAuthorToShow() )
return false;
if ( rSettings.HasComment() )
{
OUString aTmp = rAction.GetDescription(rDocument);
OUString aComStr = rAction.GetComment() + " (" + aTmp + ")";
if(!rSettings.IsValidComment(&aComStr))
return false;
}
if ( rSettings.HasRange() )
if ( !rSettings.GetTheRangeList().Intersects( rAction.GetBigRange().MakeRange( rDocument ) ) )
return false;
if (rSettings.HasDate() && rSettings.GetTheDateMode() != SvxRedlinDateMode::NONE)
{
DateTime aDateTime = rAction.GetDateTime();
const DateTime& rFirst = rSettings.GetTheFirstDateTime();
const DateTime& rLast = rSettings.GetTheLastDateTime();
switch ( rSettings.GetTheDateMode() )
{ // corresponds with ScHighlightChgDlg::OKBtnHdl
case SvxRedlinDateMode::BEFORE:
if ( aDateTime > rFirst )
return false;
break;
case SvxRedlinDateMode::SINCE:
if ( aDateTime < rFirst )
return false;
break;
case SvxRedlinDateMode::EQUAL:
case SvxRedlinDateMode::BETWEEN:
if ( aDateTime < rFirst || aDateTime > rLast )
return false;
break;
case SvxRedlinDateMode::NOTEQUAL:
if ( aDateTime >= rFirst && aDateTime <= rLast )
return false;
break;
case SvxRedlinDateMode::SAVE:
{
ScChangeTrack* pTrack = rDocument.GetChangeTrack();
if ( !pTrack || pTrack->GetLastSavedActionNumber() >=
rAction.GetActionNumber() )
return false;
}
break;
default:
{
// added to avoid warnings
}
}
}
if ( rSettings.HasActionRange() )
{
sal_uLong nAction = rAction.GetActionNumber();
sal_uLong nFirstAction;
sal_uLong nLastAction;
rSettings.GetTheActionRange( nFirstAction, nLastAction );
if ( nAction < nFirstAction || nAction > nLastAction )
{
return false;
}
}
return true;
}
void ScViewUtil::UnmarkFiltered( ScMarkData& rMark, const ScDocument& rDoc )
{
rMark.MarkToMulti();
const ScRange& aMultiArea = rMark.GetMultiMarkArea();
SCCOL nStartCol = aMultiArea.aStart.Col();
SCROW nStartRow = aMultiArea.aStart.Row();
SCCOL nEndCol = aMultiArea.aEnd.Col();
SCROW nEndRow = aMultiArea.aEnd.Row();
bool bChanged = false;
for (const SCTAB& nTab : rMark)
{
for (SCROW nRow = nStartRow; nRow <= nEndRow; ++nRow)
{
SCROW nLastRow = nRow;
if (rDoc.RowFiltered(nRow, nTab, nullptr, &nLastRow))
{
// use nStartCol/nEndCol, so the multi mark area isn't extended to all columns
// (visible in repaint for indentation)
rMark.SetMultiMarkArea(
ScRange(nStartCol, nRow, nTab, nEndCol, nLastRow, nTab), false);
bChanged = true;
nRow = nLastRow;
}
}
}
if ( bChanged && !rMark.HasAnyMultiMarks() )
rMark.ResetMark();
rMark.MarkToSimple();
}
bool ScViewUtil::FitToUnfilteredRows( ScRange & rRange, const ScDocument& rDoc, size_t nRows )
{
SCTAB nTab = rRange.aStart.Tab();
bool bOneTabOnly = (nTab == rRange.aEnd.Tab());
// Always fit the range on its first sheet.
OSL_ENSURE( bOneTabOnly, "ScViewUtil::ExtendToUnfilteredRows: works only on one sheet");
SCROW nStartRow = rRange.aStart.Row();
SCROW nLastRow = rDoc.LastNonFilteredRow(nStartRow, rDoc.MaxRow(), nTab);
if (rDoc.ValidRow(nLastRow))
rRange.aEnd.SetRow(nLastRow);
SCROW nCount = rDoc.CountNonFilteredRows(nStartRow, rDoc.MaxRow(), nTab);
return static_cast<size_t>(nCount) == nRows && bOneTabOnly;
}
bool ScViewUtil::HasFiltered( const ScRange& rRange, const ScDocument& rDoc )
{
SCROW nStartRow = rRange.aStart.Row();
SCROW nEndRow = rRange.aEnd.Row();
for (SCTAB nTab=rRange.aStart.Tab(); nTab<=rRange.aEnd.Tab(); nTab++)
{
if (rDoc.HasFilteredRows(nStartRow, nEndRow, nTab))
return true;
}
return false;
}
void ScViewUtil::HideDisabledSlot( SfxItemSet& rSet, SfxBindings& rBindings, sal_uInt16 nSlotId )
{
SvtCTLOptions aCTLOptions;
bool bEnabled = true;
switch( nSlotId )
{
case SID_CHINESE_CONVERSION:
case SID_HANGUL_HANJA_CONVERSION:
bEnabled = SvtCJKOptions::IsAnyEnabled();
break;
case SID_TRANSLITERATE_HALFWIDTH:
case SID_TRANSLITERATE_FULLWIDTH:
case SID_TRANSLITERATE_HIRAGANA:
case SID_TRANSLITERATE_KATAKANA:
bEnabled = SvtCJKOptions::IsChangeCaseMapEnabled();
break;
case SID_INSERT_RLM:
case SID_INSERT_LRM:
bEnabled = SvtCTLOptions::IsCTLFontEnabled();
break;
default:
OSL_FAIL( "ScViewUtil::HideDisabledSlot - unknown slot ID" );
return;
}
rBindings.SetVisibleState( nSlotId, bEnabled );
if( !bEnabled )
rSet.DisableItem( nSlotId );
}
void ScViewUtil::ExecuteCharMap(const SvxFontItem& rOldFont,
const ScTabViewShell& rShell)
{
SvxAbstractDialogFactory* pFact = SvxAbstractDialogFactory::Create();
SfxViewFrame& rFrame = rShell.GetViewFrame();
SfxAllItemSet aSet( rFrame.GetObjectShell()->GetPool() );
aSet.Put( SfxBoolItem( FN_PARAM_1, false ) );
aSet.Put( SvxFontItem( rOldFont.GetFamily(), rOldFont.GetFamilyName(), rOldFont.GetStyleName(), rOldFont.GetPitch(), rOldFont.GetCharSet(), aSet.GetPool()->GetWhichIDFromSlotID( SID_ATTR_CHAR_FONT ) ) );
auto xFrame = rFrame.GetFrame().GetFrameInterface();
VclPtr<SfxAbstractDialog> pDlg(pFact->CreateCharMapDialog(rShell.GetFrameWeld(), aSet, xFrame));
pDlg->StartExecuteAsync(
[pDlg] (sal_Int32 /*nResult*/)->void
{
pDlg->disposeOnce();
}
);
}
bool ScViewUtil::IsFullScreen( const SfxViewShell& rViewShell )
{
SfxBindings& rBindings = rViewShell.GetViewFrame().GetBindings();
std::unique_ptr<SfxBoolItem> pItem;
bool bIsFullScreen = false;
if (rBindings.QueryState( SID_WIN_FULLSCREEN, pItem ) >= SfxItemState::DEFAULT)
bIsFullScreen = pItem->GetValue();
return bIsFullScreen;
}
void ScViewUtil::SetFullScreen( const SfxViewShell& rViewShell, bool bSet )
{
if( IsFullScreen( rViewShell ) != bSet )
{
SfxBoolItem aItem( SID_WIN_FULLSCREEN, bSet );
rViewShell.GetDispatcher()->ExecuteList(SID_WIN_FULLSCREEN,
SfxCallMode::RECORD, { &aItem });
}
}
ScUpdateRect::ScUpdateRect( SCCOL nX1, SCROW nY1, SCCOL nX2, SCROW nY2 )
: nNewStartX(0)
, nNewStartY(0)
, nNewEndX(0)
, nNewEndY(0)
{
PutInOrder( nX1, nX2 );
PutInOrder( nY1, nY2 );
nOldStartX = nX1;
nOldStartY = nY1;
nOldEndX = nX2;
nOldEndY = nY2;
}
void ScUpdateRect::SetNew( SCCOL nX1, SCROW nY1, SCCOL nX2, SCROW nY2 )
{
PutInOrder( nX1, nX2 );
PutInOrder( nY1, nY2 );
nNewStartX = nX1;
nNewStartY = nY1;
nNewEndX = nX2;
nNewEndY = nY2;
}
bool ScUpdateRect::GetDiff( SCCOL& rX1, SCROW& rY1, SCCOL& rX2, SCROW& rY2 )
{
if ( nNewStartX == nOldStartX && nNewEndX == nOldEndX &&
nNewStartY == nOldStartY && nNewEndY == nOldEndY )
{
rX1 = nNewStartX;
rY1 = nNewStartY;
rX2 = nNewStartX;
rY2 = nNewStartY;
return false;
}
rX1 = std::min(nNewStartX,nOldStartX);
rY1 = std::min(nNewStartY,nOldStartY);
rX2 = std::max(nNewEndX,nOldEndX);
rY2 = std::max(nNewEndY,nOldEndY);
if ( nNewStartX == nOldStartX && nNewEndX == nOldEndX )
{
if ( nNewStartY == nOldStartY )
{
rY1 = std::min( nNewEndY, nOldEndY );
rY2 = std::max( nNewEndY, nOldEndY );
}
else if ( nNewEndY == nOldEndY )
{
rY1 = std::min( nNewStartY, nOldStartY );
rY2 = std::max( nNewStartY, nOldStartY );
}
}
else if ( nNewStartY == nOldStartY && nNewEndY == nOldEndY )
{
if ( nNewStartX == nOldStartX )
{
rX1 = std::min( nNewEndX, nOldEndX );
rX2 = std::max( nNewEndX, nOldEndX );
}
else if ( nNewEndX == nOldEndX )
{
rX1 = std::min( nNewStartX, nOldStartX );
rX2 = std::max( nNewStartX, nOldStartX );
}
}
return true;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */