/************************************************************************* * * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * Copyright 2000, 2010 Oracle and/or its affiliates. * * OpenOffice.org - a multi-platform office productivity suite * * This file is part of OpenOffice.org. * * OpenOffice.org is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License version 3 * only, as published by the Free Software Foundation. * * OpenOffice.org is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License version 3 for more details * (a copy is included in the LICENSE file that accompanied this code). * * You should have received a copy of the GNU Lesser General Public License * version 3 along with OpenOffice.org. If not, see * * for a copy of the LGPLv3 License. * ************************************************************************/ // MARKER(update_precomp.py): autogen include statement, do not remove #include "precompiled_sc.hxx" // INCLUDE --------------------------------------------------------------- #include #include "scitems.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "viewutil.hxx" #include "global.hxx" #include "chgtrack.hxx" #include "chgviset.hxx" #include "markdata.hxx" #include //CHINA001 #include //CHINA001 // STATIC DATA ----------------------------------------------------------- //================================================================== // static void ScViewUtil::PutItemScript( SfxItemSet& rShellSet, const SfxItemSet& rCoreSet, sal_uInt16 nWhichId, sal_uInt16 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 = SFX_ITEM_SET, so defaults from rCoreSet // (document pool) are read and put into rShellSet (MessagePool) aSetItem.GetItemSet().PutExtended( rCoreSet, SFX_ITEM_DONTCARE, SFX_ITEM_SET ); const SfxPoolItem* pI = aSetItem.GetItemOfScript( nScript ); if (pI) rShellSet.Put( *pI, nWhichId ); else rShellSet.InvalidateItem( nWhichId ); } // static sal_uInt16 ScViewUtil::GetEffLanguage( ScDocument* pDoc, const ScAddress& rPos ) { // used for thesaurus sal_uInt8 nScript = pDoc->GetScriptType( rPos.Col(), rPos.Row(), rPos.Tab() ); sal_uInt16 nWhich = ( nScript == SCRIPTTYPE_ASIAN ) ? ATTR_CJK_FONT_LANGUAGE : ( ( nScript == SCRIPTTYPE_COMPLEX ) ? ATTR_CTL_FONT_LANGUAGE : ATTR_FONT_LANGUAGE ); const SfxPoolItem* pItem = pDoc->GetAttr( rPos.Col(), rPos.Row(), rPos.Tab(), nWhich); SvxLanguageItem* pLangIt = PTR_CAST( SvxLanguageItem, pItem ); LanguageType eLnge; if (pLangIt) { eLnge = (LanguageType) pLangIt->GetValue(); if (eLnge == LANGUAGE_DONTKNOW) //! can this happen? { LanguageType eLatin, eCjk, eCtl; pDoc->GetLanguage( eLatin, eCjk, eCtl ); eLnge = ( nScript == SCRIPTTYPE_ASIAN ) ? eCjk : ( ( nScript == SCRIPTTYPE_COMPLEX ) ? eCtl : eLatin ); } } else eLnge = LANGUAGE_ENGLISH_US; if ( eLnge == LANGUAGE_SYSTEM ) eLnge = Application::GetSettings().GetLanguage(); // never use SYSTEM for spelling return eLnge; } // static sal_Int32 ScViewUtil::GetTransliterationType( sal_uInt16 nSlotID ) { sal_Int32 nType = 0; switch ( nSlotID ) { case SID_TRANSLITERATE_SENTENCE_CASE: nType = com::sun::star::i18n::TransliterationModulesExtra::SENTENCE_CASE; break; case SID_TRANSLITERATE_TITLE_CASE: nType = com::sun::star::i18n::TransliterationModulesExtra::TITLE_CASE; break; case SID_TRANSLITERATE_TOGGLE_CASE: nType = com::sun::star::i18n::TransliterationModulesExtra::TOGGLE_CASE; break; case SID_TRANSLITERATE_UPPER: nType = com::sun::star::i18n::TransliterationModules_LOWERCASE_UPPERCASE; break; case SID_TRANSLITERATE_LOWER: nType = com::sun::star::i18n::TransliterationModules_UPPERCASE_LOWERCASE; break; case SID_TRANSLITERATE_HALFWIDTH: nType = com::sun::star::i18n::TransliterationModules_FULLWIDTH_HALFWIDTH; break; case SID_TRANSLITERATE_FULLWIDTH: nType = com::sun::star::i18n::TransliterationModules_HALFWIDTH_FULLWIDTH; break; case SID_TRANSLITERATE_HIRAGANA: nType = com::sun::star::i18n::TransliterationModules_KATAKANA_HIRAGANA; break; case SID_TRANSLITERATE_KATAGANA: nType = com::sun::star::i18n::TransliterationModules_HIRAGANA_KATAKANA; break; } return nType; } // static sal_Bool ScViewUtil::IsActionShown( const ScChangeAction& rAction, const ScChangeViewSettings& rSettings, ScDocument& rDocument ) { // abgelehnte werden durch eine invertierende akzeptierte Action dargestellt, // die Reihenfolge von ShowRejected/ShowAccepted ist deswegen wichtig if ( !rSettings.IsShowRejected() && rAction.IsRejecting() ) return sal_False; if ( !rSettings.IsShowAccepted() && rAction.IsAccepted() && !rAction.IsRejecting() ) return sal_False; if ( rSettings.HasAuthor() ) { if ( rSettings.IsEveryoneButMe() ) { // GetUser() am ChangeTrack ist der aktuelle Benutzer ScChangeTrack* pTrack = rDocument.GetChangeTrack(); if ( !pTrack || rAction.GetUser() == pTrack->GetUser() ) return sal_False; } else if ( rAction.GetUser() != rSettings.GetTheAuthorToShow() ) return sal_False; } if ( rSettings.HasComment() ) { String aComStr=rAction.GetComment(); aComStr.AppendAscii(RTL_CONSTASCII_STRINGPARAM( " (" )); rAction.GetDescription( aComStr, &rDocument ); aComStr+=')'; if(!rSettings.IsValidComment(&aComStr)) return sal_False; } if ( rSettings.HasRange() ) if ( !rSettings.GetTheRangeList().Intersects( rAction.GetBigRange().MakeRange() ) ) return sal_False; if ( rSettings.HasDate() && rSettings.GetTheDateMode() != SCDM_NO_DATEMODE ) { DateTime aDateTime = rAction.GetDateTime(); const DateTime& rFirst = rSettings.GetTheFirstDateTime(); const DateTime& rLast = rSettings.GetTheLastDateTime(); switch ( rSettings.GetTheDateMode() ) { // korrespondiert mit ScHighlightChgDlg::OKBtnHdl case SCDM_DATE_BEFORE: if ( aDateTime > rFirst ) return sal_False; break; case SCDM_DATE_SINCE: if ( aDateTime < rFirst ) return sal_False; break; case SCDM_DATE_EQUAL: case SCDM_DATE_BETWEEN: if ( aDateTime < rFirst || aDateTime > rLast ) return sal_False; break; case SCDM_DATE_NOTEQUAL: if ( aDateTime >= rFirst && aDateTime <= rLast ) return sal_False; break; case SCDM_DATE_SAVE: { ScChangeTrack* pTrack = rDocument.GetChangeTrack(); if ( !pTrack || pTrack->GetLastSavedActionNumber() >= rAction.GetActionNumber() ) return sal_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 sal_False; } } return sal_True; } // static void ScViewUtil::UnmarkFiltered( ScMarkData& rMark, ScDocument* pDoc ) { rMark.MarkToMulti(); ScRange aMultiArea; rMark.GetMultiMarkArea( aMultiArea ); SCCOL nStartCol = aMultiArea.aStart.Col(); SCROW nStartRow = aMultiArea.aStart.Row(); SCCOL nEndCol = aMultiArea.aEnd.Col(); SCROW nEndRow = aMultiArea.aEnd.Row(); bool bChanged = false; SCTAB nTabCount = pDoc->GetTableCount(); for (SCTAB nTab=0; nTabRowFiltered(nRow, nTab, NULL, &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(); } // static bool ScViewUtil::FitToUnfilteredRows( ScRange & rRange, ScDocument * pDoc, size_t nRows ) { SCTAB nTab = rRange.aStart.Tab(); bool bOneTabOnly = (nTab == rRange.aEnd.Tab()); // Always fit the range on its first sheet. DBG_ASSERT( bOneTabOnly, "ScViewUtil::ExtendToUnfilteredRows: works only on one sheet"); SCROW nStartRow = rRange.aStart.Row(); SCROW nLastRow = pDoc->LastNonFilteredRow(nStartRow, MAXROW, nTab); if (ValidRow(nLastRow)) rRange.aEnd.SetRow(nLastRow); SCROW nCount = pDoc->CountNonFilteredRows(nStartRow, MAXROW, nTab); return static_cast(nCount) == nRows && bOneTabOnly; } // static bool ScViewUtil::HasFiltered( const ScRange& rRange, ScDocument* pDoc ) { SCROW nStartRow = rRange.aStart.Row(); SCROW nEndRow = rRange.aEnd.Row(); for (SCTAB nTab=rRange.aStart.Tab(); nTab<=rRange.aEnd.Tab(); nTab++) { if (pDoc->HasFilteredRows(nStartRow, nEndRow, nTab)) return true; } return false; } // static void ScViewUtil::HideDisabledSlot( SfxItemSet& rSet, SfxBindings& rBindings, sal_uInt16 nSlotId ) { SvtCJKOptions aCJKOptions; SvtCTLOptions aCTLOptions; bool bEnabled = true; switch( nSlotId ) { case SID_CHINESE_CONVERSION: case SID_HANGUL_HANJA_CONVERSION: bEnabled = aCJKOptions.IsAnyEnabled(); break; case SID_TRANSLITERATE_HALFWIDTH: case SID_TRANSLITERATE_FULLWIDTH: case SID_TRANSLITERATE_HIRAGANA: case SID_TRANSLITERATE_KATAGANA: bEnabled = aCJKOptions.IsChangeCaseMapEnabled(); break; case SID_INSERT_RLM: case SID_INSERT_LRM: case SID_INSERT_ZWNBSP: case SID_INSERT_ZWSP: bEnabled = aCTLOptions.IsCTLFontEnabled(); break; default: DBG_ERRORFILE( "ScViewUtil::HideDisabledSlot - unknown slot ID" ); return; } rBindings.SetVisibleState( nSlotId, bEnabled ); if( !bEnabled ) rSet.DisableItem( nSlotId ); } //================================================================== sal_Bool ScViewUtil::ExecuteCharMap( const SvxFontItem& rOldFont, SfxViewFrame& rFrame, SvxFontItem& rNewFont, String& rString ) { sal_Bool bRet = sal_False; SvxAbstractDialogFactory* pFact = SvxAbstractDialogFactory::Create(); if(pFact) { SfxAllItemSet aSet( rFrame.GetObjectShell()->GetPool() ); aSet.Put( SfxBoolItem( FN_PARAM_1, sal_False ) ); aSet.Put( SvxFontItem( rOldFont.GetFamily(), rOldFont.GetFamilyName(), rOldFont.GetStyleName(), rOldFont.GetPitch(), rOldFont.GetCharSet(), aSet.GetPool()->GetWhich( SID_ATTR_CHAR_FONT ) ) ); SfxAbstractDialog* pDlg = pFact->CreateSfxDialog( &rFrame.GetWindow(), aSet, rFrame.GetFrame().GetFrameInterface(), RID_SVXDLG_CHARMAP ); if ( pDlg->Execute() == RET_OK ) { SFX_ITEMSET_ARG( pDlg->GetOutputItemSet(), pItem, SfxStringItem, SID_CHARMAP, sal_False ); SFX_ITEMSET_ARG( pDlg->GetOutputItemSet(), pFontItem, SvxFontItem, SID_ATTR_CHAR_FONT, sal_False ); if ( pItem ) rString = pItem->GetValue(); if ( pFontItem ) rNewFont = SvxFontItem( pFontItem->GetFamily(), pFontItem->GetFamilyName(), pFontItem->GetStyleName(), pFontItem->GetPitch(), pFontItem->GetCharSet(), rNewFont.Which() ); bRet = sal_True; } delete pDlg; } return bRet; } bool ScViewUtil::IsFullScreen( SfxViewShell& rViewShell ) { SfxBindings& rBindings = rViewShell.GetViewFrame()->GetBindings(); SfxPoolItem* pItem = 0; bool bIsFullScreen = false; if (rBindings.QueryState( SID_WIN_FULLSCREEN, pItem ) >= SFX_ITEM_DEFAULT) bIsFullScreen = static_cast< SfxBoolItem* >( pItem )->GetValue(); return bIsFullScreen; } void ScViewUtil::SetFullScreen( SfxViewShell& rViewShell, bool bSet ) { if( IsFullScreen( rViewShell ) != bSet ) { SfxBoolItem aItem( SID_WIN_FULLSCREEN, bSet ); rViewShell.GetDispatcher()->Execute( SID_WIN_FULLSCREEN, SFX_CALLMODE_RECORD, &aItem, 0L ); } } //------------------------------------------------------------------ ScUpdateRect::ScUpdateRect( SCCOL nX1, SCROW nY1, SCCOL nX2, SCROW nY2 ) { 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; } sal_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 sal_False; } rX1 = Min(nNewStartX,nOldStartX); rY1 = Min(nNewStartY,nOldStartY); rX2 = Max(nNewEndX,nOldEndX); rY2 = Max(nNewEndY,nOldEndY); if ( nNewStartX == nOldStartX && nNewEndX == nOldEndX ) { if ( nNewStartY == nOldStartY ) { rY1 = Min( nNewEndY, nOldEndY ); rY2 = Max( nNewEndY, nOldEndY ); } else if ( nNewEndY == nOldEndY ) { rY1 = Min( nNewStartY, nOldStartY ); rY2 = Max( nNewStartY, nOldStartY ); } } else if ( nNewStartY == nOldStartY && nNewEndY == nOldEndY ) { if ( nNewStartX == nOldStartX ) { rX1 = Min( nNewEndX, nOldEndX ); rX2 = Max( nNewEndX, nOldEndX ); } else if ( nNewEndX == nOldEndX ) { rX1 = Min( nNewStartX, nOldStartX ); rX2 = Max( nNewStartX, nOldStartX ); } } return sal_True; } #ifdef OLD_SELECTION_PAINT sal_Bool ScUpdateRect::GetXorDiff( SCCOL& rX1, SCROW& rY1, SCCOL& rX2, SCROW& rY2, sal_Bool& rCont ) { rCont = sal_False; if (nNewStartX == nOldStartX && nNewEndX == nOldEndX && nNewStartY == nOldStartY && nNewEndY == nOldEndY) { rX1 = nNewStartX; rY1 = nNewStartY; rX2 = nNewStartX; rY2 = nNewStartY; return sal_False; } rX1 = Min(nNewStartX,nOldStartX); rY1 = Min(nNewStartY,nOldStartY); rX2 = Max(nNewEndX,nOldEndX); rY2 = Max(nNewEndY,nOldEndY); if (nNewStartX == nOldStartX && nNewEndX == nOldEndX) // nur vertikal { if (nNewStartY == nOldStartY) { rY1 = Min( nNewEndY, nOldEndY ) + 1; rY2 = Max( nNewEndY, nOldEndY ); } else if (nNewEndY == nOldEndY) { rY1 = Min( nNewStartY, nOldStartY ); rY2 = Max( nNewStartY, nOldStartY ) - 1; } else { rY1 = Min( nNewStartY, nOldStartY ); rY2 = Max( nNewStartY, nOldStartY ) - 1; rCont = sal_True; nContY1 = Min( nNewEndY, nOldEndY ) + 1; nContY2 = Max( nNewEndY, nOldEndY ); nContX1 = rX1; nContX2 = rX2; } } else if (nNewStartY == nOldStartY && nNewEndY == nOldEndY) // nur horizontal { if (nNewStartX == nOldStartX) { rX1 = Min( nNewEndX, nOldEndX ) + 1; rX2 = Max( nNewEndX, nOldEndX ); } else if (nNewEndX == nOldEndX) { rX1 = Min( nNewStartX, nOldStartX ); rX2 = Max( nNewStartX, nOldStartX ) - 1; } else { rX1 = Min( nNewStartX, nOldStartX ); rX2 = Max( nNewStartX, nOldStartX ) - 1; rCont = sal_True; nContX1 = Min( nNewEndX, nOldEndX ) + 1; nContX2 = Max( nNewEndX, nOldEndX ); nContY1 = rY1; nContY2 = rY2; } } else if (nNewEndX == nOldEndX && nNewEndY == nOldEndY) // links oben { if ((nNewStartX