diff options
Diffstat (limited to 'kspread/dialogs/kspread_dlg_subtotal.cc')
-rw-r--r-- | kspread/dialogs/kspread_dlg_subtotal.cc | 321 |
1 files changed, 321 insertions, 0 deletions
diff --git a/kspread/dialogs/kspread_dlg_subtotal.cc b/kspread/dialogs/kspread_dlg_subtotal.cc new file mode 100644 index 00000000..f64664b5 --- /dev/null +++ b/kspread/dialogs/kspread_dlg_subtotal.cc @@ -0,0 +1,321 @@ +/* This file is part of the KDE project + Copyright (C) 2002-2003 Norbert Andres <nandres@web.de> + (C) 2002 Philipp Mueller <philipp.mueller@gmx.de> + (C) 2002 Laurent Montel <montel@kde.org> + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library 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 + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + * Boston, MA 02110-1301, USA. +*/ + +#include <qcheckbox.h> +#include <qcombobox.h> +#include <qlistview.h> +#include <qmemarray.h> + +#include <kdebug.h> +#include <klocale.h> +#include <kmessagebox.h> + +#include "kspreadsubtotal.h" +#include "kspread_sheet.h" +#include "kspread_view.h" +#include "kspread_doc.h" +#include "kspread_util.h" + +#include "kspread_dlg_subtotal.h" + +using namespace KSpread; + +SubtotalDialog::SubtotalDialog( View * parent, QRect const & selection, const char * name ) + : KDialogBase(parent, name, true, i18n( "Subtotals" ), Ok | Cancel | User1, Ok, true, KGuiItem(i18n( "Remove All" )) ), + m_pView( parent ), + m_pSheet( m_pView->activeSheet() ), + m_selection( selection ), + m_dialog( new Subtotal( this ) ) +{ + setButtonBoxOrientation( Vertical ); + setMainWidget( m_dialog ); + + fillColumnBoxes(); + fillFunctionBox(); +} + +SubtotalDialog::~SubtotalDialog() +{ +} + +void SubtotalDialog::slotOk() +{ + int numOfCols = m_selection.width(); + QMemArray<int> columns( numOfCols ); + + int n = 0; + bool empty = true; + int left = m_selection.left(); + for ( QListViewItem * item = m_dialog->m_columnList->firstChild(); item; item = item->nextSibling() ) + { + if ( ((QCheckListItem * ) item)->isOn() ) + { + columns[n] = left + n; + empty = false; + } + else + columns[n] = -1; + ++n; + } + + if ( empty ) + { + KMessageBox::sorry( this, i18n("You need to select at least one column for adding subtotals.") ); + return; + } + + if ( m_dialog->m_replaceSubtotals->isChecked() ) + removeSubtotalLines(); + + int mainCol = left + m_dialog->m_columnBox->currentItem(); + int bottom = m_selection.bottom(); + int top = m_selection.top(); + left = m_selection.left(); + QString oldText = m_pSheet->cellAt( mainCol, top )->strOutText(); + QString newText; + QString result( " " + i18n("Result") ); + int lastChangedRow = top; + + m_pView->doc()->emitBeginOperation( false ); + bool ignoreEmptyCells = m_dialog->m_IgnoreBox->isChecked(); + bool addRow; + if ( !m_dialog->m_summaryOnly->isChecked() ) + { + int y = top + 1; + kdDebug() << "Starting in row " << y << endl; + while ( y <= bottom ) + { + addRow = true; + newText = m_pSheet->cellAt( mainCol, y )->strOutText(); + + if ( ignoreEmptyCells && (newText.length() == 0) ) + { + ++y; + kdDebug() << "Still the same -> " << y << endl; + continue; + } + + if (newText != oldText) + { + int saveY = y; + for (int x = 0; x < numOfCols; ++x) + { + kdDebug() << "Column: " << x << ", " << columns[x] << endl; + if (columns[x] != -1) + { + if (!addSubtotal( mainCol, columns[x], y - 1, lastChangedRow, addRow, oldText + result)) + reject(); + + if ( addRow ) + { + ++saveY; + ++bottom; + } + + addRow = false; + } + } + y = saveY; + lastChangedRow = y; + } + oldText = newText; + ++y; + } + + addRow = true; + for ( int x = 0; x < numOfCols; ++x ) + { + if ( columns[x] != -1 ) + { + if ( !addSubtotal( mainCol, columns[x], y - 1, lastChangedRow, addRow, oldText + result ) ) + reject(); + addRow = false; + } + } + ++y; + } + + if ( m_dialog->m_summaryBelow->isChecked() ) + { + addRow = true; + int bottom = m_selection.bottom(); + for (int x = 0; x < numOfCols; ++x) + { + if (columns[x] != -1) + { + addSubtotal( mainCol, columns[x], bottom, top, addRow, i18n("Grand Total") ); + addRow = false; + } + } + } + + m_pView->slotUpdateView( m_pView->activeSheet() ); + accept(); +} + +void SubtotalDialog::slotCancel() +{ + reject(); +} + +void SubtotalDialog::slotUser1() +{ + m_pView->doc()->emitBeginOperation( false ); + removeSubtotalLines(); + m_pView->slotUpdateView( m_pView->activeSheet() ); + accept(); +} + +void SubtotalDialog::removeSubtotalLines() +{ + kdDebug() << "Removing subtotal lines" << endl; + + int r = m_selection.right(); + int l = m_selection.left(); + int t = m_selection.top(); + + Cell * cell; + QString text; + + for ( int y = m_selection.bottom(); y >= t; --y ) + { + kdDebug() << "Checking row: " << y << endl; + bool containsSubtotal = false; + for (int x = l; x <= r; ++x ) + { + cell = m_pSheet->cellAt( x, y ); + if ( cell->isDefault() || !cell->isFormula() ) + continue; + + text = cell->text(); + if ( text.find( "SUBTOTAL" ) != -1 ) + { + containsSubtotal = true; + break; + } + } + + if ( containsSubtotal ) + { + kdDebug() << "Line " << y << " contains a subtotal " << endl; + QRect rect( l, y, m_selection.width(), 1 ); + + m_pSheet->unshiftColumn( rect ); + m_selection.setHeight( m_selection.height() - 1 ); + } + } + kdDebug() << "Done removing subtotals" << endl; +} + +void SubtotalDialog::fillColumnBoxes() +{ + int r = m_selection.right(); + int row = m_selection.top(); + + Cell * cell; + QCheckListItem * item; + + QString text; + QString col( i18n( "Column '%1' ") ); + + for ( int i = m_selection.left(); i <= r; ++i ) + { + cell = m_pSheet->cellAt( i, row ); + text = cell->strOutText(); + + if ( text.length() > 0 ) + { + text = col.arg( Cell::columnName( i ) ); + } + + m_dialog->m_columnBox->insertItem( text ); + + item = new QCheckListItem( m_dialog->m_columnList, + text, + QCheckListItem::CheckBox ); + item->setOn(false); + m_dialog->m_columnList->insertItem( item ); + } +} + +void SubtotalDialog::fillFunctionBox() +{ + QStringList lst; + lst << i18n( "Average" ); + lst << i18n( "Count" ); + lst << i18n( "CountA" ); + lst << i18n( "Max" ); + lst << i18n( "Min" ); + lst << i18n( "Product" ); + lst << i18n( "StDev" ); + lst << i18n( "StDevP" ); + lst << i18n( "Sum" ); + lst << i18n( "Var" ); + lst << i18n( "VarP" ); + m_dialog->m_functionBox->insertStringList(lst); +} + +bool SubtotalDialog::addSubtotal( int mainCol, int column, int row, int topRow, + bool addRow, QString const & text ) +{ + kdDebug() << "Adding subtotal: " << mainCol << ", " << column << ", Rows: " << row << ", " << topRow + << ": addRow: " << addRow << ", Text: " << text << endl; + if ( addRow ) + { + QRect rect(m_selection.left(), row + 1, m_selection.width(), 1); + if ( !m_pSheet->shiftColumn( rect ) ) + return false; + + m_selection.setHeight( m_selection.height() + 1 ); + + Cell * cell = m_pSheet->nonDefaultCell( mainCol, row + 1 ); + cell->setCellText( text ); + cell->format()->setTextFontBold( true ); + cell->format()->setTextFontItalic( true ); + cell->format()->setTextFontUnderline( true ); + } + + QString colName = Cell::columnName( column ); + + QString formula("=SUBTOTAL("); + formula += QString::number( m_dialog->m_functionBox->currentItem() + 1 ); + formula += "; "; + formula += colName; + formula += QString::number( topRow ); + // if ( topRow != row ) + { + formula += ":"; + formula += colName; + formula += QString::number( row ); + } + formula += ")"; + + Cell * cell = m_pSheet->nonDefaultCell( column, row + 1 ); + cell->setCellText( formula ); + cell->format()->setTextFontBold( true ); + cell->format()->setTextFontItalic( true ); + cell->format()->setTextFontUnderline( true ); + + return true; +} + +#include "kspread_dlg_subtotal.moc" + |