/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include void ScTabViewShell::SetCurRefDlgId( sal_uInt16 nNew ) { // CurRefDlgId is stored in ScModule to find if a ref dialog is open, // and in the view to identify the view that has opened the dialog nCurRefDlgId = nNew; } //ugly hack to call Define Name from Manage Names void ScTabViewShell::SwitchBetweenRefDialogs(SfxModelessDialogController* pDialog) { sal_uInt16 nSlotId = SC_MOD()->GetCurRefDlgId(); if( nSlotId == FID_ADD_NAME ) { static_cast(pDialog)->GetNewData(maName, maScope); static_cast(pDialog)->Close(); sal_uInt16 nId = ScNameDlgWrapper::GetChildWindowId(); SfxViewFrame* pViewFrm = GetViewFrame(); SfxChildWindow* pWnd = pViewFrm->GetChildWindow( nId ); SC_MOD()->SetRefDialog( nId, pWnd == nullptr ); } else if (nSlotId == FID_DEFINE_NAME) { mbInSwitch = true; static_cast(pDialog)->GetRangeNames(m_RangeMap); static_cast(pDialog)->Close(); sal_uInt16 nId = ScNameDefDlgWrapper::GetChildWindowId(); SfxViewFrame* pViewFrm = GetViewFrame(); SfxChildWindow* pWnd = pViewFrm->GetChildWindow( nId ); SC_MOD()->SetRefDialog( nId, pWnd == nullptr ); } } VclPtr ScTabViewShell::CreateRefDialog( SfxBindings* pB, SfxChildWindow* pCW, const SfxChildWinInfo* pInfo, vcl::Window* pParent, sal_uInt16 nSlotId ) { // only open dialog when called through ScModule::SetRefDialog, // so that it does not re appear for instance after a crash (#42341#). if ( SC_MOD()->GetCurRefDlgId() != nSlotId ) return nullptr; if ( nCurRefDlgId != nSlotId ) { // the dialog has been opened in a different view // -> lock the dispatcher for this view (modal mode) GetViewData().GetDispatcher().Lock( true ); // lock is reset when closing dialog return nullptr; } VclPtr pResult; if(pCW) pCW->SetHideNotDelete(true); switch( nSlotId ) { case WID_CONDFRMT_REF: { bool bFound = false; const ScCondFormatDlgItem* pDlgItem = nullptr; // Get the pool item stored by Conditional Format Manager Dialog. const SfxPoolItem* pItem = nullptr; auto itemsRange = GetPool().GetItemSurrogates(SCITEM_CONDFORMATDLGDATA); if (itemsRange.begin() != itemsRange.end()) { pItem = *itemsRange.begin(); pDlgItem = static_cast(pItem); bFound = true; } ScViewData& rViewData = GetViewData(); rViewData.SetRefTabNo( rViewData.GetTabNo() ); pResult = VclPtr::Create( pB, pCW, pParent, &rViewData, pDlgItem ); // Remove the pool item stored by Conditional Format Manager Dialog. if ( bFound && pItem ) GetPool().Remove( *pItem ); } break; } if (pResult) { // the dialogs are always displayed with the option button collapsed, // the size has to be carried over initialize // (or store the option status !!!) Size aSize = pResult->GetSizePixel(); pResult->Initialize( pInfo ); pResult->SetSizePixel(aSize); } return pResult; } std::unique_ptr ScTabViewShell::CreateRefDialogController( SfxBindings* pB, SfxChildWindow* pCW, const SfxChildWinInfo* pInfo, weld::Window* pParent, sal_uInt16 nSlotId) { // only open dialog when called through ScModule::SetRefDialog, // so that it does not re appear for instance after a crash (#42341#). if ( SC_MOD()->GetCurRefDlgId() != nSlotId ) return nullptr; if ( nCurRefDlgId != nSlotId ) { // the dialog has been opened in a different view // -> lock the dispatcher for this view (modal mode) GetViewData().GetDispatcher().Lock( true ); // lock is reset when closing dialog return nullptr; } std::unique_ptr xResult; if(pCW) pCW->SetHideNotDelete(true); ScDocument* pDoc = GetViewData().GetDocument(); switch( nSlotId ) { case SID_CORRELATION_DIALOG: xResult.reset(new ScCorrelationDialog(pB, pCW, pParent, &GetViewData())); break; case SID_SAMPLING_DIALOG: xResult.reset(new ScSamplingDialog(pB, pCW, pParent, &GetViewData())); break; case SID_DESCRIPTIVE_STATISTICS_DIALOG: xResult.reset(new ScDescriptiveStatisticsDialog(pB, pCW, pParent, &GetViewData())); break; case SID_ANALYSIS_OF_VARIANCE_DIALOG: xResult.reset(new ScAnalysisOfVarianceDialog(pB, pCW, pParent, &GetViewData())); break; case SID_COVARIANCE_DIALOG: xResult.reset(new ScCovarianceDialog(pB, pCW, pParent, &GetViewData())); break; case SID_EXPONENTIAL_SMOOTHING_DIALOG: xResult.reset(new ScExponentialSmoothingDialog(pB, pCW, pParent, &GetViewData())); break; case SID_MOVING_AVERAGE_DIALOG: xResult.reset(new ScMovingAverageDialog(pB, pCW, pParent, &GetViewData())); break; case SID_REGRESSION_DIALOG: xResult.reset(new ScRegressionDialog(pB, pCW, pParent, &GetViewData())); break; case SID_FTEST_DIALOG: xResult.reset(new ScFTestDialog(pB, pCW, pParent, &GetViewData())); break; case SID_TTEST_DIALOG: xResult.reset(new ScTTestDialog(pB, pCW, pParent, &GetViewData())); break; case SID_ZTEST_DIALOG: xResult.reset(new ScZTestDialog(pB, pCW, pParent, &GetViewData())); break; case SID_CHI_SQUARE_TEST_DIALOG: xResult.reset(new ScChiSquareTestDialog(pB, pCW, pParent, &GetViewData())); break; case SID_FOURIER_ANALYSIS_DIALOG: xResult.reset(new ScFourierAnalysisDialog(pB, pCW, pParent, &GetViewData())); break; case WID_SIMPLE_REF: { // dialog checks, what is in the cell ScViewData& rViewData = GetViewData(); rViewData.SetRefTabNo( rViewData.GetTabNo() ); xResult.reset(new ScSimpleRefDlg(pB, pCW, pParent)); break; } case FID_DEFINE_NAME: { if (!mbInSwitch) { xResult.reset(new ScNameDlg(pB, pCW, pParent, &GetViewData(), ScAddress( GetViewData().GetCurX(), GetViewData().GetCurY(), GetViewData().GetTabNo() ) )); } else { xResult.reset(new ScNameDlg( pB, pCW, pParent, &GetViewData(), ScAddress( GetViewData().GetCurX(), GetViewData().GetCurY(), GetViewData().GetTabNo() ), &m_RangeMap)); static_cast(xResult.get())->SetEntry(maName, maScope); mbInSwitch = false; } break; } case FID_ADD_NAME: { if (!mbInSwitch) { std::map aRangeMap; pDoc->GetRangeNameMap(aRangeMap); xResult.reset(new ScNameDefDlg(pB, pCW, pParent, &GetViewData(), aRangeMap, ScAddress(GetViewData().GetCurX(), GetViewData().GetCurY(), GetViewData().GetTabNo()), true)); } else { std::map aRangeMap; for (auto const& itr : m_RangeMap) { aRangeMap.insert(std::pair(itr.first, itr.second.get())); } xResult.reset(new ScNameDefDlg(pB, pCW, pParent, &GetViewData(), aRangeMap, ScAddress(GetViewData().GetCurX(), GetViewData().GetCurY(), GetViewData().GetTabNo()), false)); } break; } case SID_RANDOM_NUMBER_GENERATOR_DIALOG: xResult.reset(new ScRandomNumberGeneratorDialog(pB, pCW, pParent, &GetViewData())); break; case SID_DEFINE_DBNAME: { // when called for an existing range, then mark GetDBData( true, SC_DB_OLD ); const ScMarkData& rMark = GetViewData().GetMarkData(); if ( !rMark.IsMarked() && !rMark.IsMultiMarked() ) MarkDataArea( false ); xResult.reset(new ScDbNameDlg(pB, pCW, pParent, &GetViewData())); break; } case SID_OPENDLG_EDIT_PRINTAREA: xResult.reset(new ScPrintAreasDlg(pB, pCW, pParent)); break; case SID_DEFINE_COLROWNAMERANGES: xResult.reset(new ScColRowNameRangesDlg(pB, pCW, pParent, &GetViewData())); break; case SID_OPENDLG_SOLVE: { ScViewData& rViewData = GetViewData(); ScAddress aCurPos( rViewData.GetCurX(), rViewData.GetCurY(), rViewData.GetTabNo()); xResult.reset(new ScSolverDlg(pB, pCW, pParent, rViewData.GetDocument(), aCurPos)); break; } case SID_OPENDLG_TABOP: { ScViewData& rViewData = GetViewData(); ScRefAddress aCurPos ( rViewData.GetCurX(), rViewData.GetCurY(), rViewData.GetTabNo()); xResult.reset(new ScTabOpDlg(pB, pCW, pParent, rViewData.GetDocument(), aCurPos)); break; } case SID_OPENDLG_CONSOLIDATE: { SfxItemSet aArgSet( GetPool(), svl::Items{} ); const ScConsolidateParam* pDlgData = pDoc->GetConsolidateDlgData(); if ( !pDlgData ) { ScConsolidateParam aConsParam; SCCOL nStartCol, nEndCol; SCROW nStartRow, nEndRow; SCTAB nStartTab, nEndTab; GetViewData().GetSimpleArea( nStartCol, nStartRow, nStartTab, nEndCol, nEndRow, nEndTab ); PutInOrder( nStartCol, nEndCol ); PutInOrder( nStartRow, nEndRow ); PutInOrder( nStartTab, nEndTab ); aConsParam.nCol = nStartCol; aConsParam.nRow = nStartRow; aConsParam.nTab = nStartTab; aArgSet.Put( ScConsolidateItem( SCITEM_CONSOLIDATEDATA, &aConsParam ) ); } else { aArgSet.Put( ScConsolidateItem( SCITEM_CONSOLIDATEDATA, pDlgData ) ); } xResult.reset(new ScConsolidateDlg(pB, pCW, pParent, aArgSet)); break; } case SID_FILTER: { ScQueryParam aQueryParam; SfxItemSet aArgSet( GetPool(), svl::Items{} ); ScDBData* pDBData = GetDBData(false, SC_DB_MAKE, ScGetDBSelection::RowDown); pDBData->ExtendDataArea(pDoc); pDBData->GetQueryParam( aQueryParam ); ScRange aArea; pDBData->GetArea(aArea); MarkRange(aArea, false); aArgSet.Put( ScQueryItem( SCITEM_QUERYDATA, &GetViewData(), &aQueryParam ) ); // mark current sheet (due to RefInput in dialog) GetViewData().SetRefTabNo( GetViewData().GetTabNo() ); xResult.reset(new ScFilterDlg(pB, pCW, pParent, aArgSet)); break; } case SID_SPECIAL_FILTER: { ScQueryParam aQueryParam; SfxItemSet aArgSet( GetPool(), svl::Items{} ); ScDBData* pDBData = GetDBData(false, SC_DB_MAKE, ScGetDBSelection::RowDown); pDBData->ExtendDataArea(pDoc); pDBData->GetQueryParam( aQueryParam ); ScRange aArea; pDBData->GetArea(aArea); MarkRange(aArea, false); ScQueryItem aItem( SCITEM_QUERYDATA, &GetViewData(), &aQueryParam ); ScRange aAdvSource; if (pDBData->GetAdvancedQuerySource(aAdvSource)) aItem.SetAdvancedQuerySource( &aAdvSource ); aArgSet.Put( aItem ); // mark current sheet (due to RefInput in dialog) GetViewData().SetRefTabNo( GetViewData().GetTabNo() ); xResult.reset(new ScSpecialFilterDlg(pB, pCW, pParent, aArgSet)); break; } case SID_OPENDLG_OPTSOLVER: { ScViewData& rViewData = GetViewData(); ScAddress aCurPos( rViewData.GetCurX(), rViewData.GetCurY(), rViewData.GetTabNo()); xResult.reset(new ScOptSolverDlg(pB, pCW, pParent, rViewData.GetDocShell(), aCurPos)); break; } case FID_CHG_SHOW: { // dialog checks, what is in the cell xResult.reset(new ScHighlightChgDlg(pB, pCW, pParent, &GetViewData())); break; } case SID_MANAGE_XML_SOURCE: { xResult.reset(new ScXMLSourceDlg(pB, pCW, pParent, pDoc)); break; } case SID_OPENDLG_PIVOTTABLE: { // all settings must be in pDialogDPObject if( pDialogDPObject ) { // Check for an existing datapilot output. ScViewData& rViewData = GetViewData(); rViewData.SetRefTabNo( rViewData.GetTabNo() ); ScDPObject* pObj = pDoc->GetDPAtCursor(rViewData.GetCurX(), rViewData.GetCurY(), rViewData.GetTabNo()); xResult.reset(new ScPivotLayoutDialog(pB, pCW, pParent, &rViewData, pDialogDPObject.get(), pObj == nullptr)); } break; } case SID_OPENDLG_FUNCTION: { // dialog checks, what is in the cell xResult.reset(new ScFormulaDlg(pB, pCW, pParent, &GetViewData(),ScGlobal::GetStarCalcFunctionMgr())); break; } } if (xResult) xResult->Initialize( pInfo ); return xResult; } int ScTabViewShell::getPart() const { return GetViewData().GetTabNo(); } void ScTabViewShell::afterCallbackRegistered() { UpdateInputHandler(true, false); } void ScTabViewShell::NotifyCursor(SfxViewShell* pOtherShell) const { ScDrawView* pDrView = const_cast(this)->GetScDrawView(); if (pDrView) { if (pDrView->GetTextEditObject()) { // Blinking cursor. EditView& rEditView = pDrView->GetTextEditOutlinerView()->GetEditView(); rEditView.RegisterOtherShell(pOtherShell); rEditView.ShowCursor(); rEditView.RegisterOtherShell(nullptr); // Text selection, if any. rEditView.DrawSelectionXOR(pOtherShell); } else { // Graphic selection. pDrView->AdjustMarkHdl(pOtherShell); } } const ScGridWindow* pWin = GetViewData().GetActiveWin(); if (pWin) pWin->updateLibreOfficeKitCellCursor(pOtherShell); } css::uno::Reference ScTabViewShell::GetClipData(vcl::Window* pWin) { SfxViewFrame* pViewFrame = nullptr; css::uno::Reference xTransferable; css::uno::Reference xClipboard; if (pWin) xClipboard = pWin->GetClipboard(); else if ((pViewFrame = SfxViewFrame::GetFirst(nullptr, false))) xClipboard = pViewFrame->GetWindow().GetClipboard(); xTransferable.set(xClipboard.is() ? xClipboard->getContents() : nullptr, css::uno::UNO_QUERY); return xTransferable; } void ScTabViewShell::notifyAllViewsHeaderInvalidation(HeaderType eHeaderType, SCTAB nCurrentTabIndex) { if (comphelper::LibreOfficeKit::isActive()) { OString aPayload; switch (eHeaderType) { case COLUMN_HEADER: aPayload = "column"; break; case ROW_HEADER: aPayload = "row"; break; case BOTH_HEADERS: default: aPayload = "all"; break; } SfxViewShell* pViewShell = SfxViewShell::GetFirst(); while (pViewShell) { ScTabViewShell* pTabViewShell = dynamic_cast(pViewShell); if (pTabViewShell && (nCurrentTabIndex == -1 || pTabViewShell->getPart() == nCurrentTabIndex)) { pViewShell->libreOfficeKitViewCallback(LOK_CALLBACK_INVALIDATE_HEADER, aPayload.getStr()); } pViewShell = SfxViewShell::GetNext(*pViewShell); } } } void ScTabViewShell::notifyAllViewsHeaderInvalidation(bool bColumns, SCTAB nCurrentTabIndex) { HeaderType eHeaderType = bColumns ? COLUMN_HEADER : ROW_HEADER; ScTabViewShell::notifyAllViewsHeaderInvalidation(eHeaderType, nCurrentTabIndex); } bool ScTabViewShell::isAnyEditViewInRange(bool bColumns, SCCOLROW nStart, SCCOLROW nEnd) { if (comphelper::LibreOfficeKit::isActive()) { SfxViewShell* pViewShell = SfxViewShell::GetFirst(); while (pViewShell) { ScTabViewShell* pTabViewShell = dynamic_cast(pViewShell); if (pTabViewShell) { ScInputHandler* pInputHandler = pTabViewShell->GetInputHandler(); if (pInputHandler && pInputHandler->GetActiveView()) { const ScViewData& rViewData = pTabViewShell->GetViewData(); SCCOLROW nPos = bColumns ? rViewData.GetCurX() : rViewData.GetCurY(); if (nStart <= nPos && nPos <= nEnd) return true; } } pViewShell = SfxViewShell::GetNext(*pViewShell); } } return false; } bool ScTabViewShell::UseSubTotal(ScRangeList* pRangeList) { bool bSubTotal = false; ScDocument* pDoc = GetViewData().GetDocument(); size_t nRangeCount (pRangeList->size()); size_t nRangeIndex (0); while (!bSubTotal && nRangeIndex < nRangeCount) { const ScRange& rRange = (*pRangeList)[nRangeIndex]; SCTAB nTabEnd(rRange.aEnd.Tab()); SCTAB nTab(rRange.aStart.Tab()); while (!bSubTotal && nTab <= nTabEnd) { SCROW nRowEnd(rRange.aEnd.Row()); SCROW nRow(rRange.aStart.Row()); while (!bSubTotal && nRow <= nRowEnd) { if (pDoc->RowFiltered(nRow, nTab)) bSubTotal = true; else ++nRow; } ++nTab; } ++nRangeIndex; } if (!bSubTotal) { const ScDBCollection::NamedDBs& rDBs = pDoc->GetDBCollection()->getNamedDBs(); for (const auto& rxDB : rDBs) { const ScDBData& rDB = *rxDB; if (!rDB.HasAutoFilter()) continue; nRangeIndex = 0; while (!bSubTotal && nRangeIndex < nRangeCount) { const ScRange & rRange = (*pRangeList)[nRangeIndex]; ScRange aDBArea; rDB.GetArea(aDBArea); if (aDBArea.Intersects(rRange)) bSubTotal = true; ++nRangeIndex; } if (bSubTotal) break; } } return bSubTotal; } const OUString ScTabViewShell::DoAutoSum(bool& rRangeFinder, bool& rSubTotal, const OpCode eCode) { OUString aFormula; const ScMarkData& rMark = GetViewData().GetMarkData(); if ( rMark.IsMarked() || rMark.IsMultiMarked() ) { ScRangeList aMarkRangeList; rRangeFinder = rSubTotal = false; rMark.FillRangeListWithMarks( &aMarkRangeList, false ); ScDocument* pDoc = GetViewData().GetDocument(); // check if one of the marked ranges is empty bool bEmpty = false; const size_t nCount = aMarkRangeList.size(); for ( size_t i = 0; i < nCount; ++i ) { const ScRange & rRange( aMarkRangeList[i] ); if ( pDoc->IsBlockEmpty( rRange.aStart.Tab(), rRange.aStart.Col(), rRange.aStart.Row(), rRange.aEnd.Col(), rRange.aEnd.Row() ) ) { bEmpty = true; break; } } if ( bEmpty ) { ScRangeList aRangeList; const bool bDataFound = GetAutoSumArea( aRangeList ); if ( bDataFound ) { ScAddress aAddr = aRangeList.back().aEnd; aAddr.IncRow(); const bool bSubTotal( UseSubTotal( &aRangeList ) ); EnterAutoSum( aRangeList, bSubTotal, aAddr, eCode ); } } else { const bool bSubTotal( UseSubTotal( &aMarkRangeList ) ); for ( size_t i = 0; i < nCount; ++i ) { const ScRange & rRange = aMarkRangeList[i]; const bool bSetCursor = ( i == nCount - 1 ); const bool bContinue = ( i != 0 ); if ( !AutoSum( rRange, bSubTotal, bSetCursor, bContinue, eCode ) ) { MarkRange( rRange, false ); SetCursor( rRange.aEnd.Col(), rRange.aEnd.Row() ); const ScRangeList aRangeList; ScAddress aAddr = rRange.aEnd; aAddr.IncRow(); aFormula = GetAutoSumFormula( aRangeList, bSubTotal, aAddr , eCode); break; } } } } else // Only insert into input row { ScRangeList aRangeList; rRangeFinder = GetAutoSumArea( aRangeList ); rSubTotal = UseSubTotal( &aRangeList ); ScAddress aAddr = GetViewData().GetCurPos(); aFormula = GetAutoSumFormula( aRangeList, rSubTotal, aAddr , eCode); } return aFormula; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */