/* -*- 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 . */ #undef SC_DLLIMPLEMENTATION #include "scitems.hxx" #include "uiitems.hxx" #include "global.hxx" #include "userlist.hxx" #include "viewdata.hxx" #include "document.hxx" #include "scresid.hxx" #include "sc.hrc" #include "subtdlg.hxx" #include "tpsubt.hxx" #include // Subtotals group tabpage: ScTpSubTotalGroup::ScTpSubTotalGroup( Window* pParent, const SfxItemSet& rArgSet ) : SfxTabPage ( pParent, "SubTotalGrpPage", "modules/scalc/ui/subtotalgrppage.ui", rArgSet ), aStrNone ( SC_RESSTR( SCSTR_NONE ) ), aStrColumn ( SC_RESSTR( SCSTR_COLUMN ) ), pViewData ( NULL ), pDoc ( NULL ), nWhichSubTotals ( rArgSet.GetPool()->GetWhich( SID_SUBTOTALS ) ), rSubTotalData ( ((const ScSubTotalItem&) rArgSet.Get( nWhichSubTotals )). GetSubTotalData() ), nFieldCount ( 0 ) { get(mpLbGroup, "group_by"); get(mpLbColumns, "columns"); get(mpLbFunctions, "functions"); long nHeight = mpLbColumns->GetTextHeight() * 14; mpLbColumns->set_height_request(nHeight); mpLbFunctions->set_height_request(nHeight); // Font is correctly initialized by SvTreeListBox ctor mpLbColumns->SetSelectionMode( SINGLE_SELECTION ); mpLbColumns->SetDragDropMode( SV_DRAGDROP_NONE ); mpLbColumns->SetSpaceBetweenEntries( 0 ); Init (); } ScTpSubTotalGroup::~ScTpSubTotalGroup() { sal_uLong nCount = mpLbColumns->GetEntryCount(); if ( nCount > 0 ) { sal_uInt16* pData = NULL; for ( sal_uLong i=0; iGetEntryData( i )); OSL_ENSURE( pData, "EntryData not found" ); delete pData; } } } void ScTpSubTotalGroup::Init() { const ScSubTotalItem& rSubTotalItem = (const ScSubTotalItem&) GetItemSet().Get( nWhichSubTotals ); pViewData = rSubTotalItem.GetViewData(); pDoc = ( pViewData ) ? pViewData->GetDocument() : NULL; OSL_ENSURE( pViewData && pDoc, "ViewData or Document not found :-(" ); mpLbGroup->SetSelectHdl ( LINK( this, ScTpSubTotalGroup, SelectHdl ) ); mpLbColumns->SetSelectHdl ( LINK( this, ScTpSubTotalGroup, SelectHdl ) ); mpLbColumns->SetCheckButtonHdl ( LINK( this, ScTpSubTotalGroup, CheckHdl ) ); mpLbFunctions->SetSelectHdl ( LINK( this, ScTpSubTotalGroup, SelectHdl ) ); nFieldArr[0] = 0; FillListBoxes(); } bool ScTpSubTotalGroup::DoReset( sal_uInt16 nGroupNo, const SfxItemSet& rArgSet ) { sal_uInt16 nGroupIdx = 0; OSL_ENSURE( (nGroupNo<=3) && (nGroupNo>0), "Invalid group" ); if ( (nGroupNo > 3) || (nGroupNo == 0) ) return false; else nGroupIdx = nGroupNo-1; // first we have to clear the listboxes... for ( sal_uLong nLbEntry = 0; nLbEntry < mpLbColumns->GetEntryCount(); ++nLbEntry ) { mpLbColumns->CheckEntryPos( nLbEntry, false ); *((sal_uInt16*)mpLbColumns->GetEntryData( nLbEntry )) = 0; } mpLbFunctions->SelectEntryPos( 0 ); ScSubTotalParam theSubTotalData( ((const ScSubTotalItem&) rArgSet.Get( nWhichSubTotals )). GetSubTotalData() ); if ( theSubTotalData.bGroupActive[nGroupIdx] ) { SCCOL nField = theSubTotalData.nField[nGroupIdx]; SCCOL nSubTotals = theSubTotalData.nSubTotals[nGroupIdx]; SCCOL* pSubTotals = theSubTotalData.pSubTotals[nGroupIdx]; ScSubTotalFunc* pFunctions = theSubTotalData.pFunctions[nGroupIdx]; mpLbGroup->SelectEntryPos( GetFieldSelPos( nField )+1 ); sal_uInt16 nFirstChecked = 0; for ( sal_uInt16 i=0; iGetEntryData( nCheckPos ); mpLbColumns->CheckEntryPos( nCheckPos ); *pFunction = FuncToLbPos( pFunctions[i] ); if (i == 0 || (i > 0 && nCheckPos < nFirstChecked)) nFirstChecked = nCheckPos; } // Select the first checked field from the top. mpLbColumns->SelectEntryPos(nFirstChecked); } else { mpLbGroup->SelectEntryPos( (nGroupNo == 1) ? 1 : 0 ); mpLbColumns->SelectEntryPos( 0 ); mpLbFunctions->SelectEntryPos( 0 ); } return true; } bool ScTpSubTotalGroup::DoFillItemSet( sal_uInt16 nGroupNo, SfxItemSet& rArgSet ) { sal_uInt16 nGroupIdx = 0; OSL_ENSURE( (nGroupNo<=3) && (nGroupNo>0), "Invalid group" ); OSL_ENSURE( (mpLbGroup->GetEntryCount() > 0) && (mpLbColumns->GetEntryCount() > 0) && (mpLbFunctions->GetEntryCount() > 0), "Non-initialized Lists" ); if ( (nGroupNo > 3) || (nGroupNo == 0) || (mpLbGroup->GetEntryCount() == 0) || (mpLbColumns->GetEntryCount() == 0) || (mpLbFunctions->GetEntryCount() == 0) ) return false; else nGroupIdx = nGroupNo-1; ScSubTotalParam theSubTotalData; // auslesen, wenn schon teilweise gefuellt SfxTabDialog* pDlg = GetTabDialog(); if ( pDlg ) { const SfxItemSet* pExample = pDlg->GetExampleSet(); const SfxPoolItem* pItem; if ( pExample && pExample->GetItemState( nWhichSubTotals, true, &pItem ) == SFX_ITEM_SET ) theSubTotalData = ((const ScSubTotalItem*)pItem)->GetSubTotalData(); } boost::scoped_array pFunctions; boost::scoped_array pSubTotals; sal_uInt16 nGroup = mpLbGroup->GetSelectEntryPos(); sal_uInt16 nEntryCount = (sal_uInt16)mpLbColumns->GetEntryCount(); sal_uInt16 nCheckCount = mpLbColumns->GetCheckedEntryCount(); theSubTotalData.nCol1 = rSubTotalData.nCol1; theSubTotalData.nRow1 = rSubTotalData.nRow1; theSubTotalData.nCol2 = rSubTotalData.nCol2; theSubTotalData.nRow2 = rSubTotalData.nRow2; theSubTotalData.bGroupActive[nGroupIdx] = (nGroup != 0); theSubTotalData.nField[nGroupIdx] = (nGroup != 0) ? nFieldArr[nGroup-1] : static_cast(0); if ( nEntryCount>0 && nCheckCount>0 && nGroup!=0 ) { sal_uInt16 nFunction = 0; pSubTotals.reset(new SCCOL [nCheckCount]); pFunctions.reset(new ScSubTotalFunc [nCheckCount]); for ( sal_uInt16 i=0, nCheck=0; iIsChecked( i ) ) { OSL_ENSURE( nCheck <= nCheckCount, "Range error :-(" ); nFunction = *((sal_uInt16*)mpLbColumns->GetEntryData( i )); pSubTotals[nCheck] = nFieldArr[i]; pFunctions[nCheck] = LbPosToFunc( nFunction ); nCheck++; } } theSubTotalData.SetSubTotals( nGroupNo, // Gruppen-Nr. pSubTotals.get(), pFunctions.get(), nCheckCount ); // Anzahl der Array-Elemente } rArgSet.Put( ScSubTotalItem( SCITEM_SUBTDATA, &theSubTotalData ) ); return true; } void ScTpSubTotalGroup::FillListBoxes() { OSL_ENSURE( pViewData && pDoc, "ViewData or Document not found :-/" ); if ( pViewData && pDoc ) { SCCOL nFirstCol = rSubTotalData.nCol1; SCROW nFirstRow = rSubTotalData.nRow1; SCTAB nTab = pViewData->GetTabNo(); SCCOL nMaxCol = rSubTotalData.nCol2; SCCOL col; OUString aFieldName; mpLbGroup->Clear(); mpLbColumns->Clear(); mpLbGroup->InsertEntry( aStrNone, 0 ); sal_uInt16 i=0; for ( col=nFirstCol; col<=nMaxCol && iGetString(col, nFirstRow, nTab); if ( aFieldName.isEmpty() ) { aFieldName = ScGlobal::ReplaceOrAppend( aStrColumn, "%1", ScColToAlpha( col )); } nFieldArr[i] = col; mpLbGroup->InsertEntry( aFieldName, i+1 ); mpLbColumns->InsertEntry( aFieldName, i ); mpLbColumns->SetEntryData( i, new sal_uInt16(0) ); i++; } // subsequent initialization of the constant: (sal_uInt16&)nFieldCount = i; } } sal_uInt16 ScTpSubTotalGroup::GetFieldSelPos( SCCOL nField ) { sal_uInt16 nFieldPos = 0; bool bFound = false; for ( sal_uInt16 n=0; nGetEntryCount() > 0) && (mpLbColumns->GetSelectionCount() > 0) ) { sal_uInt16 nFunction = mpLbFunctions->GetSelectEntryPos(); sal_uInt16 nColumn = mpLbColumns->GetSelectEntryPos(); sal_uInt16* pFunction = (sal_uInt16*)mpLbColumns->GetEntryData( nColumn ); OSL_ENSURE( pFunction, "EntryData not found!" ); if ( !pFunction ) return 0; if ( ((SvxCheckListBox*)pLb) == mpLbColumns ) { mpLbFunctions->SelectEntryPos( *pFunction ); } else if ( pLb == mpLbFunctions ) { *pFunction = nFunction; mpLbColumns->CheckEntryPos( nColumn, true ); } } return 0; } IMPL_LINK( ScTpSubTotalGroup, CheckHdl, ListBox *, pLb ) { if ( ((SvxCheckListBox*)pLb) == mpLbColumns ) { SvTreeListEntry* pEntry = mpLbColumns->GetHdlEntry(); if ( pEntry ) { mpLbColumns->SelectEntryPos( (sal_uInt16)mpLbColumns->GetModel()->GetAbsPos( pEntry ) ); SelectHdl( pLb ); } } return 0; } // Derived Group TabPages: SfxTabPage* ScTpSubTotalGroup1::Create( Window* pParent, const SfxItemSet& rArgSet ) { return ( new ScTpSubTotalGroup1( pParent, rArgSet ) ); } SfxTabPage* ScTpSubTotalGroup2::Create( Window* pParent, const SfxItemSet& rArgSet ) { return ( new ScTpSubTotalGroup2( pParent, rArgSet ) ); } SfxTabPage* ScTpSubTotalGroup3::Create( Window* pParent, const SfxItemSet& rArgSet ) { return ( new ScTpSubTotalGroup3( pParent, rArgSet ) ); } ScTpSubTotalGroup1::ScTpSubTotalGroup1( Window* pParent, const SfxItemSet& rArgSet ) : ScTpSubTotalGroup( pParent, rArgSet ) {} ScTpSubTotalGroup2::ScTpSubTotalGroup2( Window* pParent, const SfxItemSet& rArgSet ) : ScTpSubTotalGroup( pParent, rArgSet ) {} ScTpSubTotalGroup3::ScTpSubTotalGroup3( Window* pParent, const SfxItemSet& rArgSet ) : ScTpSubTotalGroup( pParent, rArgSet ) {} #define RESET(i) (ScTpSubTotalGroup::DoReset( (i), rArgSet )) void ScTpSubTotalGroup1::Reset( const SfxItemSet& rArgSet ) { RESET(1); } void ScTpSubTotalGroup2::Reset( const SfxItemSet& rArgSet ) { RESET(2); } void ScTpSubTotalGroup3::Reset( const SfxItemSet& rArgSet ) { RESET(3); } #undef RESET #define FILLSET(i) (ScTpSubTotalGroup::DoFillItemSet( (i), rArgSet )) bool ScTpSubTotalGroup1::FillItemSet( SfxItemSet& rArgSet ) { return FILLSET(1); } bool ScTpSubTotalGroup2::FillItemSet( SfxItemSet& rArgSet ) { return FILLSET(2); } bool ScTpSubTotalGroup3::FillItemSet( SfxItemSet& rArgSet ) { return FILLSET(3); } #undef FILL // Optionen-Tabpage: ScTpSubTotalOptions::ScTpSubTotalOptions( Window* pParent, const SfxItemSet& rArgSet ) : SfxTabPage ( pParent, "SubTotalOptionsPage", "modules/scalc/ui/subtotaloptionspage.ui" , rArgSet ), pViewData ( NULL ), pDoc ( NULL ), nWhichSubTotals ( rArgSet.GetPool()->GetWhich( SID_SUBTOTALS ) ), rSubTotalData ( ((const ScSubTotalItem&) rArgSet.Get( nWhichSubTotals )). GetSubTotalData() ) { get(pBtnPagebreak,"pagebreak"); get(pBtnCase,"case"); get(pBtnSort,"sort"); get(pFlSort,"label2"); get(pBtnAscending,"ascending"); get(pBtnDescending,"descending"); get(pBtnFormats,"formats"); get(pBtnUserDef,"btnuserdef"); get(pLbUserDef,"lbuserdef"); Init(); } ScTpSubTotalOptions::~ScTpSubTotalOptions() { } void ScTpSubTotalOptions::Init() { const ScSubTotalItem& rSubTotalItem = (const ScSubTotalItem&) GetItemSet().Get( nWhichSubTotals ); pViewData = rSubTotalItem.GetViewData(); pDoc = ( pViewData ) ? pViewData->GetDocument() : NULL; OSL_ENSURE( pViewData && pDoc, "ViewData or Document not found!" ); pBtnSort->SetClickHdl ( LINK( this, ScTpSubTotalOptions, CheckHdl ) ); pBtnUserDef->SetClickHdl ( LINK( this, ScTpSubTotalOptions, CheckHdl ) ); FillUserSortListBox(); } SfxTabPage* ScTpSubTotalOptions::Create( Window* pParent, const SfxItemSet& rArgSet ) { return ( new ScTpSubTotalOptions( pParent, rArgSet ) ); } void ScTpSubTotalOptions::Reset( const SfxItemSet& /* rArgSet */ ) { pBtnPagebreak->Check ( rSubTotalData.bPagebreak ); pBtnCase->Check ( rSubTotalData.bCaseSens ); pBtnFormats->Check ( rSubTotalData.bIncludePattern ); pBtnSort->Check ( rSubTotalData.bDoSort ); pBtnAscending->Check ( rSubTotalData.bAscending ); pBtnDescending->Check( !rSubTotalData.bAscending ); if ( rSubTotalData.bUserDef ) { pBtnUserDef->Check( true ); pLbUserDef->Enable(); pLbUserDef->SelectEntryPos( rSubTotalData.nUserIndex ); } else { pBtnUserDef->Check( false ); pLbUserDef->Disable(); pLbUserDef->SelectEntryPos( 0 ); } CheckHdl( pBtnSort ); } bool ScTpSubTotalOptions::FillItemSet( SfxItemSet& rArgSet ) { ScSubTotalParam theSubTotalData; // auslesen, wenn schon teilweise gefuellt SfxTabDialog* pDlg = GetTabDialog(); if ( pDlg ) { const SfxItemSet* pExample = pDlg->GetExampleSet(); const SfxPoolItem* pItem; if ( pExample && pExample->GetItemState( nWhichSubTotals, true, &pItem ) == SFX_ITEM_SET ) theSubTotalData = ((const ScSubTotalItem*)pItem)->GetSubTotalData(); } theSubTotalData.bPagebreak = pBtnPagebreak->IsChecked(); theSubTotalData.bReplace = true; theSubTotalData.bCaseSens = pBtnCase->IsChecked(); theSubTotalData.bIncludePattern = pBtnFormats->IsChecked(); theSubTotalData.bDoSort = pBtnSort->IsChecked(); theSubTotalData.bAscending = pBtnAscending->IsChecked(); theSubTotalData.bUserDef = pBtnUserDef->IsChecked(); theSubTotalData.nUserIndex = (pBtnUserDef->IsChecked()) ? pLbUserDef->GetSelectEntryPos() : 0; rArgSet.Put( ScSubTotalItem( nWhichSubTotals, &theSubTotalData ) ); return true; } void ScTpSubTotalOptions::FillUserSortListBox() { ScUserList* pUserLists = ScGlobal::GetUserList(); pLbUserDef->Clear(); if ( pUserLists ) { size_t nCount = pUserLists->size(); for ( size_t i=0; iInsertEntry( (*pUserLists)[i]->GetString() ); } } // Handler: IMPL_LINK( ScTpSubTotalOptions, CheckHdl, CheckBox *, pBox ) { if ( pBox == pBtnSort ) { if ( pBtnSort->IsChecked() ) { pFlSort->Enable(); pBtnFormats->Enable(); pBtnUserDef->Enable(); pBtnAscending->Enable(); pBtnDescending->Enable(); if ( pBtnUserDef->IsChecked() ) pLbUserDef->Enable(); } else { pFlSort->Disable(); pBtnFormats->Disable(); pBtnUserDef->Disable(); pBtnAscending->Disable(); pBtnDescending->Disable(); pLbUserDef->Disable(); } } else if ( pBox == pBtnUserDef ) { if ( pBtnUserDef->IsChecked() ) { pLbUserDef->Enable(); pLbUserDef->GrabFocus(); } else pLbUserDef->Disable(); } return 0; } ScTpSubTotalGroup1::~ScTpSubTotalGroup1() { } ScTpSubTotalGroup2::~ScTpSubTotalGroup2() { } ScTpSubTotalGroup3::~ScTpSubTotalGroup3() { } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */