/* -*- 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 "uiitems.hxx" #include "rangenam.hxx" #include "dbdata.hxx" #include "reffact.hxx" #include "viewdata.hxx" #include "document.hxx" #include "docsh.hxx" #include "scresid.hxx" #include "foptmgr.hxx" #include "globstr.hrc" #include "filter.hrc" #include "filtdlg.hxx" #include // DEFINE -------------------------------------------------------------------- #define ERRORBOX(rid) ScopedVclPtrInstance(this, ScGlobal::GetRscString(rid))->Execute() // class ScSpecialFilterDialog ScSpecialFilterDlg::ScSpecialFilterDlg( SfxBindings* pB, SfxChildWindow* pCW, vcl::Window* pParent, const SfxItemSet& rArgSet ) : ScAnyRefDlg ( pB, pCW, pParent, "AdvancedFilterDialog", "modules/scalc/ui/advancedfilterdialog.ui" ), aStrUndefined ( SC_RESSTR(SCSTR_UNDEFINED) ), pOptionsMgr ( nullptr ), nWhichQuery ( rArgSet.GetPool()->GetWhich( SID_QUERY ) ), theQueryData ( static_cast( rArgSet.Get( nWhichQuery )).GetQueryData() ), pOutItem ( nullptr ), pViewData ( nullptr ), pDoc ( nullptr ), pRefInputEdit ( nullptr ), bRefInputMode ( false ), pIdle ( nullptr ) { get(pLbFilterArea,"lbfilterarea"); get(pEdFilterArea,"edfilterarea"); pEdFilterArea->SetReferences(this, get("filterframe")->get_label_widget()); get(pRbFilterArea,"rbfilterarea"); pRbFilterArea->SetReferences(this, pEdFilterArea); get(pBtnCase,"case"); get(pBtnRegExp,"regexp"); get(pBtnHeader,"header"); get(pBtnUnique,"unique"); get(pBtnCopyResult,"copyresult"); get(pLbCopyArea,"lbcopyarea"); get(pEdCopyArea,"edcopyarea"); pEdCopyArea->SetReferences(this, pBtnCopyResult); get(pRbCopyArea,"rbcopyarea"); pRbCopyArea->SetReferences(this, pEdCopyArea); get(pBtnDestPers,"destpers"); get(pFtDbAreaLabel,"dbarealabel"); get(pFtDbArea,"dbarea"); get(pBtnOk,"ok"); get(pBtnCancel,"cancel"); get(pExpander,"more"); Init( rArgSet ); pEdFilterArea->GrabFocus(); // Hack: RefInput-Kontrolle pIdle = new Idle; // FIXME: this is an abomination pIdle->SetPriority( SchedulerPriority::LOWEST ); pIdle->SetIdleHdl( LINK( this, ScSpecialFilterDlg, TimeOutHdl ) ); pIdle->Start(); } ScSpecialFilterDlg::~ScSpecialFilterDlg() { disposeOnce(); } void ScSpecialFilterDlg::dispose() { const sal_Int32 nEntries = pLbFilterArea->GetEntryCount(); for ( sal_Int32 i=1; i(pLbFilterArea->GetEntryData( i )); delete pOptionsMgr; delete pOutItem; // Hack: RefInput-Kontrolle pIdle->Stop(); delete pIdle; pLbFilterArea.clear(); pEdFilterArea.clear(); pRbFilterArea.clear(); pExpander.clear(); pBtnCase.clear(); pBtnRegExp.clear(); pBtnHeader.clear(); pBtnUnique.clear(); pBtnCopyResult.clear(); pLbCopyArea.clear(); pEdCopyArea.clear(); pRbCopyArea.clear(); pBtnDestPers.clear(); pFtDbAreaLabel.clear(); pFtDbArea.clear(); pBtnOk.clear(); pBtnCancel.clear(); pRefInputEdit.clear(); ScAnyRefDlg::dispose(); } void ScSpecialFilterDlg::Init( const SfxItemSet& rArgSet ) { const ScQueryItem& rQueryItem = static_cast( rArgSet.Get( nWhichQuery )); pBtnOk->SetClickHdl ( LINK( this, ScSpecialFilterDlg, EndDlgHdl ) ); pBtnCancel->SetClickHdl ( LINK( this, ScSpecialFilterDlg, EndDlgHdl ) ); pLbFilterArea->SetSelectHdl ( LINK( this, ScSpecialFilterDlg, FilterAreaSelHdl ) ); pEdFilterArea->SetModifyHdl ( LINK( this, ScSpecialFilterDlg, FilterAreaModHdl ) ); pViewData = rQueryItem.GetViewData(); pDoc = pViewData ? pViewData->GetDocument() : nullptr; pEdFilterArea->SetText( EMPTY_OUSTRING ); // may be overwritten below if ( pViewData && pDoc ) { if(pDoc->GetChangeTrack()!=nullptr) pBtnCopyResult->Disable(); ScRangeName* pRangeNames = pDoc->GetRangeName(); pLbFilterArea->Clear(); pLbFilterArea->InsertEntry( aStrUndefined, 0 ); if (!pRangeNames->empty()) { ScRangeName::const_iterator itr = pRangeNames->begin(), itrEnd = pRangeNames->end(); for (; itr != itrEnd; ++itr) { if (!itr->second->HasType(ScRangeData::Type::Criteria)) continue; const sal_Int32 nInsert = pLbFilterArea->InsertEntry(itr->second->GetName()); OUString aSymbol; itr->second->GetSymbol(aSymbol); pLbFilterArea->SetEntryData(nInsert, new OUString(aSymbol)); } } // is there a stored source range? ScRange aAdvSource; if (rQueryItem.GetAdvancedQuerySource(aAdvSource)) { OUString aRefStr(aAdvSource.Format(ScRefFlags::RANGE_ABS_3D, pDoc, pDoc->GetAddressConvention())); pEdFilterArea->SetRefString( aRefStr ); } } pLbFilterArea->SelectEntryPos( 0 ); // Optionen initialisieren lassen: pOptionsMgr = new ScFilterOptionsMgr( pViewData, theQueryData, pBtnCase, pBtnRegExp, pBtnHeader, pBtnUnique, pBtnCopyResult, pBtnDestPers, pLbCopyArea, pEdCopyArea, pRbCopyArea, pFtDbAreaLabel, pFtDbArea, aStrUndefined ); // Spezialfilter braucht immer Spaltenkoepfe pBtnHeader->Check(); pBtnHeader->Disable(); // Modal-Modus einschalten // SetDispatcherLock( true ); //@BugID 54702 Enablen/Disablen nur noch in Basisklasse //SFX_APPWINDOW->Disable(false); //! allgemeine Methode im ScAnyRefDlg } bool ScSpecialFilterDlg::Close() { if (pViewData) pViewData->GetDocShell()->CancelAutoDBRange(); return DoClose( ScSpecialFilterDlgWrapper::GetChildWindowId() ); } // Uebergabe eines mit der Maus selektierten Tabellenbereiches, der dann als // neue Selektion im Referenz-Edit angezeigt wird. void ScSpecialFilterDlg::SetReference( const ScRange& rRef, ScDocument* pDocP ) { if ( bRefInputMode && pRefInputEdit ) // Nur moeglich, wenn im Referenz-Editmodus { if ( rRef.aStart != rRef.aEnd ) RefInputStart( pRefInputEdit ); OUString aRefStr; const formula::FormulaGrammar::AddressConvention eConv = pDocP->GetAddressConvention(); if ( pRefInputEdit == pEdCopyArea) aRefStr = rRef.aStart.Format(ScRefFlags::ADDR_ABS_3D, pDocP, eConv); else if ( pRefInputEdit == pEdFilterArea) aRefStr = rRef.Format(ScRefFlags::RANGE_ABS_3D, pDocP, eConv); pRefInputEdit->SetRefString( aRefStr ); } } void ScSpecialFilterDlg::SetActive() { if ( bRefInputMode ) { if ( pRefInputEdit == pEdCopyArea ) { pEdCopyArea->GrabFocus(); pEdCopyArea->GetModifyHdl().Call( *pEdCopyArea ); } else if ( pRefInputEdit == pEdFilterArea ) { pEdFilterArea->GrabFocus(); FilterAreaModHdl( *pEdFilterArea ); } } else GrabFocus(); RefInputDone(); } ScQueryItem* ScSpecialFilterDlg::GetOutputItem( const ScQueryParam& rParam, const ScRange& rSource ) { if ( pOutItem ) DELETEZ( pOutItem ); pOutItem = new ScQueryItem( nWhichQuery, &rParam ); pOutItem->SetAdvancedQuerySource( &rSource ); return pOutItem; } bool ScSpecialFilterDlg::IsRefInputMode() const { return bRefInputMode; } // Handler: IMPL_LINK_TYPED( ScSpecialFilterDlg, EndDlgHdl, Button*, pBtn, void ) { OSL_ENSURE( pDoc && pViewData, "Document or ViewData not found. :-/" ); if ( (pBtn == pBtnOk) && pDoc && pViewData ) { OUString theCopyStr( pEdCopyArea->GetText() ); OUString theAreaStr( pEdFilterArea->GetText() ); ScQueryParam theOutParam( theQueryData ); ScAddress theAdrCopy; bool bEditInputOk = true; bool bQueryOk = false; ScRange theFilterArea; const formula::FormulaGrammar::AddressConvention eConv = pDoc->GetAddressConvention(); if ( pBtnCopyResult->IsChecked() ) { sal_Int32 nColonPos = theCopyStr.indexOf( ':' ); if ( -1 != nColonPos ) theCopyStr = theCopyStr.copy( 0, nColonPos ); ScRefFlags nResult = theAdrCopy.Parse( theCopyStr, pDoc, eConv ); if ( (nResult & ScRefFlags::VALID) == ScRefFlags::ZERO ) { if (!pExpander->get_expanded()) pExpander->set_expanded(true); ERRORBOX( STR_INVALID_TABREF ); pEdCopyArea->GrabFocus(); bEditInputOk = false; } } if ( bEditInputOk ) { ScRefFlags nResult = ScRange().Parse( theAreaStr, pDoc, eConv ); if ( (nResult & ScRefFlags::VALID) == ScRefFlags::ZERO ) { ERRORBOX( STR_INVALID_TABREF ); pEdFilterArea->GrabFocus(); bEditInputOk = false; } } if ( bEditInputOk ) { /* * Alle Edit-Felder enthalten gueltige Bereiche. * Nun wird versucht aus dem Filterbereich * ein ScQueryParam zu erzeugen: */ ScRefFlags nResult = theFilterArea.Parse( theAreaStr, pDoc, eConv ); if ( (nResult & ScRefFlags::VALID) == ScRefFlags::VALID ) { ScAddress& rStart = theFilterArea.aStart; ScAddress& rEnd = theFilterArea.aEnd; if ( pBtnCopyResult->IsChecked() ) { theOutParam.bInplace = false; theOutParam.nDestTab = theAdrCopy.Tab(); theOutParam.nDestCol = theAdrCopy.Col(); theOutParam.nDestRow = theAdrCopy.Row(); } else { theOutParam.bInplace = true; theOutParam.nDestTab = 0; theOutParam.nDestCol = 0; theOutParam.nDestRow = 0; } theOutParam.bHasHeader = pBtnHeader->IsChecked(); theOutParam.bByRow = true; theOutParam.bCaseSens = pBtnCase->IsChecked(); theOutParam.eSearchType = pBtnRegExp->IsChecked() ? utl::SearchParam::SRCH_REGEXP : utl::SearchParam::SRCH_NORMAL; theOutParam.bDuplicate = !pBtnUnique->IsChecked(); theOutParam.bDestPers = pBtnDestPers->IsChecked(); bQueryOk = pDoc->CreateQueryParam( rStart.Col(), rStart.Row(), rEnd.Col(), rEnd.Row(), rStart.Tab(), theOutParam ); } } if ( bQueryOk ) { SetDispatcherLock( false ); SwitchToDocument(); GetBindings().GetDispatcher()->ExecuteList(FID_FILTER_OK, SfxCallMode::SLOT | SfxCallMode::RECORD, { GetOutputItem(theOutParam, theFilterArea) }); Close(); } else { ERRORBOX( STR_INVALID_QUERYAREA ); pEdFilterArea->GrabFocus(); } } else if ( pBtn == pBtnCancel ) { Close(); } } IMPL_LINK_TYPED( ScSpecialFilterDlg, TimeOutHdl, Idle*, _pIdle, void ) { // every 50ms check whether RefInputMode is still true if( (_pIdle == pIdle) && IsActive() ) { if( pEdCopyArea->HasFocus() || pRbCopyArea->HasFocus() ) { pRefInputEdit = pEdCopyArea; bRefInputMode = true; } else if( pEdFilterArea->HasFocus() || pRbFilterArea->HasFocus() ) { pRefInputEdit = pEdFilterArea; bRefInputMode = true; } else if( bRefInputMode ) { pRefInputEdit = nullptr; bRefInputMode = false; } } pIdle->Start(); } IMPL_LINK_TYPED( ScSpecialFilterDlg, FilterAreaSelHdl, ListBox&, rLb, void ) { if ( &rLb == pLbFilterArea ) { OUString aString; const sal_Int32 nSelPos = pLbFilterArea->GetSelectEntryPos(); if ( nSelPos > 0 ) aString = *static_cast(pLbFilterArea->GetEntryData( nSelPos )); pEdFilterArea->SetText( aString ); } } IMPL_LINK_TYPED( ScSpecialFilterDlg, FilterAreaModHdl, Edit&, rEd, void ) { if ( &rEd == pEdFilterArea ) { if ( pDoc && pViewData ) { OUString theCurAreaStr = rEd.GetText(); ScRefFlags nResult = ScRange().Parse( theCurAreaStr, pDoc ); if ( (nResult & ScRefFlags::VALID) == ScRefFlags::VALID ) { const sal_Int32 nCount = pLbFilterArea->GetEntryCount(); for ( sal_Int32 i=1; i(pLbFilterArea->GetEntryData( i )); if (theCurAreaStr == *pStr) { pLbFilterArea->SelectEntryPos( i ); return; } } pLbFilterArea->SelectEntryPos( 0 ); } } else pLbFilterArea->SelectEntryPos( 0 ); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */