You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
7349 lines
232 KiB
7349 lines
232 KiB
/* This file is part of the KDE project
|
|
|
|
Copyright 2006 Stefan Nikolaus <stefan.nikolaus@kdemail.net>
|
|
Copyright 2005 Raphael Langerhorst <raphael.langerhorst@kdemail.net>
|
|
Copyright 2004-2005 Tomas Mecir <mecirt@gmail.com>
|
|
Copyright 2004-2006 Inge Wallin <inge@lysator.liu.se>
|
|
Copyright 1999-2002,2004,2005 Laurent Montel <montel@kde.org>
|
|
Copyright 2002-2005 Ariya Hidayat <ariya@kde.org>
|
|
Copyright 2001-2003 Philipp Mueller <philipp.mueller@gmx.de>
|
|
Copyright 2002-2003 Norbert Andres <nandres@web.de>
|
|
Copyright 2003 Reinhart Geiser <geiseri@kde.org>
|
|
Copyright 2003-2005 Meni Livne <livne@kde.org>
|
|
Copyright 2003 Peter Simonsson <psn@linux.se>
|
|
Copyright 1999-2002 David Faure <faure@kde.org>
|
|
Copyright 2000-2002 Werner Trobin <trobin@kde.org>
|
|
Copyright 1999,2002 Harri Porten <porten@kde.org>
|
|
Copyright 2002 John Dailey <dailey@vt.edu>
|
|
Copyright 1998-2000 Torben Weis <weis@kde.org>
|
|
Copyright 2000 Bernd Wuebben <wuebben@kde.org>
|
|
Copyright 2000 Simon Hausmann <hausmann@kde.org
|
|
Copyright 1999 Stephan Kulow <coolo@kde.org>
|
|
Copyright 1999 Michael Reiher <michael.reiher.gmx.de>
|
|
Copyright 1999 Boris Wedl <boris.wedl@kfunigraz.ac.at>
|
|
Copyright 1998-1999 Reginald Stadlbauer <reggie@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 <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <float.h>
|
|
#include <math.h>
|
|
|
|
#include <tqapplication.h>
|
|
#include <tqpopupmenu.h>
|
|
#include <tqregexp.h>
|
|
|
|
#include "kspread_canvas.h"
|
|
#include "kspread_condition.h"
|
|
#include "kspread_doc.h"
|
|
#include "kspread_format.h"
|
|
#include "kspread_global.h"
|
|
#include "kspread_map.h"
|
|
#include "kspread_sheet.h"
|
|
#include "kspread_sheetprint.h"
|
|
#include "kspread_style.h"
|
|
#include "kspread_style_manager.h"
|
|
#include "kspread_util.h"
|
|
#include "ksploadinginfo.h"
|
|
#include "kspread_genvalidationstyle.h"
|
|
#include "kspread_locale.h"
|
|
#include "kspread_value.h"
|
|
#include "kspread_view.h"
|
|
#include "kspread_value.h"
|
|
#include "formula.h"
|
|
#include "selection.h"
|
|
#include "valueconverter.h"
|
|
#include "valueformatter.h"
|
|
#include "valueparser.h"
|
|
|
|
#include <KoStyleStack.h>
|
|
#include <KoRect.h>
|
|
#include <KoXmlNS.h>
|
|
#include <KoDom.h>
|
|
#include <KoXmlWriter.h>
|
|
#include <KoOasisStyles.h>
|
|
|
|
#include <tdemessagebox.h>
|
|
|
|
#include <kdebug.h>
|
|
|
|
using namespace KSpread;
|
|
|
|
#define BORDER_SPACE 1
|
|
|
|
|
|
/**
|
|
* A pointer to the decimal separator
|
|
*/
|
|
|
|
namespace Cell_LNS
|
|
{
|
|
TQChar decimal_point = '\0';
|
|
}
|
|
|
|
using namespace Cell_LNS;
|
|
|
|
|
|
// Some variables are placed in Cell::Extra because normally they're
|
|
// not required in simple case of cell(s). For example, most plain
|
|
// text cells don't need to store information about spanned columns
|
|
// and rows, as this is only the case with merged cells.
|
|
//
|
|
// When the cell is getting complex (e.g. merged with other cells,
|
|
// contains rich text, has validation criteria, etc), this Cell::Extra
|
|
// is allocated by Cell::Private and starts to be
|
|
// available. Otherwise, it won't exist at all.
|
|
|
|
class Cell::Extra
|
|
{
|
|
public:
|
|
Extra() {}
|
|
|
|
// Not empty when the cell holds a link
|
|
TQString link;
|
|
|
|
// Number of cells explicitly merged by the user in X and Y directions.
|
|
int mergedXCells;
|
|
int mergedYCells;
|
|
|
|
// Number of additional cells.
|
|
int extraXCells;
|
|
int extraYCells;
|
|
|
|
// If this cell overlaps other cells, then we have the cells width and
|
|
// height stored here. These values do not mean anything unless
|
|
// extraXCells and/or extraYCells are different from 0.
|
|
double extraWidth;
|
|
double extraHeight;
|
|
|
|
// A list of cells that obscure this one.
|
|
// If this list is not empty, then this cell is obscured by another
|
|
// enlarged object. This means that we have to call this object in order
|
|
// of painting it for example instead of painting 'this'.
|
|
//
|
|
// FIXME (comment): If the list consists of more than one obscuring
|
|
// element, then is there an order between them that
|
|
// is important?
|
|
TQValueList<Cell*> obscuringCells;
|
|
|
|
// If non-NULL, contains a pointer to a condition or a validity test.
|
|
Conditions *conditions;
|
|
Validity *validity;
|
|
|
|
// Store the number of line when multirow is used (default is 0)
|
|
int nbLines;
|
|
|
|
private:
|
|
// Don't allow implicit copy.
|
|
Extra& operator=( const Extra& );
|
|
};
|
|
|
|
|
|
class Cell::Private
|
|
{
|
|
public:
|
|
|
|
Private();
|
|
~Private();
|
|
|
|
public:
|
|
|
|
// This cell's row and column. If either of them is 0, this is the
|
|
// default cell and its row/column can not be determined. Note that
|
|
// in the isDefault() method, only column is tested.
|
|
int row;
|
|
int column;
|
|
|
|
// Value of the cell, either typed by user or as result of formula
|
|
Value value;
|
|
|
|
// Holds the user's input.
|
|
//
|
|
// FIXME:
|
|
// Eventually, we'll want to get rid of strText and generate
|
|
// user's input on-the-fly. Then, for normal cells, we'll generate
|
|
// this string using converter()->asString
|
|
// (value()).
|
|
//
|
|
// Here the problem is, that strText also holds the formula -
|
|
// we'll need to provide some method to generate it from the
|
|
// parsed version, created in KSpread::Formula. Hence, we won't be
|
|
// able to get rid of strText until we switch to the new formula
|
|
// parser and until we write some method that re-generates the
|
|
// input formula...
|
|
//
|
|
// Alternately, we can keep using strText for formulas and
|
|
// generate it dynamically for static cells...
|
|
//
|
|
// /Tomas
|
|
//
|
|
TQString strText;
|
|
|
|
// This is the text we want to display. Not necessarily the same
|
|
// as strText, e.g. strText="1" and strOutText="1.00" Also holds
|
|
// value that we got from calculation, formerly known as
|
|
// strFormulaOut
|
|
TQString strOutText;
|
|
|
|
// the Formula object for the cell
|
|
KSpread::Formula *formula;
|
|
|
|
// Position and dimension of displayed text.
|
|
// FIXME (comment): Which coordinate system? pixels? mm/cm? zoom?
|
|
double textX;
|
|
double textY;
|
|
double textWidth;
|
|
double textHeight;
|
|
|
|
// result of "fm.ascent()" in makeLayout. used in offsetAlign.
|
|
int fmAscent;
|
|
|
|
// Pointers to neighboring cells.
|
|
// FIXME (comment): Which order?
|
|
Cell *nextCell;
|
|
Cell *previousCell;
|
|
|
|
bool hasExtra() const { return (cellExtra != 0); };
|
|
Extra *extra();
|
|
|
|
Format *format;
|
|
TQ_UINT32 flags;
|
|
|
|
private:
|
|
// "Extra stuff", see explanation for Cell::Extra.
|
|
Extra *cellExtra;
|
|
};
|
|
|
|
|
|
Cell::Private::Private()
|
|
{
|
|
// Some basic data.
|
|
row = 0;
|
|
column = 0;
|
|
value = Value::empty();
|
|
formula = 0;
|
|
|
|
// Formatting
|
|
textX = 0.0;
|
|
textY = 0.0;
|
|
textWidth = 0.0;
|
|
textHeight = 0.0;
|
|
fmAscent = 0;
|
|
|
|
nextCell = 0;
|
|
previousCell = 0;
|
|
|
|
// Default is to not have the "extra" stuff in a cell.
|
|
cellExtra = 0;
|
|
format = 0;
|
|
flags = 0;
|
|
}
|
|
|
|
|
|
Cell::Private::~Private()
|
|
{
|
|
delete cellExtra;
|
|
delete formula;
|
|
}
|
|
|
|
|
|
Cell::Extra* Cell::Private::extra()
|
|
{
|
|
if ( !cellExtra ) {
|
|
cellExtra = new Extra;
|
|
cellExtra->conditions = 0;
|
|
cellExtra->validity = 0;
|
|
|
|
cellExtra->mergedXCells = 0;
|
|
cellExtra->mergedYCells = 0;
|
|
cellExtra->extraXCells = 0;
|
|
cellExtra->extraYCells = 0;
|
|
cellExtra->extraWidth = 0.0;
|
|
cellExtra->extraHeight = 0.0;
|
|
cellExtra->nbLines = 0;
|
|
// cellExtra->highlight = TQColor(0,0,0);
|
|
}
|
|
|
|
return cellExtra;
|
|
}
|
|
|
|
|
|
/*****************************************************************************
|
|
*
|
|
* Cell
|
|
*
|
|
*****************************************************************************/
|
|
|
|
|
|
Cell::Cell( Sheet * _sheet, int _column, int _row )
|
|
{
|
|
d = new Private;
|
|
d->row = _row;
|
|
d->column = _column;
|
|
d->format = new Format(_sheet, _sheet->doc()->styleManager()->defaultStyle());
|
|
d->format->setCell(this);
|
|
clearAllErrors();
|
|
}
|
|
|
|
|
|
Cell::Cell( Sheet * _sheet, Style * _style, int _column, int _row )
|
|
{
|
|
d = new Private;
|
|
d->row = _row;
|
|
d->column = _column;
|
|
d->format = new Format( _sheet, _style );
|
|
d->format->setCell(this);
|
|
|
|
clearAllErrors();
|
|
}
|
|
|
|
Format* Cell::format() const
|
|
{
|
|
return d->format;
|
|
}
|
|
|
|
// Return the sheet that this cell belongs to.
|
|
Sheet * Cell::sheet() const
|
|
{
|
|
return d->format->sheet();
|
|
}
|
|
|
|
// Return true if this is the default cell.
|
|
bool Cell::isDefault() const
|
|
{
|
|
return ( d->column == 0 );
|
|
}
|
|
|
|
// Return the row number of this cell.
|
|
int Cell::row() const
|
|
{
|
|
// Make sure this isn't called for the default cell. This assert
|
|
// can save you (could have saved me!) the hassle of some very
|
|
// obscure bugs.
|
|
|
|
if ( isDefault() )
|
|
{
|
|
kdWarning(36001) << "Error: Calling Cell::row() for default cell" << endl;
|
|
return 0;
|
|
}
|
|
|
|
return d->row;
|
|
}
|
|
|
|
|
|
// Return the column number of this cell.
|
|
//
|
|
int Cell::column() const
|
|
{
|
|
// Make sure this isn't called for the default cell. This assert
|
|
// can save you (could have saved me!) the hassle of some very
|
|
// obscure bugs.
|
|
if ( isDefault() )
|
|
{
|
|
kdWarning(36001) << "Error: Calling Cell::column() for default cell" << endl;
|
|
return 0;
|
|
}
|
|
return d->column;
|
|
}
|
|
|
|
|
|
// Return the name of this cell, i.e. the string that the user would
|
|
// use to reference it. Example: A1, BZ16
|
|
//
|
|
TQString Cell::name() const
|
|
{
|
|
return name( d->column, d->row );
|
|
}
|
|
|
|
|
|
// Return the name of any cell given by (col, row).
|
|
//
|
|
TQString Cell::name( int col, int row )
|
|
{
|
|
return columnName( col ) + TQString::number( row );
|
|
}
|
|
|
|
|
|
// Return the name of this cell, including the sheet name.
|
|
// Example: sheet1!A5
|
|
//
|
|
TQString Cell::fullName() const
|
|
{
|
|
return fullName( sheet(), d->column, d->row );
|
|
}
|
|
|
|
|
|
// Return the full name of any cell given a sheet and (col, row).
|
|
//
|
|
TQString Cell::fullName( const Sheet* s, int col, int row )
|
|
{
|
|
return s->sheetName() + "!" + name( col, row );
|
|
}
|
|
|
|
|
|
// Return the symbolic name of the column of this cell. Examples: A, BB.
|
|
//
|
|
TQString Cell::columnName() const
|
|
{
|
|
return columnName( d->column );
|
|
}
|
|
|
|
TDELocale* Cell::locale() const
|
|
{
|
|
return d->format->sheet()->doc()->locale();
|
|
}
|
|
|
|
// Return the symbolic name of any column.
|
|
//
|
|
TQString Cell::columnName( uint column )
|
|
{
|
|
TQString str;
|
|
unsigned digits = 1;
|
|
unsigned offset = 0;
|
|
|
|
column--;
|
|
|
|
if( column > 4058115285U ) return TQString("@@@");
|
|
|
|
for( unsigned limit = 26; column >= limit+offset; limit *= 26, digits++ )
|
|
offset += limit;
|
|
|
|
for( unsigned c = column - offset; digits; --digits, c/=26 )
|
|
str.prepend( TQChar( 'A' + (c%26) ) );
|
|
|
|
return str;
|
|
}
|
|
|
|
|
|
// Return true if this cell is a formula.
|
|
//
|
|
bool Cell::isFormula() const
|
|
{
|
|
return d->strText[0] == '=';
|
|
}
|
|
|
|
|
|
// Return the input text of this cell. This could, for instance, be a
|
|
// formula.
|
|
//
|
|
// FIXME: These two functions are inconsistently named. It should be
|
|
// either text() and outText() or strText() and strOutText().
|
|
//
|
|
TQString Cell::text() const
|
|
{
|
|
return d->strText;
|
|
}
|
|
|
|
|
|
// Return the out text, i.e. the text that is visible in the cells
|
|
// square when shown. This could, for instance, be the calculated
|
|
// result of a formula.
|
|
//
|
|
TQString Cell::strOutText() const
|
|
{
|
|
return d->strOutText;
|
|
}
|
|
|
|
Formula *Cell::formula () const
|
|
{
|
|
return d->formula;
|
|
}
|
|
|
|
|
|
// Return the value of this cell.
|
|
//
|
|
const Value Cell::value() const
|
|
{
|
|
return d->value;
|
|
}
|
|
|
|
|
|
// Set the value of this cell. It also clears all errors if the value
|
|
// itself is not an error.
|
|
//
|
|
// In addition to this, it calculates the outstring and sets the dirty
|
|
// flags so that a redraw is forced.
|
|
//
|
|
void Cell::setValue( const Value& v )
|
|
{
|
|
if (v.type() != Value::Error)
|
|
clearAllErrors();
|
|
|
|
//If the value has not changed then we don't need to do anything
|
|
//(ie. no need to relayout, update dependant cells etc.),
|
|
//unless this cell contains a formula, in which case its dependancies might have changed
|
|
//even though the value has not. For example, if this cell was previously empty (and its value is
|
|
//therefore empty) and a new dependency upon an empty cell has been added. The new value would still
|
|
//be empty, but the dependencies need to be updated (via the call to valueChanged() below).
|
|
if ( ( d->value == v ) && ( !isFormula() ) )
|
|
return;
|
|
|
|
d->value = v;
|
|
|
|
setFlag(Flag_LayoutDirty);
|
|
setFlag(Flag_TextFormatDirty);
|
|
|
|
// Format and set the outText.
|
|
setOutputText();
|
|
|
|
// Set the displayed text, if we hold an error value.
|
|
if (d->value.type() == Value::Error)
|
|
d->strOutText = d->value.errorMessage ();
|
|
|
|
// Value of the cell has changed - trigger necessary actions
|
|
valueChanged ();
|
|
|
|
if ( !format()->sheet()->isLoading() )
|
|
format()->sheet()->setRegionPaintDirty(cellRect());
|
|
}
|
|
|
|
void Cell::setCellValue (const Value &v, FormatType fmtType, const TQString &txt)
|
|
{
|
|
if ( !txt.isNull() )
|
|
{
|
|
d->strText = txt;
|
|
if ( isFormula() )
|
|
makeFormula();
|
|
}
|
|
else if ( !isFormula() )
|
|
d->strText = sheet()->doc()->converter()->asString (v).asString();
|
|
if (fmtType != No_format)
|
|
format()->setFormatType (fmtType);
|
|
setValue (v);
|
|
}
|
|
|
|
// FIXME: Continue commenting and cleaning here (ingwa)
|
|
|
|
|
|
Cell* Cell::previousCell() const
|
|
{
|
|
return d->previousCell;
|
|
}
|
|
|
|
Cell* Cell::nextCell() const
|
|
{
|
|
return d->nextCell;
|
|
}
|
|
|
|
void Cell::setPreviousCell( Cell* c )
|
|
{
|
|
d->previousCell = c;
|
|
}
|
|
|
|
void Cell::setNextCell( Cell* c )
|
|
{
|
|
d->nextCell = c;
|
|
}
|
|
|
|
Validity* Cell::getValidity( int newStruct )
|
|
{
|
|
if ( (!newStruct) && (!d->hasExtra()))
|
|
//we don't have validity struct and we don't want one
|
|
return 0;
|
|
|
|
if( ( d->extra()->validity == 0 ) && ( newStruct == -1 ) )
|
|
d->extra()->validity = new Validity;
|
|
return d->extra()->validity;
|
|
}
|
|
|
|
void Cell::removeValidity()
|
|
{
|
|
if (!d->hasExtra())
|
|
return;
|
|
|
|
delete d->extra()->validity;
|
|
d->extra()->validity = 0;
|
|
}
|
|
|
|
|
|
void Cell::copyFormat( const int column , const int row )
|
|
{
|
|
const Cell * cell = format()->sheet()->cellAt( column , row );
|
|
|
|
copyFormat( cell );
|
|
}
|
|
|
|
void Cell::copyFormat( const Cell* cell )
|
|
{
|
|
|
|
Q_ASSERT(cell);
|
|
|
|
d->value.setFormat(cell->d->value.format());
|
|
format()->copy(*(cell->format()));
|
|
|
|
/*format()->setAlign( cell->format()->align( _column, _row ) );
|
|
format()->setAlignY( cell->format()->alignY( _column, _row ) );
|
|
format()->setTextFont( cell->format()->textFont( _column, _row ) );
|
|
format()->setTextColor( cell->format()->textColor( _column, _row ) );
|
|
format()->setBgColor( cell->bgColor( _column, _row) );
|
|
setLeftBorderPen( cell->leftBorderPen( _column, _row ) );
|
|
setTopBorderPen( cell->topBorderPen( _column, _row ) );
|
|
setBottomBorderPen( cell->bottomBorderPen( _column, _row ) );
|
|
setRightBorderPen( cell->rightBorderPen( _column, _row ) );
|
|
format()->setFallDiagonalPen( cell->format()->fallDiagonalPen( _column, _row ) );
|
|
format()->setGoUpDiagonalPen( cell->format()->goUpDiagonalPen( _column, _row ) );
|
|
format()->setBackGroundBrush( cell->backGroundBrush( _column, _row) );
|
|
format()->setPrecision( cell->format()->precision( _column, _row ) );
|
|
format()->setPrefix( cell->format()->prefix( _column, _row ) );
|
|
format()->setPostfix( cell->format()->postfix( _column, _row ) );
|
|
format()->setFloatFormat( cell->format()->floatFormat( _column, _row ) );
|
|
format()->setFloatColor( cell->format()->floatColor( _column, _row ) );
|
|
format()->setMultiRow( cell->format()->multiRow( _column, _row ) );
|
|
format()->setVerticalText( cell->format()->verticalText( _column, _row ) );
|
|
format()->setDontPrintText( cell->format()->getDontprintText(_column, _row ) );
|
|
format()->setNotProtected( cell->format()->notProtected(_column, _row ) );
|
|
format()->setHideAll(cell->format()->isHideAll(_column, _row ) );
|
|
format()->setHideFormula(cell->format()->isHideFormula(_column, _row ) );
|
|
format()->setIndent( cell->format()->getIndent(_column, _row ) );
|
|
format()->setAngle( cell->format()->getAngle(_column, _row) );
|
|
format()->setFormatType( cell->format()->getFormatType(_column, _row) );
|
|
Format::Currency c;
|
|
if ( cell->format()->currencyInfo( c ) )
|
|
format()->setCurrency( c );*/
|
|
|
|
TQValueList<Conditional> conditionList = cell->conditionList();
|
|
if (d->hasExtra())
|
|
delete d->extra()->conditions;
|
|
if ( cell->d->hasExtra() && cell->d->extra()->conditions )
|
|
setConditionList( conditionList );
|
|
else
|
|
if (d->hasExtra())
|
|
d->extra()->conditions = 0;
|
|
|
|
/*format()->setComment( cell->format()->comment( _column, _row ) );*/
|
|
}
|
|
|
|
void Cell::copyAll( Cell *cell )
|
|
{
|
|
Q_ASSERT( !isDefault() ); // trouble ahead...
|
|
copyFormat( cell );
|
|
copyContent( cell );
|
|
}
|
|
|
|
void Cell::copyContent( const Cell* cell )
|
|
{
|
|
Q_ASSERT( !isDefault() ); // trouble ahead...
|
|
|
|
if (cell->isFormula() && cell->column() > 0 && cell->row() > 0)
|
|
{
|
|
// change all the references, e.g. from A1 to A3 if copying
|
|
// from e.g. B2 to B4
|
|
TQString d = cell->encodeFormula();
|
|
setCellText( cell->decodeFormula( d ) );
|
|
}
|
|
else
|
|
setCellText( cell->text() );
|
|
|
|
}
|
|
|
|
void Cell::defaultStyle()
|
|
{
|
|
format()->defaultStyleFormat();
|
|
|
|
if (!d->hasExtra())
|
|
return;
|
|
|
|
if ( d->extra()->conditions )
|
|
{
|
|
delete d->extra()->conditions;
|
|
d->extra()->conditions = 0;
|
|
}
|
|
|
|
delete d->extra()->validity;
|
|
d->extra()->validity = 0L;
|
|
}
|
|
|
|
|
|
// Merge a number of cells, i.e. make this cell obscure a number of
|
|
// other cells. If _x and _y == 0, then the merging is removed.
|
|
|
|
void Cell::mergeCells( int _col, int _row, int _x, int _y )
|
|
{
|
|
// Start by unobscuring the cells that we obscure right now
|
|
int extraXCells = d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
int extraYCells = d->hasExtra() ? d->extra()->extraYCells : 0;
|
|
for ( int x = _col; x <= _col + extraXCells; ++x ) {
|
|
for ( int y = _row; y <= _row + extraYCells; ++y ) {
|
|
if ( x != _col || y != _row )
|
|
format()->sheet()->nonDefaultCell( x, y )->unobscure(this);
|
|
}
|
|
}
|
|
|
|
// If no merging, then remove all traces, and return.
|
|
if ( _x == 0 && _y == 0 ) {
|
|
clearFlag( Flag_Merged );
|
|
if (d->hasExtra()) {
|
|
d->extra()->extraXCells = 0;
|
|
d->extra()->extraYCells = 0;
|
|
d->extra()->extraWidth = 0.0;
|
|
d->extra()->extraHeight = 0.0;
|
|
d->extra()->mergedXCells = 0;
|
|
d->extra()->mergedYCells = 0;
|
|
}
|
|
|
|
// Refresh the layout
|
|
setFlag( Flag_LayoutDirty );
|
|
return;
|
|
}
|
|
|
|
// At this point, we know that we will merge some cells.
|
|
setFlag(Flag_Merged);
|
|
d->extra()->extraXCells = _x;
|
|
d->extra()->extraYCells = _y;
|
|
d->extra()->mergedXCells = _x;
|
|
d->extra()->mergedYCells = _y;
|
|
|
|
// Obscure the cells
|
|
for ( int x = _col; x <= _col + _x; ++x ) {
|
|
for ( int y = _row; y <= _row + _y; ++y ) {
|
|
if ( x != _col || y != _row )
|
|
format()->sheet()->nonDefaultCell( x, y )->obscure( this, true );
|
|
}
|
|
}
|
|
|
|
// Refresh the layout
|
|
setFlag( Flag_LayoutDirty );
|
|
}
|
|
|
|
void Cell::move( int col, int row )
|
|
{
|
|
setLayoutDirtyFlag();
|
|
setCalcDirtyFlag();
|
|
setDisplayDirtyFlag();
|
|
|
|
//int ex = extraXCells();
|
|
//int ey = d->extra()->extraYCells();
|
|
|
|
if (d->hasExtra())
|
|
d->extra()->obscuringCells.clear();
|
|
|
|
// Unobscure the objects we obscure right now
|
|
int extraXCells = d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
int extraYCells = d->hasExtra() ? d->extra()->extraYCells : 0;
|
|
for( int x = d->column; x <= d->column + extraXCells; ++x )
|
|
for( int y = d->row; y <= d->row + extraYCells; ++y )
|
|
if ( x != d->column || y != d->row )
|
|
{
|
|
Cell *cell = format()->sheet()->nonDefaultCell( x, y );
|
|
cell->unobscure(this);
|
|
}
|
|
|
|
d->column = col;
|
|
d->row = row;
|
|
|
|
if (d->hasExtra())
|
|
{
|
|
// d->extra()->extraXCells = 0;
|
|
// d->extra()->extraYCells = 0;
|
|
d->extra()->mergedXCells = 0;
|
|
d->extra()->mergedYCells = 0;
|
|
}
|
|
|
|
// Cell value has been changed (because we're another cell now).
|
|
valueChanged ();
|
|
}
|
|
|
|
void Cell::setLayoutDirtyFlag( bool format )
|
|
{
|
|
setFlag( Flag_LayoutDirty );
|
|
if ( format )
|
|
setFlag( Flag_TextFormatDirty );
|
|
|
|
if (!d->hasExtra())
|
|
return;
|
|
|
|
TQValueList<Cell*>::iterator it = d->extra()->obscuringCells.begin();
|
|
TQValueList<Cell*>::iterator end = d->extra()->obscuringCells.end();
|
|
for ( ; it != end; ++it ) {
|
|
(*it)->setLayoutDirtyFlag( format );
|
|
}
|
|
}
|
|
|
|
bool Cell::needsPrinting() const
|
|
{
|
|
if ( isDefault() )
|
|
return false;
|
|
|
|
if ( !d->strText.stripWhiteSpace().isEmpty() ) {
|
|
return true;
|
|
}
|
|
|
|
// Cell borders?
|
|
if ( format()->hasProperty( Format::PTopBorder )
|
|
|| format()->hasProperty( Format::PLeftBorder )
|
|
|| format()->hasProperty( Format::PRightBorder )
|
|
|| format()->hasProperty( Format::PBottomBorder )
|
|
|| format()->hasProperty( Format::PFallDiagonal )
|
|
|| format()->hasProperty( Format::PGoUpDiagonal ) ) {
|
|
return true;
|
|
}
|
|
|
|
// Background color or brush?
|
|
if ( format()->hasProperty( Format::PBackgroundBrush ) ) {
|
|
|
|
const TQBrush& brush=backGroundBrush(column(),row());
|
|
|
|
//Only brushes that are visible (ie. they have a brush style and are not white)
|
|
//need to be drawn
|
|
if ( (brush.style() != TQt::NoBrush) &&
|
|
(brush.color() != TQt::white || brush.pixmap()) )
|
|
return true;
|
|
|
|
}
|
|
|
|
if ( format()->hasProperty( Format::PBackgroundColor ) ) {
|
|
kdDebug() << "needsPrinting: Has background colour" << endl;
|
|
TQColor backgroundColor=bgColor(column(),row());
|
|
|
|
//We don't need to print anything if the background is white
|
|
if (backgroundColor != TQt::white)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool Cell::isEmpty() const
|
|
{
|
|
return isDefault() || d->strText.isEmpty();
|
|
}
|
|
|
|
|
|
// Return true if this cell is obscured by some other cell.
|
|
|
|
bool Cell::isObscured() const
|
|
{
|
|
if (!d->hasExtra())
|
|
return false;
|
|
|
|
return !( d->extra()->obscuringCells.isEmpty() );
|
|
}
|
|
|
|
|
|
// Return true if this cell is part of a merged cell, but not the
|
|
// master cell.
|
|
|
|
bool Cell::isPartOfMerged() const
|
|
{
|
|
if (!d->hasExtra())
|
|
return false;
|
|
|
|
TQValueList<Cell*>::const_iterator it = d->extra()->obscuringCells.begin();
|
|
TQValueList<Cell*>::const_iterator end = d->extra()->obscuringCells.end();
|
|
for ( ; it != end; ++it ) {
|
|
Cell *cell = *it;
|
|
|
|
if (cell->doesMergeCells()) {
|
|
// The cell might merge extra cells, and then overlap even
|
|
// beyond that so just knowing that the obscuring cell merges
|
|
// extra isn't enough. We have to know that this cell is one of
|
|
// the ones it is forcing over.
|
|
if (column() <= cell->column() + cell->d->extra()->mergedXCells
|
|
&& row() <= cell->row() + cell->mergedYCells() )
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
// Return the cell that obscures this one. If no cell is obscuring,
|
|
// then return this. This method is slightly complicated because we
|
|
// can have several layers of obscuring.
|
|
//
|
|
// Update: it seems that if we do an actual merge, then the obscuring
|
|
// cell is prepended and if just expanding, then it is appended. This
|
|
// means that we should be able to just look at the first one.
|
|
|
|
Cell *Cell::ultimateObscuringCell() const
|
|
{
|
|
if (!d->hasExtra())
|
|
return (Cell *) this;
|
|
|
|
else if (d->extra()->obscuringCells.isEmpty())
|
|
return (Cell *) this;
|
|
|
|
else
|
|
return d->extra()->obscuringCells.first();
|
|
|
|
#if 0
|
|
TQValueList<Cell*>::const_iterator it = d->extra()->obscuringCells.begin();
|
|
TQValueList<Cell*>::const_iterator end = d->extra()->obscuringCells.end();
|
|
for ( ; it != end; ++it ) {
|
|
Cell *cell = *it;
|
|
|
|
if (cell->doesMergeCells()) {
|
|
// The cell might merge extra cells, and then overlap even
|
|
// beyond that so just knowing that the obscuring cell merges
|
|
// extra isn't enough. We have to know that this cell is one of
|
|
// the ones it is forcing over.
|
|
if (column() <= cell->column() + cell->d->extra()->mergedXCells
|
|
&& row() <= cell->row() + cell->mergedYCells() )
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
|
|
TQValueList<Cell*> Cell::obscuringCells() const
|
|
{
|
|
if (!d->hasExtra())
|
|
{
|
|
TQValueList<Cell*> empty;
|
|
return empty;
|
|
}
|
|
return d->extra()->obscuringCells;
|
|
}
|
|
|
|
void Cell::clearObscuringCells()
|
|
{
|
|
if (!d->hasExtra())
|
|
return;
|
|
d->extra()->obscuringCells.clear();
|
|
}
|
|
|
|
void Cell::obscure( Cell *cell, bool isForcing )
|
|
{
|
|
if (d->hasExtra())
|
|
{
|
|
d->extra()->obscuringCells.remove( cell ); // removes *all* occurrences
|
|
cell->clearObscuringCells();
|
|
}
|
|
if ( isForcing )
|
|
{
|
|
d->extra()->obscuringCells.prepend( cell );
|
|
}
|
|
else
|
|
{
|
|
d->extra()->obscuringCells.append( cell );
|
|
}
|
|
setFlag(Flag_LayoutDirty);
|
|
format()->sheet()->setRegionPaintDirty( cellRect() );
|
|
}
|
|
|
|
void Cell::unobscure( Cell * cell )
|
|
{
|
|
if (d->hasExtra())
|
|
d->extra()->obscuringCells.remove( cell );
|
|
setFlag( Flag_LayoutDirty );
|
|
format()->sheet()->setRegionPaintDirty( cellRect() );
|
|
}
|
|
|
|
TQString Cell::encodeFormula( bool _era, int _col, int _row ) const
|
|
{
|
|
if ( _col == -1 )
|
|
_col = d->column;
|
|
if ( _row == -1 )
|
|
_row = d->row;
|
|
|
|
TQString erg = "";
|
|
|
|
if(d->strText.isEmpty())
|
|
return d->strText;
|
|
|
|
bool fix1 = false;
|
|
bool fix2 = false;
|
|
bool onNumber = false;
|
|
unsigned int pos = 0;
|
|
const unsigned int length = d->strText.length();
|
|
|
|
// All this can surely be made 10 times faster, but I just "ported" it to TQString
|
|
// without any attempt to optimize things -- this is really brittle (Werner)
|
|
while ( pos < length )
|
|
{
|
|
if ( d->strText[pos] == '"' )
|
|
{
|
|
erg += d->strText[pos++];
|
|
while ( pos < length && d->strText[pos] != '"' ) // till the end of the world^H^H^H "string"
|
|
{
|
|
erg += d->strText[pos++];
|
|
// Allow escaped double quotes (\")
|
|
if ( pos < length && d->strText[pos] == '\\' && d->strText[pos+1] == '"' )
|
|
{
|
|
erg += d->strText[pos++];
|
|
erg += d->strText[pos++];
|
|
}
|
|
}
|
|
if ( pos < length ) // also copy the trailing double quote
|
|
erg += d->strText[pos++];
|
|
|
|
onNumber = false;
|
|
}
|
|
else if ( d->strText[pos].isDigit() )
|
|
{
|
|
erg += d->strText[pos++];
|
|
fix1 = fix2 = false;
|
|
onNumber = true;
|
|
}
|
|
else if ( d->strText[pos] != '$' && !d->strText[pos].isLetter() )
|
|
{
|
|
erg += d->strText[pos++];
|
|
fix1 = fix2 = false;
|
|
onNumber = false;
|
|
}
|
|
else
|
|
{
|
|
TQString tmp = "";
|
|
if ( d->strText[pos] == '$' )
|
|
{
|
|
tmp = "$";
|
|
pos++;
|
|
fix1 = true;
|
|
}
|
|
if ( d->strText[pos].isLetter() )
|
|
{
|
|
TQString buffer;
|
|
unsigned int pos2 = 0;
|
|
while ( pos < length && d->strText[pos].isLetter() )
|
|
{
|
|
tmp += d->strText[pos];
|
|
buffer[pos2++] = d->strText[pos++];
|
|
}
|
|
if ( d->strText[pos] == '$' )
|
|
{
|
|
tmp += "$";
|
|
pos++;
|
|
fix2 = true;
|
|
}
|
|
if ( d->strText[pos].isDigit() )
|
|
{
|
|
const unsigned int oldPos = pos;
|
|
while ( pos < length && d->strText[pos].isDigit() ) ++pos;
|
|
int row = 0;
|
|
if ( pos != oldPos )
|
|
row = d->strText.mid(oldPos, pos-oldPos).toInt();
|
|
// Is it a sheet name || is it a function name like DEC2HEX
|
|
/* or if we're parsing a number, this could just be the
|
|
exponential part of it (1.23E4) */
|
|
if ( ( d->strText[pos] == '!' ) ||
|
|
d->strText[pos].isLetter() ||
|
|
onNumber )
|
|
{
|
|
erg += tmp;
|
|
fix1 = fix2 = false;
|
|
pos = oldPos;
|
|
}
|
|
else // It must be a cell identifier
|
|
{
|
|
//now calculate the row as integer value
|
|
int col = 0;
|
|
col = util_decodeColumnLabelText( buffer );
|
|
if ( fix1 )
|
|
erg += TQString( "$%1" ).arg( col );
|
|
else
|
|
if (_era)
|
|
erg += TQChar(0xA7) + TQString( "%1" ).arg( col );
|
|
else
|
|
erg += TQString( "#%1" ).arg( col - _col );
|
|
|
|
if ( fix2 )
|
|
erg += TQString( "$%1#").arg( row );
|
|
else
|
|
if (_era)
|
|
erg += TQChar(0xA7) + TQString( "%1#" ).arg( row );
|
|
else
|
|
erg += TQString( "#%1#" ).arg( row - _row );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
erg += tmp;
|
|
fix1 = fix2 = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
erg += tmp;
|
|
fix1 = false;
|
|
}
|
|
onNumber = false;
|
|
}
|
|
}
|
|
|
|
return erg;
|
|
}
|
|
|
|
TQString Cell::decodeFormula( const TQString &_text, int _col, int _row) const
|
|
{
|
|
if ( _col == -1 )
|
|
_col = d->column;
|
|
if ( _row == -1 )
|
|
_row = d->row;
|
|
|
|
TQString erg = "";
|
|
unsigned int pos = 0;
|
|
const unsigned int length = _text.length();
|
|
|
|
if ( _text.isEmpty() )
|
|
return TQString();
|
|
|
|
while ( pos < length )
|
|
{
|
|
if ( _text[pos] == '"' )
|
|
{
|
|
erg += _text[pos++];
|
|
while ( pos < length && _text[pos] != '"' )
|
|
{
|
|
erg += _text[pos++];
|
|
// Allow escaped double quotes (\")
|
|
if ( pos < length && _text[pos] == '\\' && _text[pos+1] == '"' )
|
|
{
|
|
erg += _text[pos++];
|
|
erg += _text[pos++];
|
|
}
|
|
}
|
|
if ( pos < length )
|
|
erg += _text[pos++];
|
|
}
|
|
else if ( _text[pos] == '#' || _text[pos] == '$' || _text[pos] == TQChar(0xA7))
|
|
{
|
|
bool abs1 = false;
|
|
bool abs2 = false;
|
|
bool era1 = false; // if 1st is relative but encoded absolutely
|
|
bool era2 = false;
|
|
|
|
TQChar _t = _text[pos++];
|
|
if ( _t == '$' )
|
|
abs1 = true;
|
|
else if ( _t == TQChar(0xA7) )
|
|
era1 = true;
|
|
|
|
int col = 0;
|
|
unsigned int oldPos = pos;
|
|
while ( pos < length && ( _text[pos].isDigit() || _text[pos] == '-' ) ) ++pos;
|
|
if ( pos != oldPos )
|
|
col = _text.mid(oldPos, pos-oldPos).toInt();
|
|
if ( !abs1 && !era1 )
|
|
col += _col;
|
|
// Skip '#' or '$'
|
|
|
|
_t = _text[pos++];
|
|
if ( _t == '$' )
|
|
abs2 = true;
|
|
else if ( _t == TQChar(0xA7) )
|
|
era2 = true;
|
|
|
|
int row = 0;
|
|
oldPos = pos;
|
|
while ( pos < length && ( _text[pos].isDigit() || _text[pos] == '-' ) ) ++pos;
|
|
if ( pos != oldPos )
|
|
row = _text.mid(oldPos, pos-oldPos).toInt();
|
|
if ( !abs2 && !era2)
|
|
row += _row;
|
|
// Skip '#' or '$'
|
|
++pos;
|
|
if ( row < 1 || col < 1 || row > KS_rowMax || col > KS_colMax )
|
|
{
|
|
kdDebug(36001) << "Cell::decodeFormula: row or column out of range (col: " << col << " | row: " << row << ")" << endl;
|
|
erg = "=\"#### " + i18n("REFERENCE TO COLUMN OR ROW IS OUT OF RANGE") + "\"";
|
|
return erg;
|
|
}
|
|
if ( abs1 )
|
|
erg += "$";
|
|
erg += Cell::columnName(col); //Get column text
|
|
|
|
if ( abs2 )
|
|
erg += "$";
|
|
erg += TQString::number( row );
|
|
}
|
|
else
|
|
erg += _text[pos++];
|
|
}
|
|
|
|
return erg;
|
|
}
|
|
|
|
|
|
void Cell::freeAllObscuredCells()
|
|
{
|
|
//
|
|
// Free all obscured cells.
|
|
//
|
|
|
|
if (!d->hasExtra())
|
|
return;
|
|
|
|
for ( int x = d->column + d->extra()->mergedXCells;
|
|
x <= d->column + d->extra()->extraXCells; ++x ) {
|
|
for ( int y = d->row + d->extra()->mergedYCells;
|
|
y <= d->row + d->extra()->extraYCells; ++y ) {
|
|
if ( x != d->column || y != d->row ) {
|
|
Cell *cell = format()->sheet()->cellAt( x, y );
|
|
cell->unobscure(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
d->extra()->extraXCells = d->extra()->mergedXCells;
|
|
d->extra()->extraYCells = d->extra()->mergedYCells;
|
|
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------
|
|
// Layout
|
|
|
|
|
|
// Recalculate the entire layout. This includes the following members:
|
|
//
|
|
// d->textX, d->textY
|
|
// d->textWidth, d->textHeight
|
|
// d->fmAscent
|
|
// d->extra()->extraXCells, d->extra()->extraYCells
|
|
// d->extra()->extraWidth, d->extra()->extraHeight
|
|
// d->extra()->nbLines (if multirow)
|
|
//
|
|
// and, of course,
|
|
//
|
|
// d->strOutText
|
|
//
|
|
|
|
void Cell::makeLayout( TQPainter &_painter, int _col, int _row )
|
|
{
|
|
// Are _col and _row really needed ?
|
|
//
|
|
// Yes they are: they are useful if this is the default layout, in
|
|
// which case d->row and d->column are 0 and 0, but _col and _row
|
|
// are the real coordinates of the cell.
|
|
|
|
// There is no need to remake the layout if it hasn't changed.
|
|
if ( !testFlag( Flag_LayoutDirty ) )
|
|
return;
|
|
|
|
// Some initializations.
|
|
if (d->hasExtra())
|
|
d->extra()->nbLines = 0;
|
|
clearFlag( Flag_CellTooShortX );
|
|
clearFlag( Flag_CellTooShortY );
|
|
|
|
// Initiate the cells that this one is obscuring to the ones that
|
|
// are actually merged.
|
|
freeAllObscuredCells();
|
|
if (d->hasExtra())
|
|
mergeCells( d->column, d->row,
|
|
d->extra()->mergedXCells, d->extra()->mergedYCells );
|
|
|
|
// If the column for this cell is hidden or the row is too low,
|
|
// there is no use in remaking the layout.
|
|
ColumnFormat *cl1 = format()->sheet()->columnFormat( _col );
|
|
RowFormat *rl1 = format()->sheet()->rowFormat( _row );
|
|
if ( cl1->isHide()
|
|
|| ( rl1->dblHeight() <= format()->sheet()->doc()->unzoomItY( 2 ) ) ) {
|
|
clearFlag( Flag_LayoutDirty );
|
|
return;
|
|
}
|
|
|
|
// Recalculate the output text, d->strOutText.
|
|
setOutputText();
|
|
|
|
// Empty text? Reset the outstring and, if this is the default
|
|
// cell, return.
|
|
if ( d->strOutText.isEmpty() ) {
|
|
d->strOutText = TQString();
|
|
|
|
if ( isDefault() ) {
|
|
clearFlag( Flag_LayoutDirty );
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Up to here, we have just cared about the contents, not the
|
|
// painting of it. Now it is time to see if the contents fits into
|
|
// the cell and, if not, maybe rearrange the outtext a bit.
|
|
//
|
|
// First, Determine the correct font with zoom taken into account,
|
|
// and apply it to _painter. Then calculate text dimensions, i.e.
|
|
// d->textWidth and d->textHeight.
|
|
applyZoomedFont( _painter, _col, _row );
|
|
textSize( _painter );
|
|
|
|
//
|
|
// Calculate the size of the cell
|
|
//
|
|
RowFormat *rl = format()->sheet()->rowFormat( d->row );
|
|
ColumnFormat *cl = format()->sheet()->columnFormat( d->column );
|
|
|
|
double width = cl->dblWidth();
|
|
double height = rl->dblHeight();
|
|
|
|
// Calculate extraWidth and extraHeight if we have a merged cell.
|
|
if ( testFlag( Flag_Merged ) ) {
|
|
int extraXCells = d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
int extraYCells = d->hasExtra() ? d->extra()->extraYCells : 0;
|
|
|
|
// FIXME: Introduce double extraWidth/Height here and use them
|
|
// instead (see FIXME about this in paintCell()).
|
|
|
|
for ( int x = _col + 1; x <= _col + extraXCells; x++ )
|
|
width += format()->sheet()->columnFormat( x )->dblWidth();
|
|
|
|
for ( int y = _row + 1; y <= _row + extraYCells; y++ )
|
|
height += format()->sheet()->rowFormat( y )->dblHeight();
|
|
}
|
|
|
|
// Cache the newly calculated extraWidth and extraHeight if we have
|
|
// already allocated a struct for it. Otherwise it will be zero, so
|
|
// don't bother.
|
|
if (d->hasExtra()) {
|
|
d->extra()->extraWidth = width;
|
|
d->extra()->extraHeight = height;
|
|
}
|
|
|
|
TQFontMetrics fm = _painter.fontMetrics();
|
|
d->fmAscent = fm.ascent();
|
|
|
|
// Check if we need to break the line into multiple lines and are
|
|
// allowed to do so. If so, set `lines' to the number of lines that
|
|
// are needed to fit into the total width of the combined cell.
|
|
//
|
|
// Also recalculate d->textHeight, d->textWidth, d->extra->nbLines
|
|
// and d->strOutText.
|
|
//
|
|
int lines = 1;
|
|
if ( d->textWidth > (width - 2 * BORDER_SPACE
|
|
- format()->leftBorderWidth( _col, _row )
|
|
- format()->rightBorderWidth( _col, _row ) )
|
|
&& format()->multiRow( _col, _row ) )
|
|
{
|
|
// Copy of d->strOutText but without the newlines.
|
|
// TQString o = d->strOutText.replace( TQChar('\n'), " " );
|
|
|
|
// don't remove the existing LF, these are intended line wraps (whishlist #9881)
|
|
TQString o = d->strOutText;
|
|
|
|
// Break the line at appropriate places, i.e. spaces, if
|
|
// necessary. This means to change the spaces where breaks occur
|
|
// into newlines.
|
|
if ( o.find(' ') != -1 )
|
|
{
|
|
d->strOutText = "";
|
|
|
|
// Make sure that we have a space at the end.
|
|
o += ' ';
|
|
|
|
int start = 0; // Start of the line we are handling now
|
|
int breakpos = 0; // The next candidate pos to break the string
|
|
int pos1 = 0;
|
|
int availableWidth = (int) ( width - 2 * BORDER_SPACE
|
|
- format()->leftBorderWidth( _col, _row )
|
|
- format()->rightBorderWidth( _col, _row ) );
|
|
|
|
do {
|
|
|
|
breakpos = o.find( ' ', breakpos );
|
|
int linefeed = o.find( '\n', pos1 );
|
|
|
|
// kdDebug() << "start: " << start << "; breakpos: " << breakpos << "; pos1: " << pos1 << "; linefeed: " << linefeed << endl;
|
|
|
|
//don't miss LF as a position to calculate current lineWidth
|
|
int work_breakpos = breakpos;
|
|
if (pos1 < linefeed && linefeed < breakpos)
|
|
work_breakpos = linefeed;
|
|
|
|
double lineWidth = format()->sheet()->doc()
|
|
->unzoomItX( fm.width( d->strOutText.mid( start, (pos1 - start) )
|
|
+ o.mid( pos1, work_breakpos - pos1 ) ) );
|
|
|
|
//linefeed could be -1 when no linefeed is found!
|
|
if (breakpos > linefeed && linefeed > 0)
|
|
{
|
|
// kdDebug() << "applying linefeed to start;" << endl;
|
|
start = linefeed;
|
|
lines++;
|
|
}
|
|
|
|
if ( lineWidth <= availableWidth ) {
|
|
// We have room for the rest of the line. End it here.
|
|
d->strOutText += o.mid( pos1, breakpos - pos1 );
|
|
pos1 = breakpos;
|
|
}
|
|
else {
|
|
// Still not enough room. Try to split further.
|
|
if ( o.at( pos1 ) == ' ' )
|
|
pos1++;
|
|
|
|
if ( pos1 != 0 && breakpos != -1 ) {
|
|
d->strOutText += "\n" + o.mid( pos1, breakpos - pos1 );
|
|
lines++;
|
|
}
|
|
else
|
|
d->strOutText += o.mid( pos1, breakpos - pos1 );
|
|
|
|
start = pos1;
|
|
pos1 = breakpos;
|
|
}
|
|
|
|
breakpos++;
|
|
} while( o.find( ' ', breakpos ) != -1 );
|
|
}
|
|
else
|
|
{
|
|
lines = o.contains('\n');
|
|
}
|
|
|
|
d->textHeight *= lines;
|
|
if (lines > 1)
|
|
d->extra()->nbLines = lines;
|
|
|
|
d->textX = 0.0;
|
|
|
|
// Calculate the maximum width, taking into account linebreaks,
|
|
// and put it in d->textWidth.
|
|
TQString t;
|
|
int i;
|
|
int pos = 0;
|
|
d->textWidth = 0.0;
|
|
do {
|
|
i = d->strOutText.find( "\n", pos );
|
|
|
|
if ( i == -1 )
|
|
t = d->strOutText.mid( pos, d->strOutText.length() - pos );
|
|
else {
|
|
t = d->strOutText.mid( pos, i - pos );
|
|
pos = i + 1;
|
|
}
|
|
|
|
double tw = format()->sheet()->doc()->unzoomItX( fm.width( t ) );
|
|
if ( tw > d->textWidth )
|
|
d->textWidth = tw;
|
|
} while ( i != -1 );
|
|
}
|
|
|
|
// Calculate d->textX and d->textY
|
|
offsetAlign( _col, _row );
|
|
|
|
int a = effAlignX();
|
|
|
|
// Get indentation. This is only used for left aligned text.
|
|
double indent = 0.0;
|
|
if ( a == Format::Left && !isEmpty() )
|
|
indent = format()->getIndent( _col, _row );
|
|
|
|
// Set Flag_CellTooShortX if the text is vertical or angled, and too
|
|
// high for the cell.
|
|
if ( format()->verticalText( _col, _row ) || format()->getAngle( _col, _row ) != 0 ) {
|
|
//RowFormat *rl = format()->sheet()->rowFormat( _row );
|
|
|
|
if ( d->textHeight >= height/*rl->dblHeight()*/ )
|
|
setFlag( Flag_CellTooShortX );
|
|
}
|
|
|
|
// Do we have to occupy additional cells to the right? This is only
|
|
// done for cells that have no merged cells in the Y direction.
|
|
//
|
|
// FIXME: Check if all cells along the merged edge to the right are
|
|
// empty and use the extra space? No, probably not.
|
|
//
|
|
if ( d->textWidth + indent > ( width - 2 * BORDER_SPACE
|
|
- format()->leftBorderWidth( _col, _row )
|
|
- format()->rightBorderWidth( _col, _row ) )
|
|
&& ( !d->hasExtra() || d->extra()->mergedYCells == 0 ) )
|
|
{
|
|
int c = d->column;
|
|
|
|
// Find free cells to the right of this one.
|
|
int end = 0;
|
|
while ( !end ) {
|
|
ColumnFormat *cl2 = format()->sheet()->columnFormat( c + 1 );
|
|
Cell *cell = format()->sheet()->visibleCellAt( c + 1, d->row );
|
|
|
|
if ( cell->isEmpty() ) {
|
|
width += cl2->dblWidth() - 1;
|
|
c++;
|
|
|
|
// Enough space?
|
|
if ( d->textWidth + indent <= ( width - 2 * BORDER_SPACE
|
|
- format()->leftBorderWidth( _col, _row )
|
|
- format()->rightBorderWidth( _col, _row ) ) )
|
|
end = 1;
|
|
}
|
|
else
|
|
// Not enough space, but the next cell is not empty
|
|
end = -1;
|
|
}
|
|
|
|
// Try to use additional space from the neighboring cells that
|
|
// were calculated in the last step. This is the place that we
|
|
// set d->extra()->extraXCells and d->extra()->extraWidth.
|
|
//
|
|
// Currently this is only done for left aligned cells. We have to
|
|
// check to make sure we haven't already force-merged enough cells
|
|
//
|
|
// FIXME: Why not right/center aligned text?
|
|
//
|
|
// FIXME: Shouldn't we check to see if end == -1 here before
|
|
// setting Flag_CellTooShortX?
|
|
//
|
|
if ( format()->align( _col, _row ) == Format::Left
|
|
|| ( format()->align( _col, _row ) == Format::Undefined
|
|
&& !value().isNumber() ) )
|
|
{
|
|
if ( c - d->column > d->extra()->mergedXCells ) {
|
|
d->extra()->extraXCells = c - d->column;
|
|
d->extra()->extraWidth = width;
|
|
for ( int i = d->column + 1; i <= c; ++i ) {
|
|
Cell *cell = format()->sheet()->nonDefaultCell( i, d->row );
|
|
cell->obscure( this );
|
|
}
|
|
|
|
// Not enough space
|
|
if ( end == -1 )
|
|
setFlag( Flag_CellTooShortX );
|
|
}
|
|
else
|
|
setFlag( Flag_CellTooShortX );
|
|
}
|
|
else
|
|
setFlag( Flag_CellTooShortX );
|
|
}
|
|
|
|
// Do we have to occupy additional cells at the bottom ?
|
|
//
|
|
// FIXME: Setting to make the current cell grow.
|
|
//
|
|
if ( format()->multiRow( _col, _row )
|
|
&& d->textHeight > ( height - 2 * BORDER_SPACE
|
|
- format()->topBorderWidth( _col, _row )
|
|
- format()->bottomBorderWidth( _col, _row ) ) )
|
|
{
|
|
int r = d->row;
|
|
int end = 0;
|
|
|
|
// Find free cells bottom to this one
|
|
while ( !end ) {
|
|
RowFormat *rl2 = format()->sheet()->rowFormat( r + 1 );
|
|
Cell *cell = format()->sheet()->visibleCellAt( d->column, r + 1 );
|
|
|
|
if ( cell->isEmpty() ) {
|
|
height += rl2->dblHeight() - 1.0;
|
|
r++;
|
|
|
|
// Enough space ?
|
|
if ( d->textHeight <= ( height - 2 * BORDER_SPACE
|
|
- format()->topBorderWidth( _col, _row )
|
|
- format()->bottomBorderWidth( _col, _row ) ) )
|
|
end = 1;
|
|
}
|
|
else
|
|
// Not enough space, but the next cell is not empty.
|
|
end = -1;
|
|
}
|
|
|
|
// Check to make sure we haven't already force-merged enough cells.
|
|
if ( r - d->row > d->extra()->mergedYCells )
|
|
{
|
|
d->extra()->extraYCells = r - d->row;
|
|
d->extra()->extraHeight = height;
|
|
|
|
for ( int i = d->row + 1; i <= r; ++i )
|
|
{
|
|
Cell *cell = format()->sheet()->nonDefaultCell( d->column, i );
|
|
cell->obscure( this );
|
|
}
|
|
|
|
// Not enough space?
|
|
if ( end == -1 )
|
|
setFlag( Flag_CellTooShortY );
|
|
}
|
|
else
|
|
setFlag( Flag_CellTooShortY );
|
|
}
|
|
|
|
clearFlag( Flag_LayoutDirty );
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
void Cell::valueChanged ()
|
|
{
|
|
update();
|
|
|
|
format()->sheet()->valueChanged (this);
|
|
}
|
|
|
|
|
|
// Recalculate d->strOutText.
|
|
//
|
|
|
|
void Cell::setOutputText()
|
|
{
|
|
if ( isDefault() ) {
|
|
d->strOutText = TQString();
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions )
|
|
d->extra()->conditions->checkMatches();
|
|
|
|
return;
|
|
}
|
|
|
|
// If nothing has changed, we don't need to remake the text layout.
|
|
if ( !testFlag(Flag_TextFormatDirty) )
|
|
return;
|
|
|
|
// We don't want to remake the layout unnecessarily.
|
|
clearFlag( Flag_TextFormatDirty );
|
|
|
|
// Display a formula if warranted. If not, display the value instead;
|
|
// this is the most common case.
|
|
if ( (!hasError()) && isFormula() && format()->sheet()->getShowFormula()
|
|
&& !( format()->sheet()->isProtected() && format()->isHideFormula( d->column, d->row ) )
|
|
|| isEmpty() )
|
|
d->strOutText = d->strText;
|
|
else {
|
|
d->strOutText = sheet()->doc()->formatter()->formatText (this,
|
|
formatType());
|
|
}
|
|
|
|
// Check conditions if needed.
|
|
if ( d->hasExtra() && d->extra()->conditions )
|
|
d->extra()->conditions->checkMatches();
|
|
}
|
|
|
|
|
|
// Recalculate d->textX and d->textY.
|
|
//
|
|
// Used in makeLayout() and calculateTextParameters().
|
|
//
|
|
|
|
void Cell::offsetAlign( int _col, int _row )
|
|
{
|
|
int a;
|
|
Format::AlignY ay;
|
|
int tmpAngle;
|
|
bool tmpVerticalText;
|
|
bool tmpMultiRow;
|
|
|
|
if ( d->hasExtra()
|
|
&& d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
{
|
|
Style *style = d->extra()->conditions->matchedStyle();
|
|
|
|
if ( style->hasFeature( Style::SAlignX, true ) )
|
|
a = style->alignX();
|
|
else
|
|
a = format()->align( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SVerticalText, true ) )
|
|
tmpVerticalText = style->hasProperty( Style::PVerticalText );
|
|
else
|
|
tmpVerticalText = format()->verticalText( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SMultiRow, true ) )
|
|
tmpMultiRow = style->hasProperty( Style::PMultiRow );
|
|
else
|
|
tmpMultiRow = format()->multiRow( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SAlignY, true ) )
|
|
ay = style->alignY();
|
|
else
|
|
ay = format()->alignY( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SAngle, true ) )
|
|
tmpAngle = style->rotateAngle();
|
|
else
|
|
tmpAngle = format()->getAngle( _col, _row );
|
|
}
|
|
else {
|
|
a = format()->align( _col, _row );
|
|
ay = format()->alignY( _col, _row );
|
|
tmpAngle = format()->getAngle( _col, _row );
|
|
tmpVerticalText = format()->verticalText( _col, _row );
|
|
tmpMultiRow = format()->multiRow( _col, _row );
|
|
}
|
|
|
|
RowFormat *rl = format()->sheet()->rowFormat( _row );
|
|
ColumnFormat *cl = format()->sheet()->columnFormat( _col );
|
|
|
|
double w = cl->dblWidth();
|
|
double h = rl->dblHeight();
|
|
|
|
if ( d->hasExtra() ) {
|
|
if ( d->extra()->extraXCells ) w = d->extra()->extraWidth;
|
|
if ( d->extra()->extraYCells ) h = d->extra()->extraHeight;
|
|
}
|
|
|
|
const double effTop = BORDER_SPACE + 0.5 * effTopBorderPen( _col, _row ).width();
|
|
const double effBottom = h - BORDER_SPACE - 0.5 * effBottomBorderPen( _col, _row ).width();
|
|
|
|
// Calculate d->textY based on the vertical alignment and a few
|
|
// other inputs.
|
|
switch( ay )
|
|
{
|
|
case Format::Top:
|
|
{
|
|
if ( tmpAngle == 0 )
|
|
{
|
|
d->textY = effTop + (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
else if ( tmpAngle < 0 )
|
|
{
|
|
d->textY = effTop;
|
|
}
|
|
else
|
|
{
|
|
d->textY = effTop
|
|
+ (double) d->fmAscent * cos( tmpAngle * M_PI / 180 ) / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
break;
|
|
}
|
|
case Format::Bottom:
|
|
{
|
|
if ( !tmpVerticalText && !tmpMultiRow && !tmpAngle )
|
|
{
|
|
d->textY = effBottom;
|
|
}
|
|
else if ( tmpAngle != 0 )
|
|
{
|
|
// Is enough place available?
|
|
if ( effBottom - effTop - d->textHeight > 0 )
|
|
{
|
|
if ( tmpAngle < 0 )
|
|
{
|
|
d->textY = effBottom - d->textHeight;
|
|
}
|
|
else
|
|
{
|
|
d->textY = effBottom - d->textHeight
|
|
+ ( (double) d->fmAscent * cos( tmpAngle * M_PI / 180 ) / format()->sheet()->doc()->zoomedResolutionY() );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( tmpAngle < 0 )
|
|
{
|
|
d->textY = effTop;
|
|
}
|
|
else
|
|
{
|
|
d->textY = effTop
|
|
+ ( (double) d->fmAscent * cos( tmpAngle * M_PI / 180 )
|
|
/ format()->sheet()->doc()->zoomedResolutionY() );
|
|
}
|
|
}
|
|
}
|
|
else if ( tmpMultiRow && !tmpVerticalText )
|
|
{
|
|
// Is enough place available?
|
|
if ( effBottom - effTop - d->textHeight > 0 )
|
|
{
|
|
d->textY = effBottom - d->textHeight
|
|
+ (double)d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
else
|
|
{
|
|
d->textY = effTop
|
|
+ (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Is enough place available?
|
|
if ( effBottom - effTop - d->textHeight > 0 )
|
|
{
|
|
d->textY = effBottom - d->textHeight
|
|
+ (double)d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
else
|
|
{
|
|
d->textY = effTop
|
|
+ (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
case Format::Middle:
|
|
case Format::UndefinedY:
|
|
{
|
|
if ( !tmpVerticalText && !tmpMultiRow && !tmpAngle )
|
|
{
|
|
d->textY = ( h - d->textHeight ) / 2
|
|
+ (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
else if ( tmpAngle != 0 )
|
|
{
|
|
// Is enough place available?
|
|
if ( effBottom - effTop - d->textHeight > 0 )
|
|
{
|
|
if ( tmpAngle < 0 )
|
|
{
|
|
d->textY = ( h - d->textHeight ) / 2;
|
|
}
|
|
else
|
|
{
|
|
d->textY = ( h - d->textHeight ) / 2
|
|
+ (double) d->fmAscent * cos( tmpAngle * M_PI / 180 ) / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( tmpAngle < 0 )
|
|
{
|
|
d->textY = effTop;
|
|
}
|
|
else
|
|
{
|
|
d->textY = effTop
|
|
+ ( (double)d->fmAscent * cos( tmpAngle * M_PI / 180 ) / format()->sheet()->doc()->zoomedResolutionY() );
|
|
}
|
|
}
|
|
}
|
|
else if ( tmpMultiRow && !tmpVerticalText )
|
|
{
|
|
// Is enough place available?
|
|
if ( effBottom - effTop - d->textHeight > 0 )
|
|
{
|
|
d->textY = ( h - d->textHeight ) / 2
|
|
+ (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
else
|
|
{
|
|
d->textY = effTop
|
|
+ (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Is enough place available?
|
|
if ( effBottom - effTop - d->textHeight > 0 )
|
|
{
|
|
d->textY = ( h - d->textHeight ) / 2
|
|
+ (double) d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
else
|
|
d->textY = effTop
|
|
+ (double)d->fmAscent / format()->sheet()->doc()->zoomedResolutionY();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
a = effAlignX();
|
|
if ( format()->sheet()->getShowFormula() &&
|
|
!( format()->sheet()->isProtected() && format()->isHideFormula( _col, _row ) ) )
|
|
{
|
|
a = Format::Left;
|
|
}
|
|
|
|
// Calculate d->textX based on alignment and textwidth.
|
|
switch ( a ) {
|
|
case Format::Left:
|
|
d->textX = 0.5 * effLeftBorderPen( _col, _row ).width() + BORDER_SPACE;
|
|
break;
|
|
case Format::Right:
|
|
d->textX = w - BORDER_SPACE - d->textWidth
|
|
- 0.5 * effRightBorderPen( _col, _row ).width();
|
|
break;
|
|
case Format::Center:
|
|
d->textX = 0.5 * ( w - BORDER_SPACE - d->textWidth -
|
|
0.5 * effRightBorderPen( _col, _row ).width() );
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// Recalculate the current text dimensions, i.e. d->textWidth and
|
|
// d->textHeight.
|
|
//
|
|
// Used in makeLayout() and calculateTextParameters().
|
|
//
|
|
void Cell::textSize( TQPainter &_paint )
|
|
{
|
|
TQFontMetrics fm = _paint.fontMetrics();
|
|
// Horizontal text ?
|
|
|
|
int tmpAngle;
|
|
int _row = row();
|
|
int _col = column();
|
|
bool tmpVerticalText;
|
|
bool fontUnderlined;
|
|
Format::AlignY ay;
|
|
|
|
// Set tmpAngle, tmpeVerticalText, ay and fontUnderlined according
|
|
// to if there is a matching condition or not.
|
|
if ( d->hasExtra()
|
|
&& d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
{
|
|
Style *style = d->extra()->conditions->matchedStyle();
|
|
|
|
if ( style->hasFeature( Style::SAngle, true ) )
|
|
tmpAngle = style->rotateAngle();
|
|
else
|
|
tmpAngle = format()->getAngle( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SVerticalText, true ) )
|
|
tmpVerticalText = style->hasProperty( Style::PVerticalText );
|
|
else
|
|
tmpVerticalText = format()->verticalText( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SAlignY, true ) )
|
|
ay = style->alignY();
|
|
else
|
|
ay = format()->alignY( _col, _row );
|
|
|
|
if ( style->hasFeature( Style::SFontFlag, true ) )
|
|
fontUnderlined = ( style->fontFlags() & (uint) Style::FUnderline );
|
|
else
|
|
fontUnderlined = format()->textFontUnderline( _col, _row );
|
|
}
|
|
else {
|
|
// The cell has no condition with a maxed style.
|
|
tmpAngle = format()->getAngle( _col, _row );
|
|
tmpVerticalText = format()->verticalText( _col, _row );
|
|
ay = format()->alignY( _col, _row );
|
|
fontUnderlined = format()->textFontUnderline( _col, _row );
|
|
}
|
|
|
|
// Set d->textWidth and d->textHeight to correct values according to
|
|
// if the text is horizontal, vertical or rotated.
|
|
if ( !tmpVerticalText && !tmpAngle ) {
|
|
// Horizontal text.
|
|
|
|
d->textWidth = format()->sheet()->doc()->unzoomItX( fm.width( d->strOutText ) );
|
|
int offsetFont = 0;
|
|
if ( ( ay == Format::Bottom ) && fontUnderlined ) {
|
|
offsetFont = fm.underlinePos() + 1;
|
|
}
|
|
|
|
d->textHeight = format()->sheet()->doc()->unzoomItY( fm.ascent() + fm.descent()
|
|
+ offsetFont );
|
|
}
|
|
else if ( tmpAngle!= 0 ) {
|
|
// Rotated text.
|
|
|
|
d->textHeight = format()->sheet()->doc()
|
|
->unzoomItY( int( cos( tmpAngle * M_PI / 180 )
|
|
* ( fm.ascent() + fm.descent() )
|
|
+ abs( int( ( fm.width( d->strOutText )
|
|
* sin( tmpAngle * M_PI / 180 ) ) ) ) ) );
|
|
|
|
d->textWidth = format()->sheet()->doc()
|
|
->unzoomItX( int( abs( int( ( sin( tmpAngle * M_PI / 180 )
|
|
* ( fm.ascent() + fm.descent() ) ) ) )
|
|
+ fm.width( d->strOutText )
|
|
* cos ( tmpAngle * M_PI / 180 ) ) );
|
|
}
|
|
else {
|
|
// Vertical text.
|
|
int width = 0;
|
|
for ( unsigned int i = 0; i < d->strOutText.length(); i++ )
|
|
width = TQMAX( width, fm.width( d->strOutText.at( i ) ) );
|
|
|
|
d->textWidth = format()->sheet()->doc()->unzoomItX( width );
|
|
d->textHeight = format()->sheet()->doc()->unzoomItY( ( fm.ascent() + fm.descent() )
|
|
* d->strOutText.length() );
|
|
}
|
|
}
|
|
|
|
|
|
// Get the effective font to use after the zooming and apply it to `painter'.
|
|
//
|
|
// Used in makeLayout() and calculateTextParameters().
|
|
//
|
|
|
|
void Cell::applyZoomedFont( TQPainter &painter, int _col, int _row )
|
|
{
|
|
TQFont tmpFont( format()->textFont( _col, _row ) );
|
|
|
|
// If there is a matching condition on this cell then set the
|
|
// according style parameters.
|
|
if ( d->hasExtra()
|
|
&& d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() ) {
|
|
|
|
Style * s = d->extra()->conditions->matchedStyle();
|
|
|
|
// Other size?
|
|
if ( s->hasFeature( Style::SFontSize, true ) )
|
|
tmpFont.setPointSizeFloat( s->fontSize() );
|
|
|
|
// Other attributes?
|
|
if ( s->hasFeature( Style::SFontFlag, true ) ) {
|
|
uint flags = s->fontFlags();
|
|
|
|
tmpFont.setBold( flags & (uint) Style::FBold );
|
|
tmpFont.setUnderline( flags & (uint) Style::FUnderline );
|
|
tmpFont.setItalic( flags & (uint) Style::FItalic );
|
|
tmpFont.setStrikeOut( flags & (uint) Style::FStrike );
|
|
}
|
|
|
|
// Other family?
|
|
if ( s->hasFeature( Style::SFontFamily, true ) )
|
|
tmpFont.setFamily( s->fontFamily() );
|
|
}
|
|
#if 0
|
|
else
|
|
/*
|
|
* could somebody please explaint why we check for isProtected or isHideFormula here
|
|
*/
|
|
if ( d->extra()->conditions
|
|
&& d->extra()->conditions->currentCondition( condition )
|
|
&& !(format()->sheet()->getShowFormula()
|
|
&& !( format()->sheet()->isProtected()
|
|
&& format()->isHideFormula( d->column, d->row ) ) ) )
|
|
{
|
|
if ( condition.fontcond )
|
|
tmpFont = *(condition.fontcond);
|
|
else
|
|
tmpFont = condition.style->font();
|
|
}
|
|
#endif
|
|
|
|
// Scale the font size according to the current zoom.
|
|
tmpFont.setPointSizeFloat( 0.01 * format()->sheet()->doc()->zoom()
|
|
* tmpFont.pointSizeFloat() );
|
|
|
|
painter.setFont( tmpFont );
|
|
}
|
|
|
|
|
|
//used in Sheet::adjustColumnHelper and Sheet::adjustRow
|
|
void Cell::calculateTextParameters( TQPainter &_painter,
|
|
int _col, int _row )
|
|
{
|
|
// Apply the correct font to _painter.
|
|
applyZoomedFont( _painter, _col, _row );
|
|
|
|
// Recalculate d->textWidth and d->textHeight
|
|
textSize( _painter );
|
|
|
|
// Recalculate d->textX and d->textY.
|
|
offsetAlign( _col, _row );
|
|
}
|
|
|
|
|
|
// ----------------------------------------------------------------
|
|
// Formula handling
|
|
|
|
|
|
bool Cell::makeFormula()
|
|
{
|
|
clearFormula ();
|
|
|
|
d->formula = new KSpread::Formula (sheet(), this);
|
|
d->formula->setExpression (d->strText);
|
|
|
|
if (!d->formula->isValid ()) {
|
|
// Did a syntax error occur ?
|
|
clearFormula();
|
|
|
|
if (format()->sheet()->doc()->getShowMessageError())
|
|
{
|
|
TQString tmp(i18n("Error in cell %1\n\n"));
|
|
tmp = tmp.arg( fullName() );
|
|
KMessageBox::error( (TQWidget*)0L, tmp);
|
|
}
|
|
setFlag(Flag_ParseError);
|
|
Value v;
|
|
v.setError ( "####" );
|
|
setValue (v);
|
|
return false;
|
|
}
|
|
|
|
// we must recalc
|
|
setCalcDirtyFlag ();
|
|
|
|
return true;
|
|
}
|
|
|
|
void Cell::clearFormula()
|
|
{
|
|
delete d->formula;
|
|
d->formula = 0L;
|
|
}
|
|
|
|
bool Cell::calc(bool delay)
|
|
{
|
|
if ( !isFormula() )
|
|
return true;
|
|
|
|
if (d->formula == 0)
|
|
{
|
|
if ( testFlag( Flag_ParseError ) ) // there was a parse error
|
|
return false;
|
|
else
|
|
{
|
|
/* we were probably at a "isLoading() = true" state when we originally
|
|
* parsed
|
|
*/
|
|
makeFormula ();
|
|
|
|
if ( d->formula == 0 ) // there was a parse error
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if ( !testFlag( Flag_CalcDirty ) )
|
|
return true;
|
|
|
|
if ( delay )
|
|
{
|
|
if ( format()->sheet()->doc()->delayCalculation() )
|
|
return true;
|
|
}
|
|
|
|
setFlag(Flag_LayoutDirty);
|
|
setFlag(Flag_TextFormatDirty);
|
|
clearFlag(Flag_CalcDirty);
|
|
|
|
Value result = d->formula->eval ();
|
|
setValue (result);
|
|
if (result.isNumber())
|
|
checkNumberFormat(); // auto-chooses number or scientific
|
|
|
|
clearFlag(Flag_CalcDirty);
|
|
setFlag(Flag_LayoutDirty);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
// ================================================================
|
|
// Painting
|
|
|
|
|
|
// Paint the cell. This is the main function that calls a lot of
|
|
// helper functions.
|
|
//
|
|
// `rect' is the rectangle that we should paint on. If the cell
|
|
// does not overlap this rectangle, we can return immediately.
|
|
// `coordinate' is the origin (the upper left) of the cell in document
|
|
// coordinates.
|
|
//
|
|
|
|
void Cell::paintCell( const KoRect &rect, TQPainter & painter,
|
|
View *view,
|
|
const KoPoint &coordinate,
|
|
const TQPoint &cellRef,
|
|
int paintBorder,
|
|
TQPen & rightPen, TQPen & bottomPen,
|
|
TQPen & leftPen, TQPen & topPen,
|
|
TQValueList<TQPoint> &mergedCellsPainted,
|
|
bool drawCursor )
|
|
{
|
|
bool paintBorderRight = paintBorder & Border_Right;
|
|
bool paintBorderBottom = paintBorder & Border_Bottom;
|
|
bool paintBorderLeft = paintBorder & Border_Left;
|
|
bool paintBorderTop = paintBorder & Border_Top;
|
|
|
|
// If we are already painting this cell, then return immediately.
|
|
// This avoids infinite recursion.
|
|
if ( testFlag( Flag_PaintingCell ) )
|
|
return;
|
|
|
|
// Indicate that we are painting this cell now.
|
|
setFlag( Flag_PaintingCell );
|
|
|
|
// This flag indicates that we are working on drawing the cells that
|
|
// another cell is obscuring. The value is the number of levels down we
|
|
// are currently working -- i.e. a cell obscured by a cell which is
|
|
// obscured by a cell.
|
|
static int paintingObscured = 0;
|
|
|
|
#if 0
|
|
if (paintingObscured == 0)
|
|
kdDebug(36001) << "painting cell " << name() << endl;
|
|
else
|
|
kdDebug(36001) << " painting obscured cell " << name() << endl;
|
|
#endif
|
|
|
|
// Sanity check: If we're working on drawing an obscured cell, that
|
|
// means this cell should have a cell that obscures it.
|
|
Q_ASSERT(!(paintingObscured > 0 && d->extra()->obscuringCells.isEmpty()));
|
|
|
|
// The parameter cellref should be *this, unless this is the default cell.
|
|
Q_ASSERT(isDefault()
|
|
|| (((cellRef.x() == d->column) && (cellRef.y() == d->row))));
|
|
|
|
Sheet::LayoutDirection sheetDir = format()->sheet()->layoutDirection();
|
|
|
|
double left = coordinate.x();
|
|
|
|
ColumnFormat * colFormat = format()->sheet()->columnFormat( cellRef.x() );
|
|
RowFormat * rowFormat = format()->sheet()->rowFormat( cellRef.y() );
|
|
|
|
// Set width, height to the total width and height that this cell
|
|
// covers, including obscured cells, and width0, height0 to the
|
|
// width and height of this cell, maybe merged but never implicitly
|
|
// extended.
|
|
double width0 = colFormat->dblWidth();
|
|
double height0 = rowFormat->dblHeight();
|
|
double width = width0;
|
|
double height = height0;
|
|
|
|
// Handle right-to-left layout.
|
|
// In an RTL sheet the cells have to be painted at their opposite horizontal
|
|
// location on the canvas, meaning that column A will be the rightmost column
|
|
// on screen, column B will be to the left of it and so on. Here we change
|
|
// the horizontal coordinate at which we start painting the cell in case the
|
|
// sheet's direction is RTL. We do this only if paintingObscured is 0,
|
|
// otherwise the cell's painting location will flip back and forth in
|
|
// consecutive calls to paintCell when painting obscured cells.
|
|
if ( sheetDir == Sheet::RightToLeft && paintingObscured == 0
|
|
&& view && view->canvasWidget() )
|
|
{
|
|
double dwidth = view->doc()->unzoomItX(view->canvasWidget()->width());
|
|
left = dwidth - coordinate.x() - width;
|
|
}
|
|
|
|
// See if this cell is merged or has overflown into neighbor cells.
|
|
// In that case, the width/height is greater than just the cell
|
|
// itself.
|
|
if (d->hasExtra()) {
|
|
if (d->extra()->mergedXCells > 0 || d->extra()->mergedYCells > 0) {
|
|
// merged cell extends to the left if sheet is RTL
|
|
if ( sheetDir == Sheet::RightToLeft ) {
|
|
left -= d->extra()->extraWidth - width;
|
|
}
|
|
width0 = d->extra()->extraWidth;
|
|
height0 = d->extra()->extraHeight;
|
|
width = d->extra()->extraWidth;
|
|
height = d->extra()->extraHeight;
|
|
}
|
|
else {
|
|
#if 0
|
|
width += d->extra()->extraXCells ? d->extra()->extraWidth : 0;
|
|
height += d->extra()->extraYCells ? d->extra()->extraHeight : 0;
|
|
#else
|
|
// FIXME: Make extraWidth/Height really contain the *extra* width/height.
|
|
if ( d->extra()->extraXCells )
|
|
width = d->extra()->extraWidth;
|
|
if ( d->extra()->extraYCells )
|
|
height = d->extra()->extraHeight;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// Check if the cell is "selected", i.e. it should be drawn with the
|
|
// color that indicates selection (dark blue). If more than one
|
|
// square is selected, the last one uses the ordinary colors. In
|
|
// that case, "selected" will be set to false even though the cell
|
|
// itself really is selected.
|
|
bool selected = false;
|
|
if ( view != NULL ) {
|
|
selected = view->selectionInfo()->contains( cellRef );
|
|
|
|
// But the cell doesn't look selected if this is the marker cell.
|
|
Cell *cell = format()->sheet()->cellAt( view->selectionInfo()->marker() );
|
|
TQPoint bottomRight( view->selectionInfo()->marker().x() + cell->extraXCells(),
|
|
view->selectionInfo()->marker().y() + cell->extraYCells() );
|
|
TQRect markerArea( view->selectionInfo()->marker(), bottomRight );
|
|
selected = selected && !( markerArea.contains( cellRef ) );
|
|
|
|
// Don't draw any selection at all when printing.
|
|
if ( painter.device()->isExtDev() || !drawCursor )
|
|
selected = false;
|
|
}
|
|
|
|
// Need to make a new layout ?
|
|
//
|
|
// FIXME: We have already used (at least) extraWidth/Height above,
|
|
// and now we are recalculating the layout. This has to be
|
|
// moved up above all uses.
|
|
//
|
|
// FIXME: This needs to be taken out eventually - it is done in
|
|
// canvas::paintUpdates().
|
|
if ( testFlag( Flag_LayoutDirty ) )
|
|
makeLayout( painter, cellRef.x(), cellRef.y() );
|
|
|
|
// ---------------- Start the actual painting. ----------------
|
|
|
|
// If the rect of this cell doesn't intersect the rect that should
|
|
// be painted, we can skip the rest and return. (Note that we need
|
|
// to calculate `left' first before we can do this.)
|
|
const KoRect cellRect( left, coordinate.y(), width, height );
|
|
const KoRect cellRect0( left, coordinate.y(), width0, height0 );
|
|
if ( !cellRect.intersects( rect ) ) {
|
|
clearFlag( Flag_PaintingCell );
|
|
return;
|
|
}
|
|
|
|
// Get the background color.
|
|
//
|
|
// If there is a condition giving the background color for this cell
|
|
// (and it matches), use that one, otherwise get the standard
|
|
// background.
|
|
TQColor backgroundColor;
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SBackgroundColor, true ) )
|
|
backgroundColor = d->extra()->conditions->matchedStyle()->bgColor();
|
|
else
|
|
backgroundColor = bgColor( cellRef.x(), cellRef.y() );
|
|
|
|
// 1. Paint the background.
|
|
if ( !isPartOfMerged() )
|
|
paintBackground( painter, cellRect0, cellRef, selected, backgroundColor );
|
|
|
|
// 2. Paint the default borders if we are on screen or if we are printing
|
|
// and the checkbox to do this is checked.
|
|
if ( painter.device()->devType() != TQInternal::Printer
|
|
|| format()->sheet()->print()->printGrid())
|
|
paintDefaultBorders( painter, rect, cellRect, cellRef,
|
|
paintBorderRight, paintBorderBottom,
|
|
paintBorderLeft, paintBorderTop,
|
|
rightPen, bottomPen, leftPen, topPen );
|
|
|
|
// 3. Paint all the cells that this one obscures. They may only be
|
|
// partially obscured.
|
|
//
|
|
// The `paintingObscured' variable is used to avoid infinite
|
|
// recursion since cells sometimes paint their obscuring cell as
|
|
// well.
|
|
paintingObscured++;
|
|
|
|
if (d->hasExtra() && (d->extra()->extraXCells > 0
|
|
|| d->extra()->extraYCells > 0)) {
|
|
//kdDebug(36001) << "painting obscured cells for " << name() << endl;
|
|
|
|
paintObscuredCells( rect, painter, view, cellRect, cellRef,
|
|
paintBorderRight, paintBorderBottom,
|
|
paintBorderLeft, paintBorderTop,
|
|
rightPen, bottomPen, leftPen, topPen,
|
|
mergedCellsPainted);
|
|
|
|
// FIXME: Is this the right place for this?
|
|
if ( d->extra()->mergedXCells > 0 || d->extra()->mergedYCells > 0 )
|
|
mergedCellsPainted.prepend( cellRef );
|
|
}
|
|
paintingObscured--;
|
|
|
|
// 4. Paint the borders of the cell if no other cell is forcing this
|
|
// one, i.e. this cell is not part of a merged cell.
|
|
//
|
|
|
|
// If we print pages, then we disable clipping, otherwise borders are
|
|
// cut in the middle at the page borders.
|
|
if ( painter.device()->isExtDev() )
|
|
painter.setClipping( false );
|
|
|
|
// Paint the borders if this cell is not part of another merged cell.
|
|
if ( !isPartOfMerged() ) {
|
|
// if (!testFlag(Flag_Highlight))
|
|
paintCellBorders( painter, rect, cellRect0,
|
|
cellRef,
|
|
paintBorderRight, paintBorderBottom,
|
|
paintBorderLeft, paintBorderTop,
|
|
rightPen, bottomPen, leftPen, topPen );
|
|
}
|
|
|
|
// Turn clipping back on.
|
|
if ( painter.device()->isExtDev() )
|
|
painter.setClipping( true );
|
|
|
|
// 5. Paint diagonal lines and page borders.
|
|
paintCellDiagonalLines( painter, cellRect0, cellRef );
|
|
|
|
paintPageBorders( painter, cellRect0, cellRef,
|
|
paintBorderRight, paintBorderBottom );
|
|
|
|
|
|
// 6. Now paint the content, if this cell isn't obscured.
|
|
if ( !isObscured() ) {
|
|
|
|
// 6a. Paint possible comment indicator.
|
|
if ( !painter.device()->isExtDev()
|
|
|| format()->sheet()->print()->printCommentIndicator() )
|
|
paintCommentIndicator( painter, cellRect, cellRef, backgroundColor );
|
|
|
|
// 6b. Paint possible formula indicator.
|
|
if ( !painter.device()->isExtDev()
|
|
|| format()->sheet()->print()->printFormulaIndicator() )
|
|
paintFormulaIndicator( painter, cellRect, backgroundColor );
|
|
|
|
// 6c. Paint possible indicator for clipped text.
|
|
paintMoreTextIndicator( painter, cellRect, backgroundColor );
|
|
|
|
//6c. Paint cell highlight
|
|
#if 0
|
|
if (highlightBorder != Border_None)
|
|
paintCellHighlight ( painter, cellRect, cellRef, highlightBorder,
|
|
rightHighlightPen, bottomHighlightPen,
|
|
leftHighlightPen, topHighlightPen );
|
|
#endif
|
|
|
|
// 6d. Paint the text in the cell unless:
|
|
// a) it is empty
|
|
// b) something indicates that the text should not be painted
|
|
// c) the sheet is protected and the cell is hidden.
|
|
if ( !d->strOutText.isEmpty()
|
|
&& ( !painter.device()->isExtDev()
|
|
|| !format()->getDontprintText( cellRef.x(), cellRef.y() ) )
|
|
&& !( format()->sheet()->isProtected()
|
|
&& format()->isHideAll( cellRef.x(), cellRef.y() ) ) )
|
|
{
|
|
paintText( painter, cellRect, cellRef );
|
|
}
|
|
}
|
|
|
|
// 7. If this cell is obscured and we are not already painting obscured
|
|
// cells, then paint the obscuring cell(s). Otherwise don't do
|
|
// anything so that we don't cause an infinite loop.
|
|
if ( isObscured() && paintingObscured == 0 &&
|
|
!( sheetDir == Sheet::RightToLeft && painter.device()->isExtDev() ) )
|
|
{
|
|
|
|
//kdDebug(36001) << "painting cells that obscure " << name() << endl;
|
|
|
|
// Store the obscuringCells list in a list of TQPoint(column, row)
|
|
// This avoids crashes during the iteration through
|
|
// obscuringCells, when the cells may get non valid or the list
|
|
// itself gets changed during a call of obscuringCell->paintCell
|
|
// (this happens e.g. when there is an updateDepend)
|
|
if (d->hasExtra()) {
|
|
TQValueList<TQPoint> listPoints;
|
|
TQValueList<Cell*>::iterator it = d->extra()->obscuringCells.begin();
|
|
TQValueList<Cell*>::iterator end = d->extra()->obscuringCells.end();
|
|
for ( ; it != end; ++it ) {
|
|
Cell *obscuringCell = *it;
|
|
|
|
listPoints.append( TQPoint( obscuringCell->column(), obscuringCell->row() ) );
|
|
}
|
|
|
|
TQValueList<TQPoint>::iterator it1 = listPoints.begin();
|
|
TQValueList<TQPoint>::iterator end1 = listPoints.end();
|
|
for ( ; it1 != end1; ++it1 ) {
|
|
TQPoint obscuringCellRef = *it1;
|
|
|
|
// Only paint those obscuring cells that haven't been already
|
|
// painted yet.
|
|
//
|
|
// This optimization removes an O(n^4) behaviour where n is
|
|
// the number of cells on one edge in a merged cell.
|
|
if ( mergedCellsPainted.contains( obscuringCellRef ) )
|
|
continue;
|
|
|
|
Cell *obscuringCell = format()->sheet()->cellAt( obscuringCellRef.x(),
|
|
obscuringCellRef.y() );
|
|
|
|
if ( obscuringCell != 0 ) {
|
|
double x = format()->sheet()->dblColumnPos( obscuringCellRef.x() );
|
|
double y = format()->sheet()->dblRowPos( obscuringCellRef.y() );
|
|
if ( view != 0 ) {
|
|
x -= view->canvasWidget()->xOffset();
|
|
y -= view->canvasWidget()->yOffset();
|
|
}
|
|
|
|
KoPoint corner( x, y );
|
|
painter.save();
|
|
|
|
// Get the effective pens for the borders. These are
|
|
// determined by possible conditions on the cell with
|
|
// associated styles.
|
|
TQPen rp( obscuringCell->effRightBorderPen( obscuringCellRef.x(),
|
|
obscuringCellRef.y() ) );
|
|
TQPen bp( obscuringCell->effBottomBorderPen( obscuringCellRef.x(),
|
|
obscuringCellRef.y() ) );
|
|
TQPen lp( obscuringCell->effLeftBorderPen( obscuringCellRef.x(),
|
|
obscuringCellRef.y() ) );
|
|
TQPen tp( obscuringCell->effTopBorderPen( obscuringCellRef.x(),
|
|
obscuringCellRef.y() ) );
|
|
|
|
|
|
//kdDebug(36001) << " painting obscuring cell "
|
|
// << obscuringCell->name() << endl;
|
|
// TQPen highlightPen;
|
|
|
|
//Note: Painting of highlight isn't quite right. If several
|
|
// cells are merged, then the whole merged cell will be
|
|
// painted with the colour of the last cell referenced
|
|
// which is inside the merged range.
|
|
obscuringCell->paintCell( rect, painter, view,
|
|
corner, obscuringCellRef,
|
|
Border_Left|Border_Top|Border_Right|Border_Bottom,
|
|
rp, bp, lp, tp,
|
|
mergedCellsPainted); // new pens
|
|
painter.restore();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// We are done with the painting, so remove the flag on the cell.
|
|
clearFlag( Flag_PaintingCell );
|
|
}
|
|
|
|
|
|
|
|
// The following code was commented out in the above function. I'll
|
|
// leave it here in case this functionality is ever re-implemented and
|
|
// someone wants some code to start from.
|
|
//
|
|
#if 0
|
|
|
|
/**
|
|
* Modification for drawing the button
|
|
*/
|
|
if ( d->style == Cell::ST_Button ) {
|
|
TQBrush fill( TQt::lightGray );
|
|
TQApplication::style().drawControl( TQStyle::CE_PushButton, &_painter, this,
|
|
TQRect( _tx + 1, _ty + 1, w2 - 1, h2 - 1 ),
|
|
defaultColorGroup ); //, selected, &fill );
|
|
}
|
|
|
|
/**
|
|
* Modification for drawing the combo box
|
|
*/
|
|
else if ( d->style == Cell::ST_Select ) {
|
|
TQApplication::style().drawComboButton( &_painter, _tx + 1, _ty + 1,
|
|
w2 - 1, h2 - 1,
|
|
defaultColorGroup, selected );
|
|
}
|
|
#endif
|
|
|
|
|
|
#if 0
|
|
void Cell::paintCellHighlight(TQPainter& painter,
|
|
const KoRect& cellRect,
|
|
const TQPoint& cellRef,
|
|
const int highlightBorder,
|
|
const TQPen& rightPen,
|
|
const TQPen& bottomPen,
|
|
const TQPen& leftPen,
|
|
const TQPen& topPen
|
|
)
|
|
{
|
|
//painter.drawLine(cellRect.left(),cellRect.top(),cellRect.right(),cellRect.bottom());
|
|
//TQPen pen(d->extra()->highlight);
|
|
//painter.setPen(highlightPen);
|
|
|
|
TQBrush nullBrush;
|
|
painter.setBrush(nullBrush);
|
|
|
|
TQRect zoomedCellRect = sheet()->doc()->zoomRect( cellRect );
|
|
|
|
//The highlight rect is just inside the main cell rect
|
|
//This saves the hassle of repainting nearby cells when the highlight is changed as the highlight areas
|
|
//do not overlap
|
|
zoomedCellRect.setLeft(zoomedCellRect.left()+1);
|
|
//zoomedCellRect.setRight(zoomedCellRect.right()-1);
|
|
zoomedCellRect.setTop(zoomedCellRect.top()+1);
|
|
//zoomedCellRect.setBottom(zoomedCellRect.bottom()-1);
|
|
|
|
if ( cellRef.x() != KS_colMax )
|
|
zoomedCellRect.setWidth( zoomedCellRect.width() - 1 );
|
|
if ( cellRef.y() != KS_rowMax )
|
|
zoomedCellRect.setHeight( zoomedCellRect.height() - 1 );
|
|
|
|
if (highlightBorder & Border_Top)
|
|
{
|
|
painter.setPen(topPen);
|
|
painter.drawLine(zoomedCellRect.left(),zoomedCellRect.top(),zoomedCellRect.right(),zoomedCellRect.top());
|
|
}
|
|
if (highlightBorder & Border_Left)
|
|
{
|
|
painter.setPen(leftPen);
|
|
painter.drawLine(zoomedCellRect.left(),zoomedCellRect.top(),zoomedCellRect.left(),zoomedCellRect.bottom());
|
|
}
|
|
if (highlightBorder & Border_Right)
|
|
{
|
|
painter.setPen(rightPen);
|
|
painter.drawLine(zoomedCellRect.right(),zoomedCellRect.top(),zoomedCellRect.right(),zoomedCellRect.bottom());
|
|
}
|
|
if (highlightBorder & Border_Bottom)
|
|
{
|
|
painter.setPen(bottomPen);
|
|
painter.drawLine(zoomedCellRect.left(),zoomedCellRect.bottom(),zoomedCellRect.right(),zoomedCellRect.bottom());
|
|
}
|
|
|
|
if (highlightBorder & Border_SizeGrip)
|
|
{
|
|
TQBrush brush(rightPen.color());
|
|
painter.setBrush(brush);
|
|
painter.setPen(rightPen);
|
|
painter.drawRect(zoomedCellRect.right()-3,zoomedCellRect.bottom()-3,4,4);
|
|
}
|
|
|
|
//painter.drawRect(zoomedCellRect.left(),zoomedCellRect.top(),zoomedCellRect.width(),zoomedCellRect.height());
|
|
}
|
|
#endif
|
|
|
|
|
|
// Paint all the cells that this cell obscures (helper function to paintCell).
|
|
//
|
|
void Cell::paintObscuredCells(const KoRect& rect, TQPainter& painter,
|
|
View* view,
|
|
const KoRect &cellRect,
|
|
const TQPoint &cellRef,
|
|
bool paintBorderRight,
|
|
bool _paintBorderBottom,
|
|
bool paintBorderLeft,
|
|
bool _paintBorderTop,
|
|
TQPen & rightPen, TQPen & _bottomPen,
|
|
TQPen & leftPen, TQPen & _topPen,
|
|
TQValueList<TQPoint> &mergedCellsPainted)
|
|
{
|
|
// If there are no obscured cells, return.
|
|
if ( !extraXCells() && !extraYCells() )
|
|
return;
|
|
|
|
double ypos = cellRect.y();
|
|
int maxY = extraYCells();
|
|
int maxX = extraXCells();
|
|
|
|
// Loop through the rectangle of squares that we obscure and paint them.
|
|
for ( int y = 0; y <= maxY; ++y ) {
|
|
double xpos = cellRect.x();
|
|
RowFormat* rl = format()->sheet()->rowFormat( cellRef.y() + y );
|
|
|
|
for( int x = 0; x <= maxX; ++ x ) {
|
|
ColumnFormat * cl = format()->sheet()->columnFormat( cellRef.x() + x );
|
|
if ( y != 0 || x != 0 ) {
|
|
uint column = cellRef.x() + x;
|
|
uint row = cellRef.y() + y;
|
|
|
|
TQPen topPen;
|
|
TQPen bottomPen;
|
|
bool paintBorderTop;
|
|
bool paintBorderBottom;
|
|
|
|
Cell *cell = format()->sheet()->cellAt( column, row );
|
|
KoPoint corner( xpos, ypos );
|
|
|
|
// Check if the upper and lower borders should be painted, and
|
|
// if so which pens we should use. There used to be a nasty
|
|
// bug here (#61452).
|
|
// Check top pen. Only check if this is not on the top row.
|
|
topPen = _topPen;
|
|
paintBorderTop = _paintBorderTop;
|
|
if ( row > 1 && !cell->isPartOfMerged() ) {
|
|
Cell *cellUp = format()->sheet()->cellAt( column, row - 1 );
|
|
|
|
if ( cellUp->isDefault() )
|
|
paintBorderTop = false;
|
|
else {
|
|
// If the cell towards the top is part of a merged cell, get
|
|
// the pointer to the master cell.
|
|
cellUp = cellUp->ultimateObscuringCell();
|
|
|
|
topPen = cellUp->effBottomBorderPen( cellUp->column(),
|
|
cellUp->row() );
|
|
|
|
#if 0
|
|
int penWidth = TQMAX(1, sheet()->doc()->zoomItY( topPen.width() ));
|
|
topPen.setWidth( penWidth );
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// FIXME: I thought we had to check bottom pen as well.
|
|
// However, it looks as if we don't need to. It works anyway.
|
|
bottomPen = _bottomPen;
|
|
paintBorderBottom = _paintBorderBottom;
|
|
|
|
int paintBorder = Border_None;
|
|
if (paintBorderLeft) paintBorder |= Cell::Border_Left;
|
|
if (paintBorderRight) paintBorder |= Cell::Border_Right;
|
|
if (paintBorderTop) paintBorder |= Cell::Border_Top;
|
|
if (paintBorderBottom) paintBorder |= Cell::Border_Bottom;
|
|
|
|
/*Cell::BorderSides highlightBorder = Border_None;
|
|
TQPen highlightPen;*/
|
|
|
|
|
|
//kdDebug(36001) << "calling paintcell for obscured cell "
|
|
// << cell->name() << endl;
|
|
cell->paintCell( rect, painter, view,
|
|
corner,
|
|
TQPoint( cellRef.x() + x, cellRef.y() + y ),
|
|
paintBorder,
|
|
rightPen, bottomPen, leftPen, topPen,
|
|
mergedCellsPainted);
|
|
}
|
|
xpos += cl->dblWidth();
|
|
}
|
|
|
|
ypos += rl->dblHeight();
|
|
}
|
|
}
|
|
|
|
|
|
// Paint the background of this cell.
|
|
//
|
|
void Cell::paintBackground( TQPainter& painter, const KoRect &cellRect,
|
|
const TQPoint &cellRef, bool selected,
|
|
TQColor &backgroundColor )
|
|
{
|
|
TQColorGroup defaultColorGroup = TQApplication::palette().active();
|
|
TQRect zoomedCellRect = sheet()->doc()->zoomRect( cellRect );
|
|
|
|
// If this is not the KS_rowMax and/or KS_colMax, then we reduce
|
|
// width and/or height by one. This is due to the fact that the
|
|
// right/bottom most pixel is shared with the left/top most pixel of
|
|
// the following cell. Only in the case of KS_colMax/KS_rowMax we
|
|
// need to draw even this pixel, as there isn't a following cell to
|
|
// draw the background pixel.
|
|
if ( cellRef.x() != KS_colMax )
|
|
zoomedCellRect.setWidth( zoomedCellRect.width() - 1 );
|
|
if ( cellRef.y() != KS_rowMax )
|
|
zoomedCellRect.setHeight( zoomedCellRect.height() - 1 );
|
|
|
|
// Determine the correct background color
|
|
if ( selected )
|
|
{
|
|
//If the cell's background color is too bright, use the default highlight color
|
|
//Otherwise use a lighter version of the cell's background color.
|
|
TQColor c;
|
|
|
|
int averageColor = (backgroundColor.red() + backgroundColor.green() + backgroundColor.blue()) / 3;
|
|
|
|
if (averageColor > 180)
|
|
if (averageColor > 225)
|
|
c = View::highlightColor();
|
|
else
|
|
c = backgroundColor.light( 115 ); //15% lighter
|
|
else
|
|
c = backgroundColor.light( 125 ); //25% lighter
|
|
|
|
painter.setBackgroundColor( c );
|
|
}
|
|
else {
|
|
TQColor bg( backgroundColor );
|
|
|
|
// Handle printers separately.
|
|
if ( !painter.device()->isExtDev() ) {
|
|
if ( bg.isValid() )
|
|
painter.setBackgroundColor( bg );
|
|
else
|
|
painter.setBackgroundColor( defaultColorGroup.base() );
|
|
}
|
|
else {
|
|
//bad hack but there is a qt bug
|
|
//so I can print backgroundcolor
|
|
TQBrush bb( bg );
|
|
if ( !bg.isValid() )
|
|
bb.setColor( TQt::white );
|
|
|
|
painter.fillRect( zoomedCellRect, bb );
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Erase the background of the cell.
|
|
if ( !painter.device()->isExtDev() )
|
|
painter.eraseRect( zoomedCellRect );
|
|
|
|
// Get a background brush
|
|
TQBrush bb;
|
|
if ( d->hasExtra()
|
|
&& d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SBackgroundBrush, true ) )
|
|
bb = d->extra()->conditions->matchedStyle()->backGroundBrush();
|
|
else
|
|
bb = backGroundBrush( cellRef.x(), cellRef.y() );
|
|
|
|
// Draw background pattern if necessary.
|
|
if ( bb.style() != TQt::NoBrush )
|
|
painter.fillRect( zoomedCellRect, bb );
|
|
|
|
backgroundColor = painter.backgroundColor();
|
|
}
|
|
|
|
|
|
// Paint the standard light grey borders that are always visible.
|
|
//
|
|
void Cell::paintDefaultBorders( TQPainter& painter, const KoRect &rect,
|
|
const KoRect &cellRect,
|
|
const TQPoint &cellRef,
|
|
bool paintBorderRight, bool /*paintBorderBottom*/,
|
|
bool paintBorderLeft, bool paintBorderTop,
|
|
TQPen const & rightPen, TQPen const & /*bottomPen*/,
|
|
TQPen const & leftPen, TQPen const & topPen )
|
|
{
|
|
/*
|
|
*** Notes about optimisation ***
|
|
|
|
This function was painting the top , left , right & bottom lines in almost all cells previously, contrary to what the comment
|
|
below says should happen. There doesn't appear to be a UI option to enable or disable showing of the grid when printing at the moment,
|
|
so I have disabled drawing of right and bottom borders for all cells.
|
|
|
|
I also couldn't work out under what conditions the variables dt / db would come out as anything other than 0 in the code
|
|
for painting the various borders. The effTopBorderPen / effBottomBorderPen calls were taking up a lot of time
|
|
according some profiling I did. If that code really is necessary, we need to find a more efficient way of getting the widths
|
|
than grabbing the whole TQPen object and asking it.
|
|
|
|
|
|
--Robert Knight (robertknight@gmail.com)
|
|
*/
|
|
Doc* doc = sheet()->doc();
|
|
|
|
Sheet::LayoutDirection sheetDir = format()->sheet()->layoutDirection();
|
|
bool paintingToExternalDevice = painter.device()->isExtDev();
|
|
|
|
// Each cell is responsible for drawing it's top and left portions
|
|
// of the "default" grid. --Or not drawing it if it shouldn't be
|
|
// there. It's also responsible to paint the right and bottom, if
|
|
// it is the last cell on a print out.
|
|
|
|
bool paintTop;
|
|
bool paintLeft;
|
|
bool paintBottom=false;
|
|
bool paintRight=false;
|
|
|
|
paintLeft = ( paintBorderLeft && leftPen.style() == TQt::NoPen
|
|
&& sheet()->getShowGrid() && sheetDir==Sheet::LeftToRight );
|
|
paintRight = ( paintBorderRight && rightPen.style() == TQt::NoPen
|
|
&& sheet()->getShowGrid() && sheetDir==Sheet::RightToLeft );
|
|
paintTop = ( paintBorderTop && topPen.style() == TQt::NoPen
|
|
&& sheet()->getShowGrid() );
|
|
// paintBottom = ( paintBorderBottom && sheet()->getShowGrid()
|
|
// && bottomPen.style() == TQt::NoPen );
|
|
|
|
|
|
//Set the single-pixel with pen for drawing the borders with.
|
|
painter.setPen( TQPen( sheet()->doc()->gridColor(), 1, TQt::SolidLine ) );
|
|
|
|
// If there are extra cells, there might be more conditions.
|
|
if (d->hasExtra()) {
|
|
TQValueList<Cell*>::const_iterator it = d->extra()->obscuringCells.begin();
|
|
TQValueList<Cell*>::const_iterator end = d->extra()->obscuringCells.end();
|
|
for ( ; it != end; ++it ) {
|
|
Cell *cell = *it;
|
|
|
|
paintTop = paintTop && ( cell->row() == cellRef.y() );
|
|
paintBottom = false;
|
|
|
|
if ( sheetDir == Sheet::RightToLeft ) {
|
|
paintRight = paintRight && ( cell->column() == cellRef.x() );
|
|
paintLeft = false;
|
|
}
|
|
else {
|
|
paintLeft = paintLeft && ( cell->column() == cellRef.x() );
|
|
paintRight = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
// The left border.
|
|
if ( paintLeft ) {
|
|
int dt = 0;
|
|
int db = 0;
|
|
|
|
#if 0
|
|
if ( cellRef.x() > 1 ) {
|
|
Cell *cell_west = format()->sheet()->cellAt( cellRef.x() - 1,
|
|
cellRef.y() );
|
|
TQPen t = cell_west->effTopBorderPen( cellRef.x() - 1, cellRef.y() );
|
|
TQPen b = cell_west->effBottomBorderPen( cellRef.x() - 1, cellRef.y() );
|
|
|
|
if ( t.style() != TQt::NoPen )
|
|
dt = ( t.width() + 1 )/2;
|
|
if ( b.style() != TQt::NoPen )
|
|
db = ( t.width() / 2);
|
|
}
|
|
#endif
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( paintingToExternalDevice ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( doc->zoomItX( TQMAX( rect.left(), cellRect.right() ) ),
|
|
doc->zoomItY( TQMAX( rect.top(), cellRect.y() + dt ) ),
|
|
doc->zoomItX( TQMIN( rect.right(), cellRect.right() ) ),
|
|
doc->zoomItY( TQMIN( rect.bottom(), cellRect.bottom() - db ) ) );
|
|
else
|
|
painter.drawLine( doc->zoomItX( TQMAX( rect.left(), cellRect.x() ) ),
|
|
doc->zoomItY( TQMAX( rect.top(), cellRect.y() + dt ) ),
|
|
doc->zoomItX( TQMIN( rect.right(), cellRect.x() ) ),
|
|
doc->zoomItY( TQMIN( rect.bottom(), cellRect.bottom() - db ) ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.y() + dt ),
|
|
doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.bottom() - db ) );
|
|
else
|
|
painter.drawLine( doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.y() + dt ),
|
|
doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.bottom() - db ) );
|
|
}
|
|
}
|
|
|
|
|
|
// The top border.
|
|
if ( paintTop ) {
|
|
int dl = 0;
|
|
int dr = 0;
|
|
|
|
#if 0
|
|
if ( cellRef.y() > 1 ) {
|
|
Cell *cell_north = format()->sheet()->cellAt( cellRef.x(),
|
|
cellRef.y() - 1 );
|
|
|
|
TQPen l = cell_north->effLeftBorderPen( cellRef.x(), cellRef.y() - 1 );
|
|
TQPen r = cell_north->effRightBorderPen( cellRef.x(), cellRef.y() - 1 );
|
|
|
|
if ( l.style() != TQt::NoPen )
|
|
dl = ( l.width() - 1 ) / 2 + 1;
|
|
if ( r.style() != TQt::NoPen )
|
|
dr = r.width() / 2;
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( paintingToExternalDevice ) {
|
|
painter.drawLine( doc->zoomItX( TQMAX( rect.left(), cellRect.x() + dl ) ),
|
|
doc->zoomItY( TQMAX( rect.top(), cellRect.y() ) ),
|
|
doc->zoomItX( TQMIN( rect.right(), cellRect.right() - dr ) ),
|
|
doc->zoomItY( TQMIN( rect.bottom(), cellRect.y() ) ) );
|
|
}
|
|
else {
|
|
painter.drawLine( doc->zoomItX( cellRect.x() + dl ),
|
|
doc->zoomItY( cellRect.y() ),
|
|
doc->zoomItX( cellRect.right() - dr ),
|
|
doc->zoomItY( cellRect.y() ) );
|
|
}
|
|
}
|
|
|
|
|
|
// The right border.
|
|
if ( paintRight ) {
|
|
int dt = 0;
|
|
int db = 0;
|
|
|
|
#if 0
|
|
if ( cellRef.x() < KS_colMax ) {
|
|
Cell *cell_east = format()->sheet()->cellAt( cellRef.x() + 1,
|
|
cellRef.y() );
|
|
|
|
TQPen t = cell_east->effTopBorderPen( cellRef.x() + 1, cellRef.y() );
|
|
TQPen b = cell_east->effBottomBorderPen( cellRef.x() + 1, cellRef.y() );
|
|
|
|
if ( t.style() != TQt::NoPen )
|
|
dt = ( t.width() + 1 ) / 2;
|
|
if ( b.style() != TQt::NoPen )
|
|
db = ( t.width() / 2);
|
|
}
|
|
#endif
|
|
|
|
//painter.setPen( TQPen( sheet()->doc()->gridColor(), 1, TQt::SolidLine ) );
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( doc->zoomItX( TQMAX( rect.left(), cellRect.x() ) ),
|
|
doc->zoomItY( TQMAX( rect.top(), cellRect.y() + dt ) ),
|
|
doc->zoomItX( TQMIN( rect.right(), cellRect.x() ) ),
|
|
doc->zoomItY( TQMIN( rect.bottom(), cellRect.bottom() - db ) ) );
|
|
else
|
|
painter.drawLine( doc->zoomItX( TQMAX( rect.left(), cellRect.right() ) ),
|
|
doc->zoomItY( TQMAX( rect.top(), cellRect.y() + dt ) ),
|
|
doc->zoomItX( TQMIN( rect.right(), cellRect.right() ) ),
|
|
doc->zoomItY( TQMIN( rect.bottom(), cellRect.bottom() - db ) ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.y() + dt ),
|
|
doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.bottom() - db ) );
|
|
else
|
|
painter.drawLine( doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.y() + dt ),
|
|
doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.bottom() - db ) );
|
|
}
|
|
}
|
|
|
|
// The bottom border.
|
|
/*if ( paintBottom ) {
|
|
int dl = 0;
|
|
int dr = 0;
|
|
if ( cellRef.y() < KS_rowMax ) {
|
|
Cell *cell_south = format()->sheet()->cellAt( cellRef.x(),
|
|
cellRef.y() + 1 );
|
|
|
|
TQPen l = cell_south->effLeftBorderPen( cellRef.x(), cellRef.y() + 1 );
|
|
TQPen r = cell_south->effRightBorderPen( cellRef.x(), cellRef.y() + 1 );
|
|
|
|
if ( l.style() != TQt::NoPen )
|
|
dl = ( l.width() - 1 ) / 2 + 1;
|
|
if ( r.style() != TQt::NoPen )
|
|
dr = r.width() / 2;
|
|
}
|
|
|
|
painter.setPen( TQPen( sheet()->doc()->gridColor(), 1, TQt::SolidLine ) );
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
painter.drawLine( doc->zoomItX( TQMAX( rect.left(), cellRect.x() + dl ) ),
|
|
doc->zoomItY( TQMAX( rect.top(), cellRect.bottom() ) ),
|
|
doc->zoomItX( TQMIN( rect.right(), cellRect.right() - dr ) ),
|
|
doc->zoomItY( TQMIN( rect.bottom(), cellRect.bottom() ) ) );
|
|
}
|
|
else {
|
|
painter.drawLine( doc->zoomItX( cellRect.x() + dl ),
|
|
doc->zoomItY( cellRect.bottom() ),
|
|
doc->zoomItX( cellRect.right() - dr ),
|
|
doc->zoomItY( cellRect.bottom() ) );
|
|
}
|
|
}*/
|
|
}
|
|
|
|
|
|
// Paint a comment indicator if the cell has a comment.
|
|
//
|
|
void Cell::paintCommentIndicator( TQPainter& painter,
|
|
const KoRect &cellRect,
|
|
const TQPoint &/*cellRef*/,
|
|
TQColor &backgroundColor )
|
|
{
|
|
Doc * doc = sheet()->doc();
|
|
|
|
// Point the little corner if there is a comment attached
|
|
// to this cell.
|
|
if ( ( format()->propertiesMask() & (uint) Format::PComment )
|
|
&& cellRect.width() > 10.0
|
|
&& cellRect.height() > 10.0
|
|
&& ( sheet()->print()->printCommentIndicator()
|
|
|| ( !painter.device()->isExtDev() && sheet()->getShowCommentIndicator() ) ) ) {
|
|
TQColor penColor = TQt::red;
|
|
|
|
// If background has high red part, switch to blue.
|
|
if ( tqRed( backgroundColor.rgb() ) > 127 &&
|
|
tqGreen( backgroundColor.rgb() ) < 80 &&
|
|
tqBlue( backgroundColor.rgb() ) < 80 )
|
|
{
|
|
penColor = TQt::blue;
|
|
}
|
|
|
|
// Get the triangle.
|
|
TQPointArray point( 3 );
|
|
if ( format()->sheet()->layoutDirection()==Sheet::RightToLeft ) {
|
|
point.setPoint( 0, doc->zoomItX( cellRect.x() + 6.0 ),
|
|
doc->zoomItY( cellRect.y() ) );
|
|
point.setPoint( 1, doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.y() ) );
|
|
point.setPoint( 2, doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.y() + 6.0 ) );
|
|
}
|
|
else {
|
|
point.setPoint( 0, doc->zoomItX( cellRect.right() - 5.0 ),
|
|
doc->zoomItY( cellRect.y() ) );
|
|
point.setPoint( 1, doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.y() ) );
|
|
point.setPoint( 2, doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.y() + 5.0 ) );
|
|
}
|
|
|
|
// And draw it.
|
|
painter.setBrush( TQBrush( penColor ) );
|
|
painter.setPen( TQt::NoPen );
|
|
painter.drawPolygon( point );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// Paint a small rectangle if this cell holds a formula.
|
|
//
|
|
void Cell::paintFormulaIndicator( TQPainter& painter,
|
|
const KoRect &cellRect,
|
|
TQColor &backgroundColor )
|
|
{
|
|
if ( isFormula() &&
|
|
format()->sheet()->getShowFormulaIndicator() &&
|
|
cellRect.width() > 10.0 &&
|
|
cellRect.height() > 10.0 )
|
|
{
|
|
Doc* doc = sheet()->doc();
|
|
|
|
TQColor penColor = TQt::blue;
|
|
// If background has high blue part, switch to red.
|
|
if ( tqRed( backgroundColor.rgb() ) < 80 &&
|
|
tqGreen( backgroundColor.rgb() ) < 80 &&
|
|
tqBlue( backgroundColor.rgb() ) > 127 )
|
|
{
|
|
penColor = TQt::red;
|
|
}
|
|
|
|
// Get the triangle...
|
|
TQPointArray point( 3 );
|
|
if ( format()->sheet()->layoutDirection()==Sheet::RightToLeft ) {
|
|
point.setPoint( 0, doc->zoomItX( cellRect.right() - 6.0 ),
|
|
doc->zoomItY( cellRect.bottom() ) );
|
|
point.setPoint( 1, doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.bottom() ) );
|
|
point.setPoint( 2, doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.bottom() - 6.0 ) );
|
|
}
|
|
else {
|
|
point.setPoint( 0, doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.bottom() - 6.0 ) );
|
|
point.setPoint( 1, doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.bottom() ) );
|
|
point.setPoint( 2, doc->zoomItX( cellRect.x() + 6.0 ),
|
|
doc->zoomItY( cellRect.bottom() ) );
|
|
}
|
|
|
|
// ...and draw it.
|
|
painter.setBrush( TQBrush( penColor ) );
|
|
painter.setPen( TQt::NoPen );
|
|
painter.drawPolygon( point );
|
|
}
|
|
}
|
|
|
|
|
|
// Paint an indicator that the text in the cell is cut.
|
|
//
|
|
void Cell::paintMoreTextIndicator( TQPainter& painter,
|
|
const KoRect &cellRect,
|
|
TQColor &backgroundColor )
|
|
{
|
|
// Show a red triangle when it's not possible to write all text in cell.
|
|
// Don't print the red triangle if we're printing.
|
|
if( testFlag( Flag_CellTooShortX ) &&
|
|
!painter.device()->isExtDev() &&
|
|
cellRect.height() > 4.0 &&
|
|
cellRect.width() > 4.0 )
|
|
{
|
|
Doc* doc = sheet()->doc();
|
|
|
|
TQColor penColor = TQt::red;
|
|
// If background has high red part, switch to blue.
|
|
if ( tqRed( backgroundColor.rgb() ) > 127
|
|
&& tqGreen( backgroundColor.rgb() ) < 80
|
|
&& tqBlue( backgroundColor.rgb() ) < 80 )
|
|
{
|
|
penColor = TQt::blue;
|
|
}
|
|
|
|
// Get the triangle...
|
|
TQPointArray point( 3 );
|
|
if ( d->strOutText.isRightToLeft() ) {
|
|
point.setPoint( 0, doc->zoomItX( cellRect.left() + 4.0 ),
|
|
doc->zoomItY( cellRect.y() + cellRect.height() / 2.0 -4.0 ) );
|
|
point.setPoint( 1, doc->zoomItX( cellRect.left() ),
|
|
doc->zoomItY( cellRect.y() + cellRect.height() / 2.0 ));
|
|
point.setPoint( 2, doc->zoomItX( cellRect.left() + 4.0 ),
|
|
doc->zoomItY( cellRect.y() + cellRect.height() / 2.0 +4.0 ) );
|
|
}
|
|
else {
|
|
point.setPoint( 0, doc->zoomItX( cellRect.right() - 4.0 ),
|
|
doc->zoomItY( cellRect.y() + cellRect.height() / 2.0 - 4.0 ) );
|
|
point.setPoint( 1, doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.y() + cellRect.height() / 2.0 ) );
|
|
point.setPoint( 2, doc->zoomItX( cellRect.right() - 4.0 ),
|
|
doc->zoomItY( cellRect.y() + cellRect.height() / 2.0 + 4.0 ) );
|
|
}
|
|
|
|
// ...and paint it.
|
|
painter.setBrush( TQBrush( penColor ) );
|
|
painter.setPen( TQt::NoPen );
|
|
painter.drawPolygon( point );
|
|
}
|
|
}
|
|
|
|
|
|
// Paint the real contents of a cell - the text.
|
|
//
|
|
void Cell::paintText( TQPainter& painter,
|
|
const KoRect &cellRect,
|
|
const TQPoint &cellRef )
|
|
{
|
|
Doc *doc = sheet()->doc();
|
|
|
|
ColumnFormat *colFormat = format()->sheet()->columnFormat( cellRef.x() );
|
|
|
|
TQColorGroup defaultColorGroup = TQApplication::palette().active();
|
|
TQColor textColorPrint = effTextColor( cellRef.x(), cellRef.y() );
|
|
|
|
// Resolve the text color if invalid (=default).
|
|
if ( !textColorPrint.isValid() ) {
|
|
if ( painter.device()->isExtDev() )
|
|
textColorPrint = TQt::black;
|
|
else
|
|
textColorPrint = TQApplication::palette().active().text();
|
|
}
|
|
|
|
TQPen tmpPen( textColorPrint );
|
|
|
|
// Set the font according to the current zoom.
|
|
applyZoomedFont( painter, cellRef.x(), cellRef.y() );
|
|
|
|
// Check for red font color for negative values.
|
|
if ( !d->hasExtra()
|
|
|| !d->extra()->conditions
|
|
|| !d->extra()->conditions->matchedStyle() ) {
|
|
if ( value().isNumber()
|
|
&& !( format()->sheet()->getShowFormula()
|
|
&& !( format()->sheet()->isProtected()
|
|
&& format()->isHideFormula( d->column, d->row ) ) ) )
|
|
{
|
|
double v = value().asFloat();
|
|
if ( format()->floatColor( cellRef.x(), cellRef.y()) == Format::NegRed
|
|
&& v < 0.0 )
|
|
tmpPen.setColor( TQt::red );
|
|
}
|
|
}
|
|
|
|
// Check for blue color, for hyperlink.
|
|
if ( !link().isEmpty() ) {
|
|
tmpPen.setColor( TQApplication::palette().active().link() );
|
|
TQFont f = painter.font();
|
|
f.setUnderline( true );
|
|
painter.setFont( f );
|
|
}
|
|
|
|
#if 0
|
|
/****
|
|
|
|
For now I am commenting this out -- with the default color display you
|
|
can read normal text through a highlighted background. Maybe this isn't
|
|
always the case, though, and we can put the highlighted text color back in.
|
|
In that case, we need to somewhere in here figure out if the text overlaps
|
|
another cell outside of the selection, otherwise that portion of the text
|
|
will be printed white on white. So just that portion would need to be
|
|
painted again in the normal color.
|
|
|
|
This should probably be done eventually, anyway, because I like using the
|
|
reverse text color for highlighted cells. I just don't like extending the
|
|
cell 'highlight' background outside of the selection rectangle because it
|
|
looks REALLY ugly.
|
|
*/
|
|
|
|
if ( selected && ( cellRef.x() != marker.x() || cellRef.y() != marker.y() ) )
|
|
{
|
|
TQPen p( tmpPen );
|
|
p.setColor( defaultColorGroup.highlightedText() );
|
|
painter.setPen( p );
|
|
}
|
|
else {
|
|
painter.setPen(tmpPen);
|
|
}
|
|
#endif
|
|
painter.setPen( tmpPen );
|
|
|
|
TQString tmpText = d->strOutText;
|
|
double tmpHeight = d->textHeight;
|
|
double tmpWidth = d->textWidth;
|
|
|
|
// If the cell is to narrow to paint the whole contents, then pick
|
|
// out a part of the content that we paint. The result of this is
|
|
// dependent on the data type of the content.
|
|
//
|
|
// FIXME: Make this dependent on the height as well.
|
|
//
|
|
if ( testFlag( Flag_CellTooShortX ) ) {
|
|
d->strOutText = textDisplaying( painter );
|
|
|
|
// Recalculate the text width and the offset.
|
|
textSize( painter );
|
|
offsetAlign( column(), row() );
|
|
}
|
|
|
|
// Hide zero.
|
|
if ( format()->sheet()->getHideZero()
|
|
&& value().isNumber()
|
|
&& value().asFloat() == 0 ) {
|
|
d->strOutText = TQString();
|
|
}
|
|
|
|
// Clear extra cell if column or row is hidden
|
|
//
|
|
// FIXME: I think this should be done before the call to
|
|
// textDisplaying() above.
|
|
//
|
|
if ( colFormat->isHide() || ( cellRect.height() <= 2 ) ) {
|
|
freeAllObscuredCells(); /* TODO: This looks dangerous...must check when I
|
|
have time */
|
|
d->strOutText = "";
|
|
}
|
|
|
|
double indent = 0.0;
|
|
double offsetCellTooShort = 0.0;
|
|
int a = effAlignX();
|
|
|
|
// Apply indent if text is align to left not when text is at right or middle.
|
|
if ( a == Format::Left && !isEmpty() ) {
|
|
// FIXME: The following condition should be remade into a call to
|
|
// a new convenience function:
|
|
// if ( hasConditionStyleFeature( Style::SIndent, true )...
|
|
// This should be done throughout the entire file.
|
|
//
|
|
if ( d->hasExtra()
|
|
&& d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SIndent, true ) )
|
|
indent = d->extra()->conditions->matchedStyle()->indent();
|
|
else
|
|
indent = format()->getIndent( column(), row() );
|
|
}
|
|
|
|
// Made an offset, otherwise ### is under red triangle.
|
|
if ( a == Format::Right && !isEmpty() && testFlag( Flag_CellTooShortX ) )
|
|
offsetCellTooShort = format()->sheet()->doc()->unzoomItX( 4 );
|
|
|
|
TQFontMetrics fm2 = painter.fontMetrics();
|
|
double offsetFont = 0.0;
|
|
|
|
if ( format()->alignY( column(), row() ) == Format::Bottom
|
|
&& format()->textFontUnderline( column(), row() ) )
|
|
offsetFont = format()->sheet()->doc()->unzoomItX( fm2.underlinePos() + 1 );
|
|
|
|
int tmpAngle;
|
|
bool tmpMultiRow;
|
|
bool tmpVerticalText;
|
|
|
|
// Check for angled or vertical text.
|
|
if ( d->hasExtra()
|
|
&& d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
{
|
|
Style *matchedStyle = d->extra()->conditions->matchedStyle();
|
|
|
|
if ( matchedStyle->hasFeature( Style::SAngle, true ) )
|
|
tmpAngle = d->extra()->conditions->matchedStyle()->rotateAngle();
|
|
else
|
|
tmpAngle = format()->getAngle( cellRef.x(), cellRef.y() );
|
|
|
|
if ( matchedStyle->hasFeature( Style::SVerticalText, true ) )
|
|
tmpVerticalText = matchedStyle->hasProperty( Style::PVerticalText );
|
|
else
|
|
tmpVerticalText = format()->verticalText( cellRef.x(), cellRef.y() );
|
|
|
|
if ( matchedStyle->hasFeature( Style::SMultiRow, true ) )
|
|
tmpMultiRow = matchedStyle->hasProperty( Style::PMultiRow );
|
|
else
|
|
tmpMultiRow = format()->multiRow( cellRef.x(), cellRef.y() );
|
|
}
|
|
else {
|
|
tmpAngle = format()->getAngle( cellRef.x(), cellRef.y() );
|
|
tmpVerticalText = format()->verticalText( cellRef.x(), cellRef.y() );
|
|
tmpMultiRow = format()->multiRow( cellRef.x(), cellRef.y() );
|
|
}
|
|
|
|
// Actually paint the text.
|
|
// There are 4 possible cases:
|
|
// - One line of text , horizontal
|
|
// - Angled text
|
|
// - Multiple rows of text , horizontal
|
|
// - Vertical text
|
|
if ( !tmpMultiRow && !tmpVerticalText && !tmpAngle ) {
|
|
// Case 1: The simple case, one line, no angle.
|
|
|
|
painter.drawText( doc->zoomItX( indent + cellRect.x() + d->textX - offsetCellTooShort ),
|
|
doc->zoomItY( cellRect.y() + d->textY - offsetFont ), d->strOutText );
|
|
}
|
|
else if ( tmpAngle != 0 ) {
|
|
// Case 2: an angle.
|
|
|
|
int angle = tmpAngle;
|
|
TQFontMetrics fm = painter.fontMetrics();
|
|
|
|
painter.rotate( angle );
|
|
double x;
|
|
if ( angle > 0 )
|
|
x = indent + cellRect.x() + d->textX;
|
|
else
|
|
x = indent + cellRect.x() + d->textX
|
|
- doc->unzoomItX((int) (( fm.descent() + fm.ascent() ) * sin( angle * M_PI / 180 )));
|
|
double y;
|
|
if ( angle > 0 )
|
|
y = cellRect.y() + d->textY;
|
|
else
|
|
y = cellRect.y() + d->textY + d->textHeight;
|
|
painter.drawText( doc->zoomItX( x * cos( angle * M_PI / 180 ) +
|
|
y * sin( angle * M_PI / 180 ) ),
|
|
doc->zoomItY( -x * sin( angle * M_PI / 180 ) +
|
|
y * cos( angle * M_PI / 180 ) ),
|
|
d->strOutText );
|
|
painter.rotate( -angle );
|
|
}
|
|
else if ( tmpMultiRow && !tmpVerticalText ) {
|
|
// Case 3: Multiple rows, but horizontal.
|
|
|
|
TQString t;
|
|
int i;
|
|
int pos = 0;
|
|
double dy = 0.0;
|
|
TQFontMetrics fm = painter.fontMetrics();
|
|
do {
|
|
i = d->strOutText.find( "\n", pos );
|
|
if ( i == -1 )
|
|
t = d->strOutText.mid( pos, d->strOutText.length() - pos );
|
|
else {
|
|
t = d->strOutText.mid( pos, i - pos );
|
|
pos = i + 1;
|
|
}
|
|
|
|
int align = effAlignX();
|
|
if ( format()->sheet()->getShowFormula()
|
|
&& !( format()->sheet()->isProtected()
|
|
&& format()->isHideFormula( d->column, d->row ) ) )
|
|
align = Format::Left;
|
|
|
|
// #### Torben: This looks duplicated for me
|
|
switch ( align ) {
|
|
case Format::Left:
|
|
d->textX = effLeftBorderPen( cellRef.x(), cellRef.y() ).width() + BORDER_SPACE;
|
|
break;
|
|
|
|
case Format::Right:
|
|
d->textX = cellRect.width() - BORDER_SPACE - doc->unzoomItX( fm.width( t ) )
|
|
- effRightBorderPen( cellRef.x(), cellRef.y() ).width();
|
|
break;
|
|
|
|
case Format::Center:
|
|
d->textX = ( cellRect.width() - doc->unzoomItX( fm.width( t ) ) ) / 2;
|
|
}
|
|
|
|
painter.drawText( doc->zoomItX( indent + cellRect.x() + d->textX ),
|
|
doc->zoomItY( cellRect.y() + d->textY + dy ), t );
|
|
dy += doc->unzoomItY( fm.descent() + fm.ascent() );
|
|
} while ( i != -1 );
|
|
}
|
|
else if ( tmpVerticalText && !d->strOutText.isEmpty() ) {
|
|
// Case 4: Vertical text.
|
|
|
|
TQString t;
|
|
int i = 0;
|
|
int len = 0;
|
|
double dy = 0.0;
|
|
TQFontMetrics fm = painter.fontMetrics();
|
|
do {
|
|
len = d->strOutText.length();
|
|
t = d->strOutText.at( i );
|
|
painter.drawText( doc->zoomItX( indent + cellRect.x() + d->textX ),
|
|
doc->zoomItY( cellRect.y() + d->textY + dy ), t );
|
|
dy += doc->unzoomItY( fm.descent() + fm.ascent() );
|
|
i++;
|
|
} while ( i != len );
|
|
}
|
|
|
|
// Check for too short cell and set the outText for future reference.
|
|
if ( testFlag( Flag_CellTooShortX ) ) {
|
|
d->strOutText = tmpText;
|
|
d->textHeight = tmpHeight;
|
|
d->textWidth = tmpWidth;
|
|
}
|
|
|
|
if ( format()->sheet()->getHideZero() && value().isNumber()
|
|
&& value().asFloat() == 0 )
|
|
d->strOutText = tmpText;
|
|
|
|
if ( colFormat->isHide() || ( cellRect.height() <= 2 ) )
|
|
d->strOutText = tmpText;
|
|
}
|
|
|
|
|
|
// Paint page borders on the page. Only do this on the screen.
|
|
//
|
|
void Cell::paintPageBorders( TQPainter& painter,
|
|
const KoRect &cellRect,
|
|
const TQPoint &cellRef,
|
|
bool paintBorderRight,
|
|
bool paintBorderBottom )
|
|
{
|
|
// Not screen? Return immediately.
|
|
if ( painter.device()->isExtDev() )
|
|
return;
|
|
|
|
if ( ! format()->sheet()->isShowPageBorders() )
|
|
return;
|
|
|
|
SheetPrint* print = format()->sheet()->print();
|
|
|
|
Sheet::LayoutDirection sheetDir = format()->sheet()->layoutDirection();
|
|
|
|
Doc* doc = sheet()->doc();
|
|
int zcellRect_left = doc->zoomItX (cellRect.left());
|
|
int zcellRect_right = doc->zoomItX (cellRect.right());
|
|
int zcellRect_top = doc->zoomItY (cellRect.top());
|
|
int zcellRect_bottom = doc->zoomItY (cellRect.bottom());
|
|
|
|
// Draw page borders
|
|
|
|
if ( cellRef.x() >= print->printRange().left()
|
|
&& cellRef.x() <= print->printRange().right() + 1
|
|
&& cellRef.y() >= print->printRange().top()
|
|
&& cellRef.y() <= print->printRange().bottom() + 1 )
|
|
{
|
|
if ( print->isOnNewPageX( cellRef.x() )
|
|
&& cellRef.y() <= print->printRange().bottom() )
|
|
{
|
|
painter.setPen( sheet()->doc()->pageBorderColor() );
|
|
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_right, zcellRect_top,
|
|
zcellRect_right, zcellRect_bottom );
|
|
else
|
|
painter.drawLine( zcellRect_left, zcellRect_top,
|
|
zcellRect_left, zcellRect_bottom );
|
|
}
|
|
|
|
if ( print->isOnNewPageY( cellRef.y() ) &&
|
|
( cellRef.x() <= print->printRange().right() ) )
|
|
{
|
|
painter.setPen( sheet()->doc()->pageBorderColor() );
|
|
painter.drawLine( zcellRect_left, zcellRect_top,
|
|
zcellRect_right, zcellRect_top );
|
|
}
|
|
|
|
if ( paintBorderRight ) {
|
|
if ( print->isOnNewPageX( cellRef.x() + 1 )
|
|
&& cellRef.y() <= print->printRange().bottom() ) {
|
|
painter.setPen( sheet()->doc()->pageBorderColor() );
|
|
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_left, zcellRect_top,
|
|
zcellRect_left, zcellRect_bottom );
|
|
else
|
|
painter.drawLine( zcellRect_right, zcellRect_top,
|
|
zcellRect_right, zcellRect_bottom );
|
|
}
|
|
}
|
|
|
|
if ( paintBorderBottom ) {
|
|
if ( print->isOnNewPageY( cellRef.y() + 1 )
|
|
&& cellRef.x() <= print->printRange().right() ) {
|
|
painter.setPen( sheet()->doc()->pageBorderColor() );
|
|
painter.drawLine( zcellRect_left, zcellRect_bottom,
|
|
zcellRect_right, zcellRect_bottom );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// Paint the cell borders.
|
|
//
|
|
void Cell::paintCellBorders( TQPainter& painter, const KoRect& rect,
|
|
const KoRect &cellRect,
|
|
const TQPoint &cellRef,
|
|
bool paintRight, bool paintBottom,
|
|
bool paintLeft, bool paintTop,
|
|
TQPen & _rightPen, TQPen & _bottomPen,
|
|
TQPen & _leftPen, TQPen & _topPen )
|
|
{
|
|
|
|
//Sanity check: If we are not painting any of the borders then the function
|
|
//really shouldn't be called at all.
|
|
if ( (!paintLeft) && (!paintRight) && (!paintTop) && (!paintBottom) )
|
|
return;
|
|
|
|
Doc * doc = sheet()->doc();
|
|
|
|
Sheet::LayoutDirection sheetDir = format()->sheet()->layoutDirection();
|
|
|
|
// compute zoomed rectangles
|
|
// I don't use KoRect, because that ends up producing lots of warnings
|
|
// about double->int conversions in calls to painter.drawLine
|
|
int zrect_left (doc->zoomItX (rect.left()));
|
|
int zrect_right (doc->zoomItX (rect.right()));
|
|
int zrect_top (doc->zoomItY (rect.top()));
|
|
int zrect_bottom (doc->zoomItY (rect.bottom()));
|
|
int zcellRect_left (doc->zoomItX (cellRect.left()));
|
|
int zcellRect_right (doc->zoomItX (cellRect.right()));
|
|
int zcellRect_top (doc->zoomItY (cellRect.top()));
|
|
int zcellRect_bottom (doc->zoomItY (cellRect.bottom()));
|
|
|
|
/* we might not paint some borders if this cell is merged with another in
|
|
that direction
|
|
bool paintLeft = paintBorderLeft;
|
|
bool paintRight = paintBorderRight;
|
|
bool paintTop = paintBorderTop;
|
|
bool paintBottom = paintBorderBottom;
|
|
*/
|
|
|
|
// paintRight = paintRight && ( extraXCells() == 0 );
|
|
// paintBottom = paintBottom && ( d->extra()->extraYCells() == 0 );
|
|
|
|
if (d->hasExtra()) {
|
|
TQValueList<Cell*>::const_iterator it = d->extra()->obscuringCells.begin();
|
|
TQValueList<Cell*>::const_iterator end = d->extra()->obscuringCells.end();
|
|
for ( ; it != end; ++it ) {
|
|
Cell* cell = *it;
|
|
|
|
int xDiff = cellRef.x() - cell->column();
|
|
int yDiff = cellRef.y() - cell->row();
|
|
paintLeft = paintLeft && xDiff == 0;
|
|
paintTop = paintTop && yDiff == 0;
|
|
|
|
// Paint the border(s) if either this one should or if we have a
|
|
// merged cell with this cell as its border.
|
|
paintRight = paintRight && cell->mergedXCells() == xDiff;
|
|
paintBottom = paintBottom && cell->mergedYCells() == yDiff;
|
|
}
|
|
}
|
|
|
|
// Must create copies of these since otherwise the zoomIt()
|
|
// operation will be performed on them repeatedly.
|
|
TQPen leftPen( _leftPen );
|
|
TQPen rightPen( _rightPen );
|
|
TQPen topPen( _topPen );
|
|
TQPen bottomPen( _bottomPen );
|
|
|
|
// Determine the pens that should be used for drawing
|
|
// the borders.
|
|
//
|
|
int left_penWidth = TQMAX( 1, doc->zoomItX( leftPen.width() ) );
|
|
int right_penWidth = TQMAX( 1, doc->zoomItX( rightPen.width() ) );
|
|
int top_penWidth = TQMAX( 1, doc->zoomItY( topPen.width() ) );
|
|
int bottom_penWidth = TQMAX( 1, doc->zoomItY( bottomPen.width() ) );
|
|
|
|
leftPen.setWidth( left_penWidth );
|
|
rightPen.setWidth( right_penWidth );
|
|
topPen.setWidth( top_penWidth );
|
|
bottomPen.setWidth( bottom_penWidth );
|
|
|
|
if ( paintLeft && leftPen.style() != TQt::NoPen ) {
|
|
int top = ( TQMAX( 0, -1 + top_penWidth ) ) / 2 +
|
|
( ( TQMAX( 0, -1 + top_penWidth ) ) % 2 );
|
|
int bottom = ( TQMAX( 0, -1 + bottom_penWidth ) ) / 2 + 1;
|
|
|
|
painter.setPen( leftPen );
|
|
|
|
//kdDebug(36001) << " painting left border of cell " << name() << endl;
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
// FIXME: There is probably Cut&Paste bugs here as well as below.
|
|
// The TQMIN/TQMAX and left/right pairs don't really make sense.
|
|
//
|
|
// UPDATE: In fact, most of these TQMIN/TQMAX combinations
|
|
// are TOTALLY BOGUS. For one thing, the idea
|
|
// that we always have full cells on paper is wrong
|
|
// since we can have embedded sheets in e.g. kword,
|
|
// and those can be arbitrarily clipped. WE HAVE TO
|
|
// REVISE THIS WHOLE BORDER PAINTING SECTION!
|
|
//
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( TQMIN( zrect_right, zcellRect_right ),
|
|
TQMAX( zrect_top, zcellRect_top - top ),
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom + bottom ) );
|
|
else
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
TQMAX( zrect_top, zcellRect_top - top ),
|
|
TQMAX( zrect_left, zcellRect_left ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom + bottom ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_right,
|
|
zcellRect_top - top,
|
|
zcellRect_right,
|
|
zcellRect_bottom + bottom );
|
|
else
|
|
painter.drawLine( zcellRect_left,
|
|
zcellRect_top - top,
|
|
zcellRect_left,
|
|
zcellRect_bottom + bottom );
|
|
}
|
|
}
|
|
|
|
if ( paintRight && rightPen.style() != TQt::NoPen ) {
|
|
int top = ( TQMAX( 0, -1 + top_penWidth ) ) / 2 +
|
|
( ( TQMAX( 0, -1 + top_penWidth ) ) % 2 );
|
|
int bottom = ( TQMAX( 0, -1 + bottom_penWidth ) ) / 2 + 1;
|
|
|
|
painter.setPen( rightPen );
|
|
|
|
//kdDebug(36001) << " painting right border of cell " << name() << endl;
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
TQMAX( zrect_top, zcellRect_top - top ),
|
|
TQMAX( zrect_left, zcellRect_left ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom + bottom ) );
|
|
else {
|
|
// FIXME: This is the way all these things should look.
|
|
// Make it so.
|
|
//
|
|
// Only print the right border if it is visible.
|
|
if ( zcellRect_right <= zrect_right + right_penWidth / 2)
|
|
painter.drawLine( zcellRect_right,
|
|
TQMAX( zrect_top, zcellRect_top - top ),
|
|
zcellRect_right,
|
|
TQMIN( zrect_bottom, zcellRect_bottom + bottom ) );
|
|
}
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_left,
|
|
zcellRect_top - top,
|
|
zcellRect_left,
|
|
zcellRect_bottom + bottom );
|
|
else
|
|
painter.drawLine( zcellRect_right,
|
|
zcellRect_top - top,
|
|
zcellRect_right,
|
|
zcellRect_bottom + bottom );
|
|
}
|
|
}
|
|
|
|
if ( paintTop && topPen.style() != TQt::NoPen ) {
|
|
painter.setPen( topPen );
|
|
|
|
//kdDebug(36001) << " painting top border of cell " << name()
|
|
// << " [" << zcellRect_left << "," << zcellRect_right
|
|
// << ": " << zcellRect_right - zcellRect_left << "]" << endl;
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( zcellRect_top >= zrect_top + top_penWidth / 2)
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
zcellRect_top,
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
zcellRect_top );
|
|
}
|
|
else {
|
|
painter.drawLine( zcellRect_left, zcellRect_top,
|
|
zcellRect_right, zcellRect_top );
|
|
}
|
|
}
|
|
|
|
if ( paintBottom && bottomPen.style() != TQt::NoPen ) {
|
|
painter.setPen( bottomPen );
|
|
|
|
//kdDebug(36001) << " painting bottom border of cell " << name()
|
|
// << " [" << zcellRect_left << "," << zcellRect_right
|
|
// << ": " << zcellRect_right - zcellRect_left << "]" << endl;
|
|
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( zcellRect_bottom <= zrect_bottom + bottom_penWidth / 2)
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
zcellRect_bottom,
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
zcellRect_bottom );
|
|
}
|
|
else {
|
|
painter.drawLine( zcellRect_left, zcellRect_bottom,
|
|
zcellRect_right, zcellRect_bottom );
|
|
}
|
|
}
|
|
|
|
// FIXME: Look very closely at when the following code is really needed.
|
|
// I can't really see any case, but I might be wrong.
|
|
// Since the code below is buggy, and incredibly complex,
|
|
// I am currently disabling it. If somebody wants to enable
|
|
// it again, then please also solve bug 68977: "Embedded KSpread
|
|
// document printing problem" at the same time.
|
|
return;
|
|
|
|
#if 0
|
|
// Look at the cells on our corners. It may happen that we
|
|
// just erased parts of their borders corner, so we might need
|
|
// to repaint these corners.
|
|
//
|
|
TQPen vert_pen, horz_pen;
|
|
int vert_penWidth, horz_penWidth;
|
|
|
|
// Some useful referenses.
|
|
Cell *cell_north = format()->sheet()->cellAt( cellRef.x(),
|
|
cellRef.y() - 1 );
|
|
Cell *cell_northwest = format()->sheet()->cellAt( cellRef.x() - 1,
|
|
cellRef.y() - 1 );
|
|
Cell *cell_west = format()->sheet()->cellAt( cellRef.x() - 1,
|
|
cellRef.y() );
|
|
Cell *cell_northeast = format()->sheet()->cellAt( cellRef.x() + 1,
|
|
cellRef.y() - 1 );
|
|
Cell *cell_east = format()->sheet()->cellAt( cellRef.x() + 1,
|
|
cellRef.y() );
|
|
Cell *cell_south = format()->sheet()->cellAt( cellRef.x(),
|
|
cellRef.y() + 1 );
|
|
Cell *cell_southwest = format()->sheet()->cellAt( cellRef.x() - 1,
|
|
cellRef.y() + 1 );
|
|
Cell *cell_southeast = format()->sheet()->cellAt( cellRef.x() + 1,
|
|
cellRef.y() + 1 );
|
|
|
|
// Fix the borders which meet at the top left corner
|
|
if ( cell_north->effLeftBorderValue( cellRef.x(), cellRef.y() - 1 )
|
|
>= cell_northwest->effRightBorderValue( cellRef.x() - 1, cellRef.y() - 1 ) )
|
|
vert_pen = cell_north->effLeftBorderPen( cellRef.x(), cellRef.y() - 1 );
|
|
else
|
|
vert_pen = cell_northwest->effRightBorderPen( cellRef.x() - 1,
|
|
cellRef.y() - 1 );
|
|
|
|
vert_penWidth = TQMAX( 1, doc->zoomItX( vert_pen.width() ) );
|
|
vert_pen.setWidth( vert_penWidth );
|
|
|
|
if ( vert_pen.style() != TQt::NoPen ) {
|
|
if ( cell_west->effTopBorderValue( cellRef.x() - 1, cellRef.y() )
|
|
>= cell_northwest->effBottomBorderValue( cellRef.x() - 1, cellRef.y() - 1 ) )
|
|
horz_pen = cell_west->effTopBorderPen( cellRef.x() - 1, cellRef.y() );
|
|
else
|
|
horz_pen = cell_northwest->effBottomBorderPen( cellRef.x() - 1,
|
|
cellRef.y() - 1 );
|
|
|
|
horz_penWidth = TQMAX( 1, doc->zoomItY( horz_pen.width() ) );
|
|
int bottom = ( TQMAX( 0, -1 + horz_penWidth ) ) / 2 + 1;
|
|
|
|
painter.setPen( vert_pen );
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_right ),
|
|
TQMAX( zrect_top, zcellRect_top ),
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
TQMIN( zrect_bottom, zcellRect_top + bottom ) );
|
|
else
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
TQMAX( zrect_top, zcellRect_top ),
|
|
TQMIN( zrect_right, zcellRect_left ),
|
|
TQMIN( zrect_bottom, zcellRect_top + bottom ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_right, zcellRect_top,
|
|
zcellRect_right, zcellRect_top + bottom );
|
|
else
|
|
painter.drawLine( zcellRect_left, zcellRect_top,
|
|
zcellRect_left, zcellRect_top + bottom );
|
|
}
|
|
}
|
|
|
|
// Fix the borders which meet at the top right corner
|
|
if ( cell_north->effRightBorderValue( cellRef.x(), cellRef.y() - 1 )
|
|
>= cell_northeast->effLeftBorderValue( cellRef.x() + 1,
|
|
cellRef.y() - 1 ) )
|
|
vert_pen = cell_north->effRightBorderPen( cellRef.x(), cellRef.y() - 1 );
|
|
else
|
|
vert_pen = cell_northeast->effLeftBorderPen( cellRef.x() + 1,
|
|
cellRef.y() - 1 );
|
|
|
|
// vert_pen = effRightBorderPen( cellRef.x(), cellRef.y() - 1 );
|
|
vert_penWidth = TQMAX( 1, doc->zoomItX( vert_pen.width() ) );
|
|
vert_pen.setWidth( vert_penWidth );
|
|
if ( ( vert_pen.style() != TQt::NoPen ) && ( cellRef.x() < KS_colMax ) ) {
|
|
if ( cell_east->effTopBorderValue( cellRef.x() + 1, cellRef.y() )
|
|
>= cell_northeast->effBottomBorderValue( cellRef.x() + 1,
|
|
cellRef.y() - 1 ) )
|
|
horz_pen = cell_east->effTopBorderPen( cellRef.x() + 1, cellRef.y() );
|
|
else
|
|
horz_pen = cell_northeast->effBottomBorderPen( cellRef.x() + 1,
|
|
cellRef.y() - 1 );
|
|
|
|
// horz_pen = effTopBorderPen( cellRef.x() + 1, cellRef.y() );
|
|
horz_penWidth = TQMAX( 1, doc->zoomItY( horz_pen.width() ) );
|
|
int bottom = ( TQMAX( 0, -1 + horz_penWidth ) ) / 2 + 1;
|
|
|
|
painter.setPen( vert_pen );
|
|
//If we are on paper printout, we limit the length of the lines
|
|
//On paper, we always have full cells, on screen not
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
TQMAX( zrect_top, zcellRect_top ),
|
|
TQMIN( zrect_right, zcellRect_left ),
|
|
TQMIN( zrect_bottom, zcellRect_top + bottom ) );
|
|
else
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_right ),
|
|
TQMAX( zrect_top, zcellRect_top ),
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
TQMIN( zrect_bottom, zcellRect_top + bottom ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_left, zcellRect_top,
|
|
zcellRect_left, zcellRect_top + bottom );
|
|
else
|
|
painter.drawLine( zcellRect_right, zcellRect_top,
|
|
zcellRect_right, zcellRect_top + bottom );
|
|
}
|
|
}
|
|
|
|
// Bottom
|
|
if ( cellRef.y() < KS_rowMax ) {
|
|
// Fix the borders which meet at the bottom left corner
|
|
if ( cell_south->effLeftBorderValue( cellRef.x(), cellRef.y() + 1 )
|
|
>= cell_southwest->effRightBorderValue( cellRef.x() - 1,
|
|
cellRef.y() + 1 ) )
|
|
vert_pen = cell_south->effLeftBorderPen( cellRef.x(), cellRef.y() + 1 );
|
|
else
|
|
vert_pen = cell_southwest->effRightBorderPen( cellRef.x() - 1,
|
|
cellRef.y() + 1 );
|
|
|
|
// vert_pen = effLeftBorderPen( cellRef.x(), cellRef.y() + 1 );
|
|
vert_penWidth = TQMAX( 1, doc->zoomItY( vert_pen.width() ) );
|
|
vert_pen.setWidth( vert_penWidth );
|
|
if ( vert_pen.style() != TQt::NoPen ) {
|
|
if ( cell_west->effBottomBorderValue( cellRef.x() - 1, cellRef.y() )
|
|
>= cell_southwest->effTopBorderValue( cellRef.x() - 1,
|
|
cellRef.y() + 1 ) )
|
|
horz_pen = cell_west->effBottomBorderPen( cellRef.x() - 1,
|
|
cellRef.y() );
|
|
else
|
|
horz_pen = cell_southwest->effTopBorderPen( cellRef.x() - 1,
|
|
cellRef.y() + 1 );
|
|
|
|
// horz_pen = effBottomBorderPen( cellRef.x() - 1, cellRef.y() );
|
|
horz_penWidth = TQMAX( 1, doc->zoomItX( horz_pen.width() ) );
|
|
int bottom = ( TQMAX( 0, -1 + horz_penWidth ) ) / 2;
|
|
|
|
painter.setPen( vert_pen );
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_right ),
|
|
TQMAX( zrect_top, zcellRect_bottom - bottom ),
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom ) );
|
|
else
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
TQMAX( zrect_top, zcellRect_bottom - bottom ),
|
|
TQMIN( zrect_right, zcellRect_left ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_right, zcellRect_bottom - bottom,
|
|
zcellRect_right, zcellRect_bottom );
|
|
else
|
|
painter.drawLine( zcellRect_left, zcellRect_bottom - bottom,
|
|
zcellRect_left, zcellRect_bottom );
|
|
}
|
|
}
|
|
|
|
// Fix the borders which meet at the bottom right corner
|
|
if ( cell_south->effRightBorderValue( cellRef.x(), cellRef.y() + 1 )
|
|
>= cell_southeast->effLeftBorderValue( cellRef.x() + 1,
|
|
cellRef.y() + 1 ) )
|
|
vert_pen = cell_south->effRightBorderPen( cellRef.x(), cellRef.y() + 1 );
|
|
else
|
|
vert_pen = cell_southeast->effLeftBorderPen( cellRef.x() + 1,
|
|
cellRef.y() + 1 );
|
|
|
|
// vert_pen = effRightBorderPen( cellRef.x(), cellRef.y() + 1 );
|
|
vert_penWidth = TQMAX( 1, doc->zoomItY( vert_pen.width() ) );
|
|
vert_pen.setWidth( vert_penWidth );
|
|
if ( ( vert_pen.style() != TQt::NoPen ) && ( cellRef.x() < KS_colMax ) ) {
|
|
if ( cell_east ->effBottomBorderValue( cellRef.x() + 1, cellRef.y() )
|
|
>= cell_southeast->effTopBorderValue( cellRef.x() + 1,
|
|
cellRef.y() + 1 ) )
|
|
|
|
horz_pen = format()->sheet()->cellAt( cellRef.x() + 1, cellRef.y() )
|
|
->effBottomBorderPen( cellRef.x() + 1, cellRef.y() );
|
|
else
|
|
horz_pen = format()->sheet()->cellAt( cellRef.x() + 1, cellRef.y() + 1 )
|
|
->effTopBorderPen( cellRef.x() + 1, cellRef.y() + 1 );
|
|
|
|
// horz_pen = effBottomBorderPen( cellRef.x() + 1, cellRef.y() );
|
|
horz_penWidth = TQMAX( 1, doc->zoomItX( horz_pen.width() ) );
|
|
int bottom = ( TQMAX( 0, -1 + horz_penWidth ) ) / 2;
|
|
|
|
painter.setPen( vert_pen );
|
|
// If we are on paper printout, we limit the length of the lines.
|
|
// On paper, we always have full cells, on screen not.
|
|
if ( painter.device()->isExtDev() ) {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_left ),
|
|
TQMAX( zrect_top, zcellRect_bottom - bottom ),
|
|
TQMIN( zrect_right, zcellRect_left ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom ) );
|
|
else
|
|
painter.drawLine( TQMAX( zrect_left, zcellRect_right ),
|
|
TQMAX( zrect_top, zcellRect_bottom - bottom ),
|
|
TQMIN( zrect_right, zcellRect_right ),
|
|
TQMIN( zrect_bottom, zcellRect_bottom ) );
|
|
}
|
|
else {
|
|
if ( sheetDir == Sheet::RightToLeft )
|
|
painter.drawLine( zcellRect_left, zcellRect_bottom - bottom,
|
|
zcellRect_left, zcellRect_bottom );
|
|
else
|
|
painter.drawLine( zcellRect_right, zcellRect_bottom - bottom,
|
|
zcellRect_right, zcellRect_bottom );
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
// Paint diagonal lines through the cell.
|
|
//
|
|
void Cell::paintCellDiagonalLines( TQPainter& painter,
|
|
const KoRect &cellRect,
|
|
const TQPoint &cellRef )
|
|
{
|
|
if ( isPartOfMerged() )
|
|
return;
|
|
|
|
Doc* doc = sheet()->doc();
|
|
|
|
if ( effFallDiagonalPen( cellRef.x(), cellRef.y() ).style() != TQt::NoPen ) {
|
|
painter.setPen( effFallDiagonalPen( cellRef.x(), cellRef.y() ) );
|
|
painter.drawLine( doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.y() ),
|
|
doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.bottom() ) );
|
|
}
|
|
|
|
if ( effGoUpDiagonalPen( cellRef.x(), cellRef.y() ).style() != TQt::NoPen ) {
|
|
painter.setPen( effGoUpDiagonalPen( cellRef.x(), cellRef.y() ) );
|
|
painter.drawLine( doc->zoomItX( cellRect.x() ),
|
|
doc->zoomItY( cellRect.bottom() ),
|
|
doc->zoomItX( cellRect.right() ),
|
|
doc->zoomItY( cellRect.y() ) );
|
|
}
|
|
}
|
|
|
|
|
|
// End of Painting
|
|
// ================================================================
|
|
|
|
|
|
int Cell::defineAlignX()
|
|
{
|
|
int a = format()->align( column(), row() );
|
|
if ( a == Format::Undefined )
|
|
{
|
|
//numbers should be right-aligned by default, as well as BiDi text
|
|
if ((formatType() == Text_format) || value().isString())
|
|
a = (d->strOutText.isRightToLeft()) ?
|
|
Format::Right : Format::Left;
|
|
else {
|
|
Value val = value();
|
|
while (val.isArray()) val = val.element (0, 0);
|
|
if (val.isBoolean() || val.isNumber())
|
|
a = Format::Right;
|
|
else
|
|
a = Format::Left;
|
|
}
|
|
}
|
|
return a;
|
|
}
|
|
|
|
int Cell::effAlignX()
|
|
{
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SAlignX, true ) )
|
|
return d->extra()->conditions->matchedStyle()->alignX();
|
|
|
|
return defineAlignX();
|
|
}
|
|
|
|
// Cut strOutText, so that it only holds the part that can be displayed.
|
|
//
|
|
// Used in paintText().
|
|
//
|
|
|
|
TQString Cell::textDisplaying( TQPainter &_painter )
|
|
{
|
|
TQFontMetrics fm = _painter.fontMetrics();
|
|
int a = format()->align( column(), row() );
|
|
|
|
bool isNumeric = value().isNumber();
|
|
|
|
if ( !format()->verticalText( column(),row() ) ) {
|
|
// Non-vertical text: the ordinary case.
|
|
|
|
// Not enough space but align to left
|
|
double len = 0.0;
|
|
int extraXCells = d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
|
|
for ( int i = column(); i <= column() + extraXCells; i++ ) {
|
|
ColumnFormat *cl2 = format()->sheet()->columnFormat( i );
|
|
len += cl2->dblWidth() - 1.0; //-1.0 because the pixel in between 2 cells is shared between both cells
|
|
}
|
|
|
|
TQString tmp;
|
|
double tmpIndent = 0.0;
|
|
if ( !isEmpty() )
|
|
tmpIndent = format()->getIndent( column(), row() );
|
|
|
|
// Start out with the whole text, cut one character at a time, and
|
|
// when the text finally fits, return it.
|
|
for ( int i = d->strOutText.length(); i != 0; i-- )
|
|
{
|
|
//Note that numbers are always treated as left-aligned since if we have to cut digits off, they should
|
|
//always be the least significant ones at the end of the string
|
|
if ( a == Format::Left || a == Format::Undefined || isNumeric)
|
|
tmp = d->strOutText.left(i);
|
|
else if ( a == Format::Right)
|
|
tmp = d->strOutText.right(i);
|
|
else
|
|
tmp = d->strOutText.mid( ( d->strOutText.length() - i ) / 2, i);
|
|
|
|
if (isNumeric)
|
|
{
|
|
//For numeric values, we can cut off digits after the decimal point to make it fit,
|
|
//but not the integer part of the number.
|
|
//If this number still contains a fraction part then we don't need to do anything, if we have run
|
|
//out of space to fit even the integer part of the number then display #########
|
|
//TODO Perhaps try to display integer part in standard form if there is not enough room for it?
|
|
|
|
if (!tmp.contains('.'))
|
|
d->strOutText=TQString().fill('#',20);
|
|
}
|
|
|
|
// 4 equal length of red triangle +1 point.
|
|
if ( format()->sheet()->doc()->unzoomItX( fm.width( tmp ) ) + tmpIndent
|
|
< len - 4.0 - 1.0 )
|
|
{
|
|
if ( format()->getAngle( column(), row() ) != 0 )
|
|
{
|
|
TQString tmp2;
|
|
RowFormat *rl = format()->sheet()->rowFormat( row() );
|
|
if ( d->textHeight > rl->dblHeight() )
|
|
{
|
|
for ( int j = d->strOutText.length(); j != 0; j-- )
|
|
{
|
|
tmp2 = d->strOutText.left( j );
|
|
if ( format()->sheet()->doc()->unzoomItY( fm.width( tmp2 ) ) < rl->dblHeight() - 1.0 )
|
|
{
|
|
return d->strOutText.left( TQMIN( tmp.length(), tmp2.length() ) );
|
|
}
|
|
}
|
|
}
|
|
else
|
|
return tmp;
|
|
|
|
}
|
|
else
|
|
return tmp;
|
|
}
|
|
}
|
|
return TQString( "" );
|
|
}
|
|
else if ( format()->verticalText( column(), row() ) ) {
|
|
// Vertical text.
|
|
|
|
RowFormat *rl = format()->sheet()->rowFormat( row() );
|
|
double tmpIndent = 0.0;
|
|
|
|
// Not enough space but align to left.
|
|
double len = 0.0;
|
|
int extraXCells = d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
|
|
for ( int i = column(); i <= column() + extraXCells; i++ ) {
|
|
ColumnFormat *cl2 = format()->sheet()->columnFormat( i );
|
|
|
|
// -1.0 because the pixel in between 2 cells is shared between both cells
|
|
len += cl2->dblWidth() - 1.0;
|
|
}
|
|
|
|
if ( !isEmpty() )
|
|
tmpIndent = format()->getIndent( column(), row() );
|
|
|
|
if ( ( d->textWidth + tmpIndent > len ) || d->textWidth == 0.0 )
|
|
return TQString( "" );
|
|
|
|
for ( int i = d->strOutText.length(); i != 0; i-- ) {
|
|
if ( format()->sheet()->doc()->unzoomItY( fm.ascent() + fm.descent() ) * i
|
|
< rl->dblHeight() - 1.0 )
|
|
return d->strOutText.left( i );
|
|
}
|
|
|
|
return TQString( "" );
|
|
}
|
|
|
|
ColumnFormat *cl = format()->sheet()->columnFormat( column() );
|
|
double w = cl->dblWidth();
|
|
|
|
if ( d->hasExtra() && (d->extra()->extraWidth != 0.0) )
|
|
w = d->extra()->extraWidth;
|
|
|
|
TQString tmp;
|
|
for ( int i = d->strOutText.length(); i != 0; i-- ) {
|
|
tmp = d->strOutText.left( i );
|
|
|
|
// 4 equals length of red triangle +1 pixel
|
|
if ( format()->sheet()->doc()->unzoomItX( fm.width( tmp ) ) < w - 4.0 - 1.0 )
|
|
return tmp;
|
|
}
|
|
|
|
return TQString();
|
|
}
|
|
|
|
|
|
double Cell::dblWidth( int _col, const Canvas *_canvas ) const
|
|
{
|
|
if ( _col < 0 )
|
|
_col = d->column;
|
|
|
|
if ( _canvas )
|
|
{
|
|
if ( testFlag(Flag_Merged) )
|
|
return d->extra()->extraWidth;
|
|
|
|
const ColumnFormat *cl = format()->sheet()->columnFormat( _col );
|
|
return cl->dblWidth( _canvas );
|
|
}
|
|
|
|
if ( testFlag(Flag_Merged) )
|
|
return d->extra()->extraWidth;
|
|
|
|
const ColumnFormat *cl = format()->sheet()->columnFormat( _col );
|
|
return cl->dblWidth();
|
|
}
|
|
|
|
int Cell::width( int _col, const Canvas *_canvas ) const
|
|
{
|
|
return int( dblWidth( _col, _canvas ) );
|
|
}
|
|
|
|
double Cell::dblHeight( int _row, const Canvas *_canvas ) const
|
|
{
|
|
if ( _row < 0 )
|
|
_row = d->row;
|
|
|
|
if ( _canvas )
|
|
{
|
|
if ( testFlag(Flag_Merged) )
|
|
return d->extra()->extraHeight;
|
|
|
|
const RowFormat *rl = format()->sheet()->rowFormat( _row );
|
|
return rl->dblHeight( _canvas );
|
|
}
|
|
|
|
if ( testFlag(Flag_Merged) )
|
|
return d->extra()->extraHeight;
|
|
|
|
const RowFormat *rl = format()->sheet()->rowFormat( _row );
|
|
return rl->dblHeight();
|
|
}
|
|
|
|
int Cell::height( int _row, const Canvas *_canvas ) const
|
|
{
|
|
return int( dblHeight( _row, _canvas ) );
|
|
}
|
|
|
|
///////////////////////////////////////////
|
|
//
|
|
// Misc Properties.
|
|
// Reimplementation of Format methods.
|
|
//
|
|
///////////////////////////////////////////
|
|
|
|
const TQBrush& Cell::backGroundBrush( int _col, int _row ) const
|
|
{
|
|
if ( d->hasExtra() && (!d->extra()->obscuringCells.isEmpty()) )
|
|
{
|
|
const Cell* cell = d->extra()->obscuringCells.first();
|
|
return cell->backGroundBrush( cell->column(), cell->row() );
|
|
}
|
|
|
|
return format()->backGroundBrush( _col, _row );
|
|
}
|
|
|
|
const TQColor& Cell::bgColor( int _col, int _row ) const
|
|
{
|
|
if ( d->hasExtra() && (!d->extra()->obscuringCells.isEmpty()) )
|
|
{
|
|
const Cell* cell = d->extra()->obscuringCells.first();
|
|
return cell->bgColor( cell->column(), cell->row() );
|
|
}
|
|
|
|
return format()->bgColor( _col, _row );
|
|
}
|
|
|
|
///////////////////////////////////////////
|
|
//
|
|
// Borders.
|
|
// Reimplementation of Format methods.
|
|
//
|
|
///////////////////////////////////////////
|
|
|
|
void Cell::setLeftBorderPen( const TQPen& p )
|
|
{
|
|
if ( column() == 1 )
|
|
{
|
|
Cell* cell = format()->sheet()->cellAt( column() - 1, row() );
|
|
if ( cell && cell->format()->hasProperty( Format::PRightBorder )
|
|
&& format()->sheet()->cellAt( column(), row() ) == this )
|
|
cell->format()->clearProperty( Format::PRightBorder );
|
|
}
|
|
|
|
format()->setLeftBorderPen( p );
|
|
}
|
|
|
|
void Cell::setTopBorderPen( const TQPen& p )
|
|
{
|
|
if ( row() == 1 )
|
|
{
|
|
Cell* cell = format()->sheet()->cellAt( column(), row() - 1 );
|
|
if ( cell && cell->format()->hasProperty( Format::PBottomBorder )
|
|
&& format()->sheet()->cellAt( column(), row() ) == this )
|
|
cell->format()->clearProperty( Format::PBottomBorder );
|
|
}
|
|
format()->setTopBorderPen( p );
|
|
}
|
|
|
|
void Cell::setRightBorderPen( const TQPen& p )
|
|
{
|
|
Cell* cell = 0L;
|
|
if ( column() < KS_colMax )
|
|
cell = format()->sheet()->cellAt( column() + 1, row() );
|
|
|
|
if ( cell && cell->format()->hasProperty( Format::PLeftBorder )
|
|
&& format()->sheet()->cellAt( column(), row() ) == this )
|
|
cell->format()->clearProperty( Format::PLeftBorder );
|
|
|
|
format()->setRightBorderPen( p );
|
|
}
|
|
|
|
void Cell::setBottomBorderPen( const TQPen& p )
|
|
{
|
|
Cell* cell = 0L;
|
|
if ( row() < KS_rowMax )
|
|
cell = format()->sheet()->cellAt( column(), row() + 1 );
|
|
|
|
if ( cell && cell->format()->hasProperty( Format::PTopBorder )
|
|
&& format()->sheet()->cellAt( column(), row() ) == this )
|
|
cell->format()->clearProperty( Format::PTopBorder );
|
|
|
|
format()->setBottomBorderPen( p );
|
|
}
|
|
|
|
const TQPen& Cell::rightBorderPen( int _col, int _row ) const
|
|
{
|
|
if ( !format()->hasProperty( Format::PRightBorder ) && ( _col < KS_colMax ) )
|
|
{
|
|
Cell * cell = format()->sheet()->cellAt( _col + 1, _row );
|
|
if ( cell && cell->format()->hasProperty( Format::PLeftBorder ) )
|
|
return cell->leftBorderPen( _col + 1, _row );
|
|
}
|
|
|
|
return format()->rightBorderPen( _col, _row );
|
|
}
|
|
|
|
const TQPen& Cell::leftBorderPen( int _col, int _row ) const
|
|
{
|
|
if ( !format()->hasProperty( Format::PLeftBorder ) )
|
|
{
|
|
const Cell * cell = format()->sheet()->cellAt( _col - 1, _row );
|
|
if ( cell && cell->format()->hasProperty( Format::PRightBorder ) )
|
|
return cell->rightBorderPen( _col - 1, _row );
|
|
}
|
|
|
|
return format()->leftBorderPen( _col, _row );
|
|
}
|
|
|
|
const TQPen& Cell::bottomBorderPen( int _col, int _row ) const
|
|
{
|
|
if ( !format()->hasProperty( Format::PBottomBorder ) && ( _row < KS_rowMax ) )
|
|
{
|
|
const Cell * cell = format()->sheet()->cellAt( _col, _row + 1 );
|
|
if ( cell && cell->format()->hasProperty( Format::PTopBorder ) )
|
|
return cell->topBorderPen( _col, _row + 1 );
|
|
}
|
|
|
|
return format()->bottomBorderPen( _col, _row );
|
|
}
|
|
|
|
const TQPen& Cell::topBorderPen( int _col, int _row ) const
|
|
{
|
|
if ( !format()->hasProperty( Format::PTopBorder ) )
|
|
{
|
|
const Cell * cell = format()->sheet()->cellAt( _col, _row - 1 );
|
|
if ( cell->format()->hasProperty( Format::PBottomBorder ) )
|
|
return cell->bottomBorderPen( _col, _row - 1 );
|
|
}
|
|
|
|
return format()->topBorderPen( _col, _row );
|
|
}
|
|
|
|
const TQColor & Cell::effTextColor( int col, int row ) const
|
|
{
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::STextPen, true ) )
|
|
return d->extra()->conditions->matchedStyle()->pen().color();
|
|
|
|
return format()->textColor( col, row );
|
|
}
|
|
|
|
const TQPen& Cell::effLeftBorderPen( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effLeftBorderPen( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SLeftBorder, true ) )
|
|
return d->extra()->conditions->matchedStyle()->leftBorderPen();
|
|
|
|
return leftBorderPen( col, row );
|
|
}
|
|
|
|
const TQPen& Cell::effTopBorderPen( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effTopBorderPen( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::STopBorder, true ) )
|
|
return d->extra()->conditions->matchedStyle()->topBorderPen();
|
|
|
|
return topBorderPen( col, row );
|
|
}
|
|
|
|
const TQPen& Cell::effRightBorderPen( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effRightBorderPen( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SRightBorder, true ) )
|
|
return d->extra()->conditions->matchedStyle()->rightBorderPen();
|
|
|
|
return rightBorderPen( col, row );
|
|
}
|
|
|
|
const TQPen& Cell::effBottomBorderPen( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effBottomBorderPen( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SBottomBorder, true ) )
|
|
return d->extra()->conditions->matchedStyle()->bottomBorderPen();
|
|
|
|
return bottomBorderPen( col, row );
|
|
}
|
|
|
|
const TQPen & Cell::effGoUpDiagonalPen( int col, int row ) const
|
|
{
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SGoUpDiagonal, true ) )
|
|
return d->extra()->conditions->matchedStyle()->goUpDiagonalPen();
|
|
|
|
return format()->goUpDiagonalPen( col, row );
|
|
}
|
|
|
|
const TQPen & Cell::effFallDiagonalPen( int col, int row ) const
|
|
{
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle()
|
|
&& d->extra()->conditions->matchedStyle()->hasFeature( Style::SFallDiagonal, true ) )
|
|
return d->extra()->conditions->matchedStyle()->fallDiagonalPen();
|
|
|
|
return format()->fallDiagonalPen( col, row );
|
|
}
|
|
|
|
uint Cell::effBottomBorderValue( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effBottomBorderValue( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
return d->extra()->conditions->matchedStyle()->bottomPenValue();
|
|
|
|
return format()->bottomBorderValue( col, row );
|
|
}
|
|
|
|
uint Cell::effRightBorderValue( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effRightBorderValue( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
return d->extra()->conditions->matchedStyle()->rightPenValue();
|
|
|
|
return format()->rightBorderValue( col, row );
|
|
}
|
|
|
|
uint Cell::effLeftBorderValue( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effLeftBorderValue( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
return d->extra()->conditions->matchedStyle()->leftPenValue();
|
|
|
|
return format()->leftBorderValue( col, row );
|
|
}
|
|
|
|
uint Cell::effTopBorderValue( int col, int row ) const
|
|
{
|
|
if ( isPartOfMerged() )
|
|
{
|
|
Cell * cell = d->extra()->obscuringCells.first();
|
|
return cell->effTopBorderValue( cell->column(), cell->row() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions
|
|
&& d->extra()->conditions->matchedStyle() )
|
|
return d->extra()->conditions->matchedStyle()->topPenValue();
|
|
|
|
return format()->topBorderValue( col, row );
|
|
}
|
|
|
|
///////////////////////////////////////////
|
|
//
|
|
// Precision
|
|
//
|
|
///////////////////////////////////////////
|
|
|
|
void Cell::incPrecision()
|
|
{
|
|
//TODO: This is ugly. Why not simply regenerate the text to display? Tomas
|
|
|
|
if ( !value().isNumber() )
|
|
return;
|
|
int tmpPreci = format()->precision( column(), row() );
|
|
|
|
if ( tmpPreci == -1 )
|
|
{
|
|
int pos = d->strOutText.find(decimal_point);
|
|
if ( pos == -1 )
|
|
pos = d->strOutText.find('.');
|
|
if ( pos == -1 )
|
|
format()->setPrecision(1);
|
|
else
|
|
{
|
|
int start = 0;
|
|
if ( d->strOutText.find('%') != -1 )
|
|
start = 2;
|
|
else if ( d->strOutText.find(locale()->currencySymbol()) == ((int)(d->strOutText.length()-locale()->currencySymbol().length())) )
|
|
start = locale()->currencySymbol().length() + 1;
|
|
else if ( (start=d->strOutText.find('E')) != -1 )
|
|
start = d->strOutText.length() - start;
|
|
|
|
//kdDebug(36001) << "start=" << start << " pos=" << pos << " length=" << d->strOutText.length() << endl;
|
|
format()->setPrecision( TQMAX( 0, (int)d->strOutText.length() - start - pos ) );
|
|
}
|
|
}
|
|
else if ( tmpPreci < 10 )
|
|
{
|
|
format()->setPrecision( ++tmpPreci );
|
|
}
|
|
setFlag(Flag_LayoutDirty);
|
|
}
|
|
|
|
void Cell::decPrecision()
|
|
{
|
|
//TODO: This is ugly. Why not simply regenerate the text to display? Tomas
|
|
|
|
if ( !value().isNumber() )
|
|
return;
|
|
int preciTmp = format()->precision( column(), row() );
|
|
// kdDebug(36001) << "decPrecision: tmpPreci = " << tmpPreci << endl;
|
|
if ( format()->precision(column(),row()) == -1 )
|
|
{
|
|
int pos = d->strOutText.find( decimal_point );
|
|
int start = 0;
|
|
if ( d->strOutText.find('%') != -1 )
|
|
start = 2;
|
|
else if ( d->strOutText.find(locale()->currencySymbol()) == ((int)(d->strOutText.length()-locale()->currencySymbol().length())) )
|
|
start = locale()->currencySymbol().length() + 1;
|
|
else if ( (start = d->strOutText.find('E')) != -1 )
|
|
start = d->strOutText.length() - start;
|
|
else
|
|
start = 0;
|
|
|
|
if ( pos == -1 )
|
|
return;
|
|
|
|
format()->setPrecision(d->strOutText.length() - pos - 2 - start);
|
|
// if ( preciTmp < 0 )
|
|
// format()->setPrecision( preciTmp );
|
|
}
|
|
else if ( preciTmp > 0 )
|
|
{
|
|
format()->setPrecision( --preciTmp );
|
|
}
|
|
setFlag( Flag_LayoutDirty );
|
|
}
|
|
|
|
//set numerical value
|
|
//used in Sheet::setSeries (nowhere else yet)
|
|
void Cell::setNumber( double number )
|
|
{
|
|
setValue( Value( number ) );
|
|
|
|
d->strText.setNum( number );
|
|
setDisplayText(d->strText);
|
|
checkNumberFormat();
|
|
}
|
|
|
|
void Cell::setCellText( const TQString& _text, bool asText )
|
|
{
|
|
// TQString ctext = _text;
|
|
|
|
// (Tomas) is this trim necessary for anything ?
|
|
// if( ctext.length() > 5000 )
|
|
// ctext = ctext.left( 5000 );
|
|
|
|
// empty string ?
|
|
if (_text.length() == 0) {
|
|
d->strOutText = d->strText = "";
|
|
setValue (Value::empty());
|
|
return;
|
|
}
|
|
|
|
// as text ?
|
|
if (asText) {
|
|
d->strOutText = _text;
|
|
d->strText = _text;
|
|
setValue (Value (_text));
|
|
|
|
return;
|
|
}
|
|
|
|
TQString oldText = d->strText;
|
|
setDisplayText( _text );
|
|
if(!format()->sheet()->isLoading() && !testValidity() )
|
|
{
|
|
//reapply old value if action == stop
|
|
setDisplayText( oldText );
|
|
}
|
|
}
|
|
|
|
void Cell::setDisplayText( const TQString& _text )
|
|
{
|
|
bool isLoading = format()->sheet()->isLoading();
|
|
|
|
if (!isLoading)
|
|
format()->sheet()->doc()->emitBeginOperation( false );
|
|
|
|
d->strText = _text;
|
|
|
|
/**
|
|
* A real formula "=A1+A2*3" was entered.
|
|
*/
|
|
if ( !d->strText.isEmpty() && d->strText[0] == '=' )
|
|
{
|
|
setFlag(Flag_LayoutDirty);
|
|
setFlag(Flag_TextFormatDirty);
|
|
|
|
if ( !makeFormula() )
|
|
kdError(36001) << "ERROR: Syntax ERROR" << endl;
|
|
setCalcDirtyFlag ();
|
|
}
|
|
|
|
/**
|
|
* Some numeric value or a string.
|
|
*/
|
|
else
|
|
{
|
|
// Find out what data type it is
|
|
checkTextInput();
|
|
|
|
setFlag(Flag_LayoutDirty);
|
|
setFlag(Flag_TextFormatDirty);
|
|
}
|
|
|
|
if ( !isLoading )
|
|
format()->sheet()->doc()->emitEndOperation( TQRect( d->column, d->row, 1, 1 ) );
|
|
}
|
|
|
|
void Cell::setLink( const TQString& link )
|
|
{
|
|
d->extra()->link = link;
|
|
|
|
if( !link.isEmpty() && d->strText.isEmpty() )
|
|
setCellText( link );
|
|
}
|
|
|
|
TQString Cell::link() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->link : TQString();
|
|
}
|
|
|
|
void Cell::update()
|
|
{
|
|
/* those obscuring us need to redo their layout cause they can't obscure us
|
|
now that we've got text.
|
|
This includes cells obscuring cells that we are obscuring
|
|
*/
|
|
for (int x = d->column; x <= d->column + extraXCells(); x++)
|
|
{
|
|
for (int y = d->row; y <= d->row + extraYCells(); y++)
|
|
{
|
|
Cell* cell = format()->sheet()->cellAt(x,y);
|
|
cell->setLayoutDirtyFlag();
|
|
}
|
|
}
|
|
|
|
setCalcDirtyFlag();
|
|
|
|
/* TODO - is this a good place for this? */
|
|
updateChart(true);
|
|
}
|
|
|
|
bool Cell::testValidity() const
|
|
{
|
|
bool valid = false;
|
|
if( d->hasExtra() && d->extra()->validity && d->extra()->validity->m_restriction != Restriction::None )
|
|
{
|
|
//fixme
|
|
if ( d->extra()->validity->allowEmptyCell && d->strText.isEmpty() )
|
|
return true;
|
|
|
|
if( value().isNumber() &&
|
|
(d->extra()->validity->m_restriction == Restriction::Number ||
|
|
(d->extra()->validity->m_restriction == Restriction::Integer &&
|
|
value().asFloat() == ceil(value().asFloat()))))
|
|
{
|
|
switch( d->extra()->validity->m_cond)
|
|
{
|
|
case Conditional::Equal:
|
|
valid = ( value().asFloat() - d->extra()->validity->valMin < DBL_EPSILON
|
|
&& value().asFloat() - d->extra()->validity->valMin >
|
|
(0.0 - DBL_EPSILON));
|
|
break;
|
|
case Conditional::DifferentTo:
|
|
valid = !( ( value().asFloat() - d->extra()->validity->valMin < DBL_EPSILON
|
|
&& value().asFloat() - d->extra()->validity->valMin >
|
|
(0.0 - DBL_EPSILON)) );
|
|
break;
|
|
case Conditional::Superior:
|
|
valid = ( value().asFloat() > d->extra()->validity->valMin);
|
|
break;
|
|
case Conditional::Inferior:
|
|
valid = ( value().asFloat() <d->extra()->validity->valMin);
|
|
break;
|
|
case Conditional::SuperiorEqual:
|
|
valid = ( value().asFloat() >= d->extra()->validity->valMin);
|
|
break;
|
|
case Conditional::InferiorEqual:
|
|
valid = (value().asFloat() <= d->extra()->validity->valMin);
|
|
break;
|
|
case Conditional::Between:
|
|
valid = ( value().asFloat() >= d->extra()->validity->valMin &&
|
|
value().asFloat() <= d->extra()->validity->valMax);
|
|
break;
|
|
case Conditional::Different:
|
|
valid = (value().asFloat() < d->extra()->validity->valMin ||
|
|
value().asFloat() > d->extra()->validity->valMax);
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
else if(d->extra()->validity->m_restriction==Restriction::Text)
|
|
{
|
|
valid = value().isString();
|
|
}
|
|
else if ( d->extra()->validity->m_restriction == Restriction::List )
|
|
{
|
|
//test int value
|
|
if ( value().isString() && d->extra()->validity->listValidity.contains( value().asString() ) )
|
|
valid = true;
|
|
}
|
|
else if(d->extra()->validity->m_restriction==Restriction::TextLength)
|
|
{
|
|
if( value().isString() )
|
|
{
|
|
int len = d->strOutText.length();
|
|
switch( d->extra()->validity->m_cond)
|
|
{
|
|
case Conditional::Equal:
|
|
if (len == d->extra()->validity->valMin)
|
|
valid = true;
|
|
break;
|
|
case Conditional::DifferentTo:
|
|
if (len != d->extra()->validity->valMin)
|
|
valid = true;
|
|
break;
|
|
case Conditional::Superior:
|
|
if(len > d->extra()->validity->valMin)
|
|
valid = true;
|
|
break;
|
|
case Conditional::Inferior:
|
|
if(len < d->extra()->validity->valMin)
|
|
valid = true;
|
|
break;
|
|
case Conditional::SuperiorEqual:
|
|
if(len >= d->extra()->validity->valMin)
|
|
valid = true;
|
|
break;
|
|
case Conditional::InferiorEqual:
|
|
if(len <= d->extra()->validity->valMin)
|
|
valid = true;
|
|
break;
|
|
case Conditional::Between:
|
|
if(len >= d->extra()->validity->valMin && len <= d->extra()->validity->valMax)
|
|
valid = true;
|
|
break;
|
|
case Conditional::Different:
|
|
if(len <d->extra()->validity->valMin || len >d->extra()->validity->valMax)
|
|
valid = true;
|
|
break;
|
|
default :
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if(d->extra()->validity->m_restriction == Restriction::Time && isTime())
|
|
{
|
|
switch( d->extra()->validity->m_cond)
|
|
{
|
|
case Conditional::Equal:
|
|
valid = (value().asTime() == d->extra()->validity->timeMin);
|
|
break;
|
|
case Conditional::DifferentTo:
|
|
valid = (value().asTime() != d->extra()->validity->timeMin);
|
|
break;
|
|
case Conditional::Superior:
|
|
valid = (value().asTime() > d->extra()->validity->timeMin);
|
|
break;
|
|
case Conditional::Inferior:
|
|
valid = (value().asTime() < d->extra()->validity->timeMin);
|
|
break;
|
|
case Conditional::SuperiorEqual:
|
|
valid = (value().asTime() >= d->extra()->validity->timeMin);
|
|
break;
|
|
case Conditional::InferiorEqual:
|
|
valid = (value().asTime() <= d->extra()->validity->timeMin);
|
|
break;
|
|
case Conditional::Between:
|
|
valid = (value().asTime() >= d->extra()->validity->timeMin &&
|
|
value().asTime() <= d->extra()->validity->timeMax);
|
|
break;
|
|
case Conditional::Different:
|
|
valid = (value().asTime() < d->extra()->validity->timeMin ||
|
|
value().asTime() > d->extra()->validity->timeMax);
|
|
break;
|
|
default :
|
|
break;
|
|
|
|
}
|
|
}
|
|
else if(d->extra()->validity->m_restriction == Restriction::Date && isDate())
|
|
{
|
|
switch( d->extra()->validity->m_cond)
|
|
{
|
|
case Conditional::Equal:
|
|
valid = (value().asDate() == d->extra()->validity->dateMin);
|
|
break;
|
|
case Conditional::DifferentTo:
|
|
valid = (value().asDate() != d->extra()->validity->dateMin);
|
|
break;
|
|
case Conditional::Superior:
|
|
valid = (value().asDate() > d->extra()->validity->dateMin);
|
|
break;
|
|
case Conditional::Inferior:
|
|
valid = (value().asDate() < d->extra()->validity->dateMin);
|
|
break;
|
|
case Conditional::SuperiorEqual:
|
|
valid = (value().asDate() >= d->extra()->validity->dateMin);
|
|
break;
|
|
case Conditional::InferiorEqual:
|
|
valid = (value().asDate() <= d->extra()->validity->dateMin);
|
|
break;
|
|
case Conditional::Between:
|
|
valid = (value().asDate() >= d->extra()->validity->dateMin &&
|
|
value().asDate() <= d->extra()->validity->dateMax);
|
|
break;
|
|
case Conditional::Different:
|
|
valid = (value().asDate() < d->extra()->validity->dateMin ||
|
|
value().asDate() > d->extra()->validity->dateMax);
|
|
break;
|
|
default :
|
|
break;
|
|
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
valid= true;
|
|
}
|
|
|
|
if(!valid &&d->extra()->validity != NULL && d->extra()->validity->displayMessage)
|
|
{
|
|
switch (d->extra()->validity->m_action )
|
|
{
|
|
case Action::Stop:
|
|
KMessageBox::error((TQWidget*)0L, d->extra()->validity->message,
|
|
d->extra()->validity->title);
|
|
break;
|
|
case Action::Warning:
|
|
KMessageBox::warningYesNo((TQWidget*)0L, d->extra()->validity->message,
|
|
d->extra()->validity->title);
|
|
break;
|
|
case Action::Information:
|
|
KMessageBox::information((TQWidget*)0L, d->extra()->validity->message,
|
|
d->extra()->validity->title);
|
|
break;
|
|
}
|
|
}
|
|
if (!d->hasExtra())
|
|
return true; //okay if there's no validity
|
|
return (valid || d->extra()->validity == NULL || d->extra()->validity->m_action != Action::Stop);
|
|
}
|
|
|
|
FormatType Cell::formatType() const
|
|
{
|
|
return format()->getFormatType( d->column, d->row );
|
|
}
|
|
|
|
double Cell::textWidth() const
|
|
{
|
|
return d->textWidth;
|
|
}
|
|
|
|
double Cell::textHeight() const
|
|
{
|
|
return d->textHeight;
|
|
}
|
|
|
|
int Cell::mergedXCells() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->mergedXCells : 0;
|
|
}
|
|
|
|
int Cell::mergedYCells() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->mergedYCells : 0;
|
|
}
|
|
|
|
int Cell::extraXCells() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
}
|
|
|
|
int Cell::extraYCells() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->extraYCells : 0;
|
|
}
|
|
|
|
double Cell::extraWidth() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->extraWidth : 0;
|
|
}
|
|
|
|
double Cell::extraHeight() const
|
|
{
|
|
return d->hasExtra() ? d->extra()->extraHeight : 0;
|
|
}
|
|
|
|
|
|
bool Cell::isDate() const
|
|
{
|
|
FormatType ft = formatType();
|
|
|
|
return (formatIsDate (ft) || ((ft == Generic_format) &&
|
|
(value().format() == Value::fmt_Date)));
|
|
}
|
|
|
|
bool Cell::isTime() const
|
|
{
|
|
FormatType ft = formatType();
|
|
|
|
return (formatIsTime (ft) || ((ft == Generic_format) &&
|
|
(value().format() == Value::fmt_Time)));
|
|
}
|
|
|
|
void Cell::setCalcDirtyFlag()
|
|
{
|
|
if ( !isFormula() )
|
|
{
|
|
//don't set the flag if we don't hold a formula
|
|
clearFlag(Flag_CalcDirty);
|
|
return;
|
|
}
|
|
setFlag(Flag_CalcDirty);
|
|
format()->sheet()->setRegionPaintDirty(cellRect());
|
|
}
|
|
|
|
|
|
bool Cell::updateChart(bool refresh)
|
|
{
|
|
// Update a chart for example if it depends on this cell.
|
|
if ( d->row != 0 && d->column != 0 )
|
|
{
|
|
CellBinding *bind;
|
|
for ( bind = format()->sheet()->firstCellBinding(); bind != 0L; bind = format()->sheet()->nextCellBinding() )
|
|
{
|
|
if ( bind->contains( d->column, d->row ) )
|
|
{
|
|
if (!refresh)
|
|
return true;
|
|
|
|
bind->cellChanged( this );
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
|
|
}
|
|
|
|
double Cell::getDouble ()
|
|
{
|
|
if (isDefault())
|
|
return 0.0;
|
|
//(Tomas) umm can't we simply call value().asFloat() ?
|
|
if (isDate())
|
|
{
|
|
TQDate date = value().asDate();
|
|
TQDate dummy (1900, 1, 1);
|
|
return (dummy.daysTo (date) + 1);
|
|
}
|
|
if (isTime())
|
|
{
|
|
TQTime time = value().asTime();
|
|
TQTime dummy;
|
|
return dummy.secsTo( time );
|
|
}
|
|
if (value().isNumber())
|
|
return value().asFloat();
|
|
|
|
return 0.0;
|
|
}
|
|
|
|
void Cell::convertToDouble ()
|
|
{
|
|
if (isDefault())
|
|
return;
|
|
|
|
setValue (getDouble ());
|
|
}
|
|
|
|
void Cell::convertToPercent ()
|
|
{
|
|
if (isDefault())
|
|
return;
|
|
|
|
setValue (getDouble ());
|
|
d->value.setFormat (Value::fmt_Percent);
|
|
}
|
|
|
|
void Cell::convertToMoney ()
|
|
{
|
|
if (isDefault())
|
|
return;
|
|
|
|
setValue (getDouble ());
|
|
d->value.setFormat (Value::fmt_Money);
|
|
format()->setPrecision (locale()->fracDigits());
|
|
}
|
|
|
|
void Cell::convertToTime ()
|
|
{
|
|
//(Tomas) This is weird. And I mean *REALLY* weird. First, we
|
|
//generate a time (TQTime), then we convert it to text, then
|
|
//we give the text to the cell and ask it to parse it. Weird...
|
|
|
|
if (isDefault() || isEmpty())
|
|
return;
|
|
|
|
setValue (getDouble ());
|
|
TQTime time = value().asDateTime().time();
|
|
int msec = (int) ( (value().asFloat() - (int) value().asFloat()) * 1000 );
|
|
time = time.addMSecs( msec );
|
|
setCellText( time.toString() );
|
|
}
|
|
|
|
void Cell::convertToDate ()
|
|
{
|
|
//(Tomas) This is weird. And I mean *REALLY* weird. First, we
|
|
//generate a date (TQDate), then we convert it to text, then
|
|
//we give the text to the cell and ask it to parse it. Weird...
|
|
|
|
if (isDefault() || isEmpty())
|
|
return;
|
|
|
|
setValue (getDouble ());
|
|
|
|
//TODO: why did we call setValue(), when we override it here?
|
|
TQDate date(1900, 1, 1);
|
|
date = date.addDays( (int) value().asFloat() - 1 );
|
|
date = value().asDateTime().date();
|
|
setCellText (locale()->formatDate (date, true));
|
|
}
|
|
|
|
void Cell::checkTextInput()
|
|
{
|
|
// Goal of this method: determine the value of the cell
|
|
clearAllErrors();
|
|
|
|
d->value = Value::empty();
|
|
|
|
// Get the text from that cell
|
|
TQString str = d->strText;
|
|
|
|
sheet()->doc()->parser()->parse (str, this);
|
|
|
|
// Parsing as time acts like an autoformat: we even change d->strText
|
|
// [h]:mm:ss -> might get set by ValueParser
|
|
if (isTime() && (formatType() != Time_format7))
|
|
d->strText = locale()->formatTime( value().asDateTime().time(), true);
|
|
|
|
// convert first letter to uppercase ?
|
|
if (format()->sheet()->getFirstLetterUpper() && value().isString() &&
|
|
(!d->strText.isEmpty()))
|
|
{
|
|
TQString str = value().asString();
|
|
setValue( Value( str[0].upper() + str.right( str.length()-1 ) ) );
|
|
}
|
|
}
|
|
|
|
//used in calc, setNumber, ValueParser
|
|
void Cell::checkNumberFormat()
|
|
{
|
|
if ( formatType() == Number_format && value().isNumber() )
|
|
{
|
|
if ( value().asFloat() > 1e+10 )
|
|
format()->setFormatType( Scientific_format );
|
|
}
|
|
}
|
|
|
|
|
|
// ================================================================
|
|
// Saving and loading
|
|
|
|
|
|
TQDomElement Cell::save( TQDomDocument& doc,
|
|
int _x_offset, int _y_offset,
|
|
bool force, bool copy, bool era )
|
|
{
|
|
// Save the position of this cell
|
|
TQDomElement cell = doc.createElement( "cell" );
|
|
cell.setAttribute( "row", d->row - _y_offset );
|
|
cell.setAttribute( "column", d->column - _x_offset );
|
|
//
|
|
// Save the formatting information
|
|
//
|
|
TQDomElement formatElement = format()->save( doc, d->column, d->row, force, copy );
|
|
if ( formatElement.hasChildNodes() || formatElement.attributes().length() ) // don't save empty tags
|
|
cell.appendChild( formatElement );
|
|
|
|
if ( doesMergeCells() )
|
|
{
|
|
if ( extraXCells() )
|
|
formatElement.setAttribute( "colspan", extraXCells() );
|
|
if ( extraYCells() )
|
|
formatElement.setAttribute( "rowspan", extraYCells() );
|
|
}
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions )
|
|
{
|
|
TQDomElement conditionElement = d->extra()->conditions->saveConditions( doc );
|
|
|
|
if ( !conditionElement.isNull() )
|
|
cell.appendChild( conditionElement );
|
|
}
|
|
|
|
if ( d->hasExtra() && (d->extra()->validity != 0) )
|
|
{
|
|
TQDomElement validity = doc.createElement("validity");
|
|
|
|
TQDomElement param=doc.createElement("param");
|
|
param.setAttribute("cond",(int)d->extra()->validity->m_cond);
|
|
param.setAttribute("action",(int)d->extra()->validity->m_action);
|
|
param.setAttribute("allow",(int)d->extra()->validity->m_restriction);
|
|
param.setAttribute("valmin",d->extra()->validity->valMin);
|
|
param.setAttribute("valmax",d->extra()->validity->valMax);
|
|
param.setAttribute("displaymessage",d->extra()->validity->displayMessage);
|
|
param.setAttribute("displayvalidationinformation",d->extra()->validity->displayValidationInformation);
|
|
param.setAttribute("allowemptycell",d->extra()->validity->allowEmptyCell);
|
|
if ( !d->extra()->validity->listValidity.isEmpty() )
|
|
param.setAttribute( "listvalidity", d->extra()->validity->listValidity.join( ";" ) );
|
|
validity.appendChild(param);
|
|
TQDomElement title = doc.createElement( "title" );
|
|
title.appendChild( doc.createTextNode( d->extra()->validity->title ) );
|
|
validity.appendChild( title );
|
|
TQDomElement message = doc.createElement( "message" );
|
|
message.appendChild( doc.createCDATASection( d->extra()->validity->message ) );
|
|
validity.appendChild( message );
|
|
|
|
TQDomElement inputTitle = doc.createElement( "inputtitle" );
|
|
inputTitle.appendChild( doc.createTextNode( d->extra()->validity->titleInfo ) );
|
|
validity.appendChild( inputTitle );
|
|
|
|
TQDomElement inputMessage = doc.createElement( "inputmessage" );
|
|
inputMessage.appendChild( doc.createTextNode( d->extra()->validity->messageInfo ) );
|
|
validity.appendChild( inputMessage );
|
|
|
|
|
|
|
|
TQString tmp;
|
|
if ( d->extra()->validity->timeMin.isValid() )
|
|
{
|
|
TQDomElement timeMin = doc.createElement( "timemin" );
|
|
tmp=d->extra()->validity->timeMin.toString();
|
|
timeMin.appendChild( doc.createTextNode( tmp ) );
|
|
validity.appendChild( timeMin );
|
|
}
|
|
if ( d->extra()->validity->timeMax.isValid() )
|
|
{
|
|
TQDomElement timeMax = doc.createElement( "timemax" );
|
|
tmp=d->extra()->validity->timeMax.toString();
|
|
timeMax.appendChild( doc.createTextNode( tmp ) );
|
|
validity.appendChild( timeMax );
|
|
}
|
|
|
|
if ( d->extra()->validity->dateMin.isValid() )
|
|
{
|
|
TQDomElement dateMin = doc.createElement( "datemin" );
|
|
TQString tmp("%1/%2/%3");
|
|
tmp = tmp.arg(d->extra()->validity->dateMin.year()).arg(d->extra()->validity->dateMin.month()).arg(d->extra()->validity->dateMin.day());
|
|
dateMin.appendChild( doc.createTextNode( tmp ) );
|
|
validity.appendChild( dateMin );
|
|
}
|
|
if ( d->extra()->validity->dateMax.isValid() )
|
|
{
|
|
TQDomElement dateMax = doc.createElement( "datemax" );
|
|
TQString tmp("%1/%2/%3");
|
|
tmp = tmp.arg(d->extra()->validity->dateMax.year()).arg(d->extra()->validity->dateMax.month()).arg(d->extra()->validity->dateMax.day());
|
|
dateMax.appendChild( doc.createTextNode( tmp ) );
|
|
validity.appendChild( dateMax );
|
|
}
|
|
|
|
cell.appendChild( validity );
|
|
}
|
|
|
|
if ( format()->comment() )
|
|
{
|
|
TQDomElement comment = doc.createElement( "comment" );
|
|
comment.appendChild( doc.createCDATASection( *format()->comment() ) );
|
|
cell.appendChild( comment );
|
|
}
|
|
|
|
//
|
|
// Save the text
|
|
//
|
|
if ( !d->strText.isEmpty() )
|
|
{
|
|
// Formulas need to be encoded to ensure that they
|
|
// are position independent.
|
|
if ( isFormula() )
|
|
{
|
|
TQDomElement text = doc.createElement( "text" );
|
|
// if we are cutting to the clipboard, relative references need to be encoded absolutely
|
|
text.appendChild( doc.createTextNode( encodeFormula( era ) ) );
|
|
cell.appendChild( text );
|
|
|
|
/* we still want to save the results of the formula */
|
|
TQDomElement formulaResult = doc.createElement( "result" );
|
|
saveCellResult( doc, formulaResult, d->strOutText );
|
|
cell.appendChild( formulaResult );
|
|
|
|
}
|
|
else if ( !link().isEmpty() )
|
|
{
|
|
// KSpread pre 1.4 saves link as rich text, marked with first char '
|
|
// Have to be saved in some CDATA section because of too many special charatcers.
|
|
TQDomElement text = doc.createElement( "text" );
|
|
TQString qml = "!<a href=\"" + link() + "\">" + d->strText + "</a>";
|
|
text.appendChild( doc.createCDATASection( qml ) );
|
|
cell.appendChild( text );
|
|
}
|
|
else
|
|
{
|
|
// Save the cell contents (in a locale-independent way)
|
|
TQDomElement text = doc.createElement( "text" );
|
|
saveCellResult( doc, text, d->strText );
|
|
cell.appendChild( text );
|
|
}
|
|
}
|
|
if ( cell.hasChildNodes() || cell.attributes().length() > 2 ) // don't save empty tags
|
|
// (the >2 is due to "row" and "column" attributes)
|
|
return cell;
|
|
else
|
|
return TQDomElement();
|
|
}
|
|
|
|
bool Cell::saveCellResult( TQDomDocument& doc, TQDomElement& result,
|
|
TQString str )
|
|
{
|
|
TQString dataType = "Other"; // fallback
|
|
|
|
if ( value().isNumber() )
|
|
{
|
|
if ( isDate() )
|
|
{
|
|
// serial number of date
|
|
TQDate dd = value().asDateTime().date();
|
|
dataType = "Date";
|
|
str = "%1/%2/%3";
|
|
str = str.arg(dd.year()).arg(dd.month()).arg(dd.day());
|
|
}
|
|
else if( isTime() )
|
|
{
|
|
// serial number of time
|
|
dataType = "Time";
|
|
str = value().asDateTime().time().toString();
|
|
}
|
|
else
|
|
{
|
|
// real number
|
|
dataType = "Num";
|
|
if (value().isInteger())
|
|
str = TQString::number(value().asInteger());
|
|
else
|
|
str = TQString::number(value().asFloat(), 'g', DBL_DIG);
|
|
}
|
|
}
|
|
|
|
if ( value().isBoolean() )
|
|
{
|
|
dataType = "Bool";
|
|
str = value().asBoolean() ? "true" : "false";
|
|
}
|
|
|
|
if ( value().isString() )
|
|
{
|
|
dataType = "Str";
|
|
str = value().asString();
|
|
}
|
|
|
|
result.setAttribute( "dataType", dataType );
|
|
if ( !d->strOutText.isEmpty() )
|
|
result.setAttribute( "outStr", d->strOutText );
|
|
result.appendChild( doc.createTextNode( str ) );
|
|
|
|
return true; /* really isn't much of a way for this function to fail */
|
|
}
|
|
|
|
void Cell::saveOasisAnnotation( KoXmlWriter &xmlwriter )
|
|
{
|
|
if ( format()->comment() )
|
|
{
|
|
//<office:annotation draw:style-name="gr1" draw:text-style-name="P1" svg:width="2.899cm" svg:height="2.691cm" svg:x="2.858cm" svg:y="0.001cm" draw:caption-point-x="-2.858cm" draw:caption-point-y="-0.001cm">
|
|
xmlwriter.startElement( "office:annotation" );
|
|
TQStringList text = TQStringList::split( "\n", *format()->comment() );
|
|
for ( TQStringList::Iterator it = text.begin(); it != text.end(); ++it ) {
|
|
xmlwriter.startElement( "text:p" );
|
|
xmlwriter.addTextNode( *it );
|
|
xmlwriter.endElement();
|
|
}
|
|
xmlwriter.endElement();
|
|
}
|
|
}
|
|
|
|
|
|
|
|
TQString Cell::saveOasisCellStyle( KoGenStyle ¤tCellStyle, KoGenStyles &mainStyles )
|
|
{
|
|
if ( d->hasExtra() && d->extra()->conditions )
|
|
{
|
|
// this has to be an automatic style
|
|
currentCellStyle = KoGenStyle( Doc::STYLE_CELL_AUTO, "table-cell" );
|
|
d->extra()->conditions->saveOasisConditions( currentCellStyle );
|
|
}
|
|
return format()->saveOasisCellStyle( currentCellStyle, mainStyles );
|
|
}
|
|
|
|
|
|
bool Cell::saveOasis( KoXmlWriter& xmlwriter, KoGenStyles &mainStyles,
|
|
int row, int column, int &repeated,
|
|
GenValidationStyles &valStyle )
|
|
{
|
|
if ( !isPartOfMerged() )
|
|
xmlwriter.startElement( "table:table-cell" );
|
|
else
|
|
xmlwriter.startElement( "table:covered-table-cell" );
|
|
#if 0
|
|
//add font style
|
|
TQFont font;
|
|
Value const value( cell->value() );
|
|
if ( !cell->isDefault() )
|
|
{
|
|
font = cell->format()->textFont( i, row );
|
|
m_styles.addFont( font );
|
|
|
|
if ( cell->format()->hasProperty( Format::PComment ) )
|
|
hasComment = true;
|
|
}
|
|
#endif
|
|
// NOTE save the value before the style as long as the Formatter does not work correctly
|
|
if ( link().isEmpty() )
|
|
saveOasisValue (xmlwriter);
|
|
|
|
KoGenStyle currentCellStyle; // the type determined in saveOasisCellStyle
|
|
saveOasisCellStyle( currentCellStyle,mainStyles );
|
|
// skip 'table:style-name' attribute for the default style
|
|
if ( !currentCellStyle.isDefaultStyle() )
|
|
xmlwriter.addAttribute( "table:style-name", mainStyles.styles()[currentCellStyle] );
|
|
|
|
// group empty cells with the same style
|
|
if ( isEmpty() && !format()->hasProperty( Format::PComment ) &&
|
|
!isPartOfMerged() && !doesMergeCells() )
|
|
{
|
|
bool refCellIsDefault = isDefault();
|
|
int j = column + 1;
|
|
Cell *nextCell = format()->sheet()->getNextCellRight( column, row );
|
|
while ( nextCell )
|
|
{
|
|
// if
|
|
// the next cell is not the adjacent one
|
|
// or
|
|
// the next cell is not empty
|
|
if ( nextCell->column() != j || !nextCell->isEmpty() )
|
|
{
|
|
if ( refCellIsDefault )
|
|
{
|
|
// if the origin cell was a default cell,
|
|
// we count the default cells
|
|
repeated = nextCell->column() - j + 1;
|
|
}
|
|
// otherwise we just stop here to process the adjacent
|
|
// cell in the next iteration of the outer loop
|
|
// (in Sheet::saveOasisCells)
|
|
break;
|
|
}
|
|
|
|
KoGenStyle nextCellStyle; // the type is determined in saveOasisCellStyle
|
|
nextCell->saveOasisCellStyle( nextCellStyle,mainStyles );
|
|
|
|
if ( nextCell->isPartOfMerged() || nextCell->doesMergeCells() ||
|
|
nextCell->format()->hasProperty( Format::PComment ) ||
|
|
!(nextCellStyle == currentCellStyle) )
|
|
{
|
|
break;
|
|
}
|
|
++repeated;
|
|
// get the next cell and set the index to the adjacent cell
|
|
nextCell = format()->sheet()->getNextCellRight( j++, row );
|
|
}
|
|
kdDebug() << "Cell::saveOasis: empty cell in column " << column << " "
|
|
<< "repeated " << repeated << " time(s)" << endl;
|
|
|
|
if ( repeated > 1 )
|
|
xmlwriter.addAttribute( "table:number-columns-repeated", TQString::number( repeated ) );
|
|
}
|
|
|
|
|
|
if (d->hasExtra() && d->extra()->validity)
|
|
{
|
|
GenValidationStyle styleVal(d->extra()->validity);
|
|
xmlwriter.addAttribute( "table:validation-name", valStyle.lookup( styleVal ) );
|
|
}
|
|
if ( isFormula() )
|
|
{
|
|
//kdDebug() << "Formula found" << endl;
|
|
TQString formula( convertFormulaToOasisFormat( text() ) );
|
|
xmlwriter.addAttribute( "table:formula", formula );
|
|
}
|
|
else if ( !link().isEmpty() )
|
|
{
|
|
//kdDebug()<<"Link found \n";
|
|
xmlwriter.startElement( "text:p" );
|
|
xmlwriter.startElement( "text:a" );
|
|
//Reference cell is started by "#"
|
|
if ( localReferenceAnchor( link() ) )
|
|
xmlwriter.addAttribute( " xlink:href", ( "#"+link() ) );
|
|
else
|
|
xmlwriter.addAttribute( " xlink:href", link() );
|
|
xmlwriter.addTextNode( text() );
|
|
xmlwriter.endElement();
|
|
xmlwriter.endElement();
|
|
}
|
|
|
|
if ( doesMergeCells() )
|
|
{
|
|
int colSpan = mergedXCells() + 1;
|
|
int rowSpan = mergedYCells() + 1;
|
|
|
|
if ( colSpan > 1 )
|
|
xmlwriter.addAttribute( "table:number-columns-spanned", TQString::number( colSpan ) );
|
|
|
|
if ( rowSpan > 1 )
|
|
xmlwriter.addAttribute( "table:number-rows-spanned", TQString::number( rowSpan ) );
|
|
}
|
|
|
|
if ( !isEmpty() && link().isEmpty() )
|
|
{
|
|
xmlwriter.startElement( "text:p" );
|
|
xmlwriter.addTextNode( strOutText().utf8() );
|
|
xmlwriter.endElement();
|
|
}
|
|
|
|
saveOasisAnnotation( xmlwriter );
|
|
|
|
xmlwriter.endElement();
|
|
return true;
|
|
}
|
|
|
|
void Cell::saveOasisValue (KoXmlWriter &xmlWriter)
|
|
{
|
|
switch (value().format())
|
|
{
|
|
case Value::fmt_None: break; //NOTHING HERE
|
|
case Value::fmt_Boolean:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "boolean" );
|
|
xmlWriter.addAttribute( "office:boolean-value", ( value().asBoolean() ?
|
|
"true" : "false" ) );
|
|
break;
|
|
}
|
|
case Value::fmt_Number:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "float" );
|
|
if (value().isInteger())
|
|
xmlWriter.addAttribute( "office:value", TQString::number( value().asInteger() ) );
|
|
else
|
|
xmlWriter.addAttribute( "office:value", TQString::number( value().asFloat(), 'g', DBL_DIG ) );
|
|
break;
|
|
}
|
|
case Value::fmt_Percent:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "percentage" );
|
|
xmlWriter.addAttribute( "office:value",
|
|
TQString::number( value().asFloat() ) );
|
|
break;
|
|
}
|
|
case Value::fmt_Money:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "currency" );
|
|
Format::Currency currency;
|
|
if (format()->currencyInfo(currency))
|
|
xmlWriter.addAttribute( "office:currency", Currency::getCurrencyCode(currency.type) );
|
|
xmlWriter.addAttribute( "office:value",
|
|
TQString::number( value().asFloat() ) );
|
|
break;
|
|
}
|
|
case Value::fmt_DateTime: break; //NOTHING HERE
|
|
case Value::fmt_Date:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "date" );
|
|
xmlWriter.addAttribute( "office:date-value",
|
|
value().asDate().toString( TQt::ISODate ) );
|
|
break;
|
|
}
|
|
case Value::fmt_Time:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "time" );
|
|
xmlWriter.addAttribute( "office:time-value",
|
|
value().asTime().toString( "PThhHmmMssS" ) );
|
|
break;
|
|
}
|
|
case Value::fmt_String:
|
|
{
|
|
xmlWriter.addAttribute( "office:value-type", "string" );
|
|
xmlWriter.addAttribute( "office:string-value", value().asString() );
|
|
break;
|
|
}
|
|
};
|
|
}
|
|
|
|
TQString Cell::convertFormulaToOasisFormat( const TQString & formula ) const
|
|
{
|
|
TQString s;
|
|
TQRegExp exp("(\\$?)([a-zA-Z]+)(\\$?)([0-9]+)");
|
|
int n = exp.search( formula, 0 );
|
|
kdDebug() << "Exp: " << formula << ", n: " << n << ", Length: " << formula.length()
|
|
<< ", Matched length: " << exp.matchedLength() << endl;
|
|
|
|
bool inQuote1 = false;
|
|
bool inQuote2 = false;
|
|
int i = 0;
|
|
int l = (int) formula.length();
|
|
if ( l <= 0 )
|
|
return formula;
|
|
while ( i < l )
|
|
{
|
|
if ( ( n != -1 ) && ( n < i ) )
|
|
{
|
|
n = exp.search( formula, i );
|
|
kdDebug() << "Exp: " << formula.right( l - i ) << ", n: " << n << endl;
|
|
}
|
|
if ( formula[i] == '"' )
|
|
{
|
|
inQuote1 = !inQuote1;
|
|
s += formula[i];
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( formula[i] == '\'' )
|
|
{
|
|
// named area
|
|
inQuote2 = !inQuote2;
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( inQuote1 || inQuote2 )
|
|
{
|
|
s += formula[i];
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( ( formula[i] == '=' ) && ( formula[i + 1] == '=' ) )
|
|
{
|
|
s += '=';
|
|
++i;++i;
|
|
continue;
|
|
}
|
|
if ( formula[i] == '!' )
|
|
{
|
|
insertBracket( s );
|
|
s += '.';
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( formula[i] == ',' )
|
|
{
|
|
s += '.';
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( n == i )
|
|
{
|
|
int ml = exp.matchedLength();
|
|
if ( formula[ i + ml ] == '!' )
|
|
{
|
|
kdDebug() << "No cell ref but sheet name" << endl;
|
|
s += formula[i];
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( ( i > 0 ) && ( formula[i - 1] != '!' ) )
|
|
s += "[.";
|
|
for ( int j = 0; j < ml; ++j )
|
|
{
|
|
s += formula[i];
|
|
++i;
|
|
}
|
|
s += ']';
|
|
continue;
|
|
}
|
|
|
|
s += formula[i];
|
|
++i;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
void Cell::loadOasisConditional( TQDomElement * style )
|
|
{
|
|
if ( style )//safe
|
|
{
|
|
TQDomElement e;
|
|
forEachElement( e, style->toElement() )
|
|
{
|
|
if ( e.localName() == "map" && e.namespaceURI() == KoXmlNS::style )
|
|
{
|
|
if (d->hasExtra())
|
|
delete d->extra()->conditions;
|
|
d->extra()->conditions = new Conditions( this );
|
|
d->extra()->conditions->loadOasisConditions( e );
|
|
d->extra()->conditions->checkMatches();
|
|
// break here
|
|
// Conditions::loadOasisConditions finishes the iteration
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Cell::loadOasis( const TQDomElement& element , KoOasisLoadingContext& oasisContext , Style* style )
|
|
{
|
|
kdDebug() << "*** Loading cell properties ***** at " << column() << "," << row () << endl;
|
|
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "style-name" ) )
|
|
{
|
|
kdDebug()<<" table:style-name: "<<element.attributeNS( KoXmlNS::table, "style-name", TQString() )<<endl;
|
|
oasisContext.fillStyleStack( element, KoXmlNS::table, "styleName", "table-cell" );
|
|
|
|
TQString str = element.attributeNS( KoXmlNS::table, "style-name", TQString() );
|
|
const TQDomElement* cellStyle = oasisContext.oasisStyles().findStyle( str, "table-cell" );
|
|
|
|
if ( cellStyle )
|
|
loadOasisConditional( const_cast<TQDomElement *>( cellStyle ) );
|
|
}
|
|
|
|
if (style)
|
|
{
|
|
format()->setStyle( style );
|
|
}
|
|
|
|
//Search and load each paragraph of text. Each paragraph is separated by a line break.
|
|
loadOasisCellText( element );
|
|
|
|
//
|
|
// formula
|
|
//
|
|
bool isFormula = false;
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "formula" ) )
|
|
{
|
|
kdDebug()<<" formula :"<<element.attributeNS( KoXmlNS::table, "formula", TQString() )<<endl;
|
|
isFormula = true;
|
|
TQString oasisFormula( element.attributeNS( KoXmlNS::table, "formula", TQString() ) );
|
|
//necessary to remove it to load formula from oocalc2.0 (use namespace)
|
|
if (oasisFormula.startsWith( "oooc:" ) )
|
|
oasisFormula= oasisFormula.mid( 5 );
|
|
else if (oasisFormula.startsWith( "kspr:" ) )
|
|
oasisFormula= oasisFormula.mid( 5 );
|
|
// TODO Stefan: merge this into Oasis::decodeFormula
|
|
checkForNamedAreas( oasisFormula );
|
|
oasisFormula = Oasis::decodeFormula( oasisFormula, locale() );
|
|
setCellText( oasisFormula );
|
|
}
|
|
else if ( d->strText.at(0) == '=' ) //prepend ' to the text to avoid = to be painted
|
|
d->strText.prepend('\'');
|
|
|
|
//
|
|
// validation
|
|
//
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "validation-name" ) )
|
|
{
|
|
kdDebug()<<" validation-name: "<<element.attributeNS( KoXmlNS::table, "validation-name", TQString() )<<endl;
|
|
loadOasisValidation( element.attributeNS( KoXmlNS::table, "validation-name", TQString() ) );
|
|
}
|
|
|
|
//
|
|
// value type
|
|
//
|
|
if( element.hasAttributeNS( KoXmlNS::office, "value-type" ) )
|
|
{
|
|
TQString valuetype = element.attributeNS( KoXmlNS::office, "value-type", TQString() );
|
|
kdDebug()<<" value-type: " << valuetype << endl;
|
|
if( valuetype == "boolean" )
|
|
{
|
|
TQString val = element.attributeNS( KoXmlNS::office, "boolean-value", TQString() ).lower();
|
|
if( ( val == "true" ) || ( val == "false" ) )
|
|
{
|
|
bool value = val == "true";
|
|
setCellValue( value );
|
|
}
|
|
}
|
|
|
|
// integer and floating-point value
|
|
else if( valuetype == "float" )
|
|
{
|
|
bool ok = false;
|
|
double value = element.attributeNS( KoXmlNS::office, "value", TQString() ).toDouble( &ok );
|
|
if( ok )
|
|
setCellValue( value );
|
|
|
|
if ( !isFormula && d->strText.isEmpty())
|
|
{
|
|
TQString str = locale()->formatNumber( value, 15 );
|
|
setCellText( str );
|
|
}
|
|
}
|
|
|
|
// currency value
|
|
else if( valuetype == "currency" )
|
|
{
|
|
bool ok = false;
|
|
double value = element.attributeNS( KoXmlNS::office, "value", TQString() ).toDouble( &ok );
|
|
if( ok )
|
|
{
|
|
setCellValue( value, Money_format );
|
|
|
|
if (element.hasAttributeNS( KoXmlNS::office, "currency" ) )
|
|
{
|
|
Currency currency(element.attributeNS( KoXmlNS::office, "currency", TQString() ) );
|
|
format()->setCurrency( currency.getIndex(), currency.getDisplayCode() );
|
|
}
|
|
}
|
|
}
|
|
else if( valuetype == "percentage" )
|
|
{
|
|
bool ok = false;
|
|
double v = element.attributeNS( KoXmlNS::office, "value", TQString() ).toDouble( &ok );
|
|
if( ok )
|
|
{
|
|
Value value;
|
|
value.setValue (v);
|
|
value.setFormat (Value::fmt_Percent);
|
|
setCellValue( value );
|
|
|
|
if ( !isFormula && d->strText.isEmpty())
|
|
{
|
|
TQString str = locale()->formatNumber( v, 15 );
|
|
setCellText( str );
|
|
}
|
|
|
|
format()->setFormatType (Percentage_format);
|
|
}
|
|
}
|
|
else if ( valuetype == "date" )
|
|
{
|
|
TQString value = element.attributeNS( KoXmlNS::office, "value", TQString() );
|
|
if ( value.isEmpty() )
|
|
value = element.attributeNS( KoXmlNS::office, "date-value", TQString() );
|
|
kdDebug() << "Type: date, value: " << value << endl;
|
|
|
|
// "1980-10-15"
|
|
int year = 0, month = 0, day = 0;
|
|
bool ok = false;
|
|
|
|
int p1 = value.find( '-' );
|
|
if ( p1 > 0 )
|
|
year = value.left( p1 ).toInt( &ok );
|
|
|
|
kdDebug() << "year: " << value.left( p1 ) << endl;
|
|
|
|
int p2 = value.find( '-', ++p1 );
|
|
|
|
if ( ok )
|
|
month = value.mid( p1, p2 - p1 ).toInt( &ok );
|
|
|
|
kdDebug() << "month: " << value.mid( p1, p2 - p1 ) << endl;
|
|
|
|
if ( ok )
|
|
day = value.right( value.length() - p2 - 1 ).toInt( &ok );
|
|
|
|
kdDebug() << "day: " << value.right( value.length() - p2 ) << endl;
|
|
|
|
if ( ok )
|
|
{
|
|
setCellValue( TQDate( year, month, day ) );
|
|
if ( style )
|
|
format()->setFormatType (style->formatType());
|
|
kdDebug() << "Set TQDate: " << year << " - " << month << " - " << day << endl;
|
|
}
|
|
|
|
}
|
|
else if ( valuetype == "time" )
|
|
{
|
|
TQString value = element.attributeNS( KoXmlNS::office, "value", TQString() );
|
|
if ( value.isEmpty() )
|
|
value = element.attributeNS( KoXmlNS::office, "time-value", TQString() );
|
|
kdDebug() << "Type: time: " << value << endl;
|
|
// "PT15H10M12S"
|
|
int hours = 0, minutes = 0, seconds = 0;
|
|
int l = value.length();
|
|
TQString num;
|
|
bool ok = false;
|
|
for ( int i = 0; i < l; ++i )
|
|
{
|
|
if ( value[i].isNumber() )
|
|
{
|
|
num += value[i];
|
|
continue;
|
|
}
|
|
else if ( value[i] == 'H' )
|
|
hours = num.toInt( &ok );
|
|
else if ( value[i] == 'M' )
|
|
minutes = num.toInt( &ok );
|
|
else if ( value[i] == 'S' )
|
|
seconds = num.toInt( &ok );
|
|
else
|
|
continue;
|
|
|
|
kdDebug() << "Num: " << num << endl;
|
|
|
|
num = "";
|
|
if ( !ok )
|
|
break;
|
|
}
|
|
kdDebug() << "Hours: " << hours << ", " << minutes << ", " << seconds << endl;
|
|
|
|
if ( ok )
|
|
{
|
|
// Value kval( timeToNum( hours, minutes, seconds ) );
|
|
// cell->setValue( kval );
|
|
setCellValue( TQTime( hours % 24, minutes, seconds ) );
|
|
if ( style )
|
|
format()->setFormatType (style->formatType());
|
|
}
|
|
}
|
|
else if( valuetype == "string" )
|
|
{
|
|
TQString value = element.attributeNS( KoXmlNS::office, "value", TQString() );
|
|
if ( value.isEmpty() && element.hasAttributeNS( KoXmlNS::office, "string-value" ))
|
|
{
|
|
//if there is not string-value entry don't overwrite value stored into <text:p>
|
|
value = element.attributeNS( KoXmlNS::office, "string-value", TQString() );
|
|
setCellValue( value );
|
|
}
|
|
format()->setFormatType (Text_format);
|
|
}
|
|
else
|
|
kdDebug()<<" type of value found : "<<valuetype<<endl;
|
|
}
|
|
|
|
//
|
|
// merged cells ?
|
|
//
|
|
int colSpan = 1;
|
|
int rowSpan = 1;
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "number-columns-spanned" ) )
|
|
{
|
|
bool ok = false;
|
|
int span = element.attributeNS( KoXmlNS::table, "number-columns-spanned", TQString() ).toInt( &ok );
|
|
if( ok ) colSpan = span;
|
|
}
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "number-rows-spanned" ) )
|
|
{
|
|
bool ok = false;
|
|
int span = element.attributeNS( KoXmlNS::table, "number-rows-spanned", TQString() ).toInt( &ok );
|
|
if( ok ) rowSpan = span;
|
|
}
|
|
if ( colSpan > 1 || rowSpan > 1 )
|
|
mergeCells( d->column, d->row, colSpan - 1, rowSpan - 1 );
|
|
|
|
//
|
|
// cell comment/annotation
|
|
//
|
|
TQDomElement annotationElement = KoDom::namedItemNS( element, KoXmlNS::office, "annotation" );
|
|
if ( !annotationElement.isNull() )
|
|
{
|
|
TQString comment;
|
|
TQDomNode node = annotationElement.firstChild();
|
|
while( !node.isNull() )
|
|
{
|
|
TQDomElement commentElement = node.toElement();
|
|
if( !commentElement.isNull() )
|
|
if( commentElement.localName() == "p" && commentElement.namespaceURI() == KoXmlNS::text )
|
|
{
|
|
if( !comment.isEmpty() ) comment.append( '\n' );
|
|
comment.append( commentElement.text() );
|
|
}
|
|
|
|
node = node.nextSibling();
|
|
}
|
|
|
|
if( !comment.isEmpty() )
|
|
format()->setComment( comment );
|
|
}
|
|
|
|
TQDomElement frame = KoDom::namedItemNS( element, KoXmlNS::draw, "frame" );
|
|
if ( !frame.isNull() )
|
|
loadOasisObjects( frame, oasisContext );
|
|
|
|
if (isFormula)
|
|
setCalcDirtyFlag (); // formulas must be recalculated
|
|
|
|
return true;
|
|
}
|
|
|
|
void Cell::loadOasisCellText( const TQDomElement& parent )
|
|
{
|
|
//Search and load each paragraph of text. Each paragraph is separated by a line break
|
|
TQDomElement textParagraphElement;
|
|
TQString cellText;
|
|
|
|
bool multipleTextParagraphsFound=false;
|
|
|
|
forEachElement( textParagraphElement , parent )
|
|
{
|
|
if ( textParagraphElement.localName()=="p" &&
|
|
textParagraphElement.namespaceURI()== KoXmlNS::text )
|
|
{
|
|
// our text, could contain formating for value or result of formul
|
|
if (cellText.isEmpty())
|
|
cellText = textParagraphElement.text();
|
|
else
|
|
{
|
|
cellText += "\n"+textParagraphElement.text();
|
|
multipleTextParagraphsFound=true;
|
|
}
|
|
|
|
TQDomElement textA = KoDom::namedItemNS( textParagraphElement, KoXmlNS::text, "a" );
|
|
if( !textA.isNull() )
|
|
{
|
|
if ( textA.hasAttributeNS( KoXmlNS::xlink, "href" ) )
|
|
{
|
|
TQString link = textA.attributeNS( KoXmlNS::xlink, "href", TQString() );
|
|
cellText = textA.text();
|
|
setCellText( cellText );
|
|
setValue( cellText );
|
|
if ( link[0]=='#' )
|
|
link=link.remove( 0, 1 );
|
|
setLink( link );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!cellText.isNull())
|
|
{
|
|
setCellText( cellText );
|
|
setValue( cellText );
|
|
}
|
|
|
|
//Enable word wrapping if multiple lines of text have been found.
|
|
if ( multipleTextParagraphsFound )
|
|
{
|
|
format()->setMultiRow(true);
|
|
}
|
|
}
|
|
|
|
void Cell::loadOasisObjects( const TQDomElement &parent, KoOasisLoadingContext& oasisContext )
|
|
{
|
|
for( TQDomElement e = parent; !e.isNull(); e = e.nextSibling().toElement() )
|
|
{
|
|
if ( e.localName() == "frame" && e.namespaceURI() == KoXmlNS::draw )
|
|
{
|
|
EmbeddedObject *obj = 0;
|
|
TQDomNode object = KoDom::namedItemNS( e, KoXmlNS::draw, "object" );
|
|
if ( !object.isNull() )
|
|
{
|
|
if ( !object.toElement().attributeNS( KoXmlNS::draw, "notify-on-update-of-ranges", TQString()).isNull() )
|
|
obj = new EmbeddedChart( sheet()->doc(), sheet() );
|
|
else
|
|
obj = new EmbeddedKOfficeObject( sheet()->doc(), sheet() );
|
|
}
|
|
else
|
|
{
|
|
TQDomNode image = KoDom::namedItemNS( e, KoXmlNS::draw, "image" );
|
|
if ( !image.isNull() )
|
|
obj = new EmbeddedPictureObject( sheet(), sheet()->doc()->pictureCollection() );
|
|
else
|
|
kdDebug() << "Object type wasn't loaded!" << endl;
|
|
}
|
|
|
|
if ( obj )
|
|
{
|
|
obj->loadOasis( e, oasisContext );
|
|
sheet()->doc()->insertObject( obj );
|
|
|
|
TQString ref = e.attributeNS( KoXmlNS::table, "end-cell-address", TQString() );
|
|
if ( ref.isNull() )
|
|
continue;
|
|
|
|
ref = Oasis::decodeFormula( ref );
|
|
Point point( ref );
|
|
if ( !point.isValid() )
|
|
continue;
|
|
|
|
KoRect geometry = obj->geometry();
|
|
geometry.setLeft( geometry.left() + sheet()->columnPos( d->column, 0 ) );
|
|
geometry.setTop( geometry.top() + sheet()->rowPos( d->row, 0 ) );
|
|
|
|
TQString str = e.attributeNS( KoXmlNS::table, "end-x", TQString() );
|
|
if ( !str.isNull() )
|
|
{
|
|
uint end_x = (uint) KoUnit::parseValue( str );
|
|
geometry.setRight( sheet()->columnPos( point.column(), 0) + end_x );
|
|
}
|
|
|
|
str = e.attributeNS( KoXmlNS::table, "end-y", TQString() );
|
|
if ( !str.isNull() )
|
|
{
|
|
uint end_y = (uint) KoUnit::parseValue( str );
|
|
geometry.setBottom( sheet()->rowPos( point.row(), 0) + end_y );
|
|
}
|
|
|
|
obj->setGeometry( geometry );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Cell::loadOasisValidation( const TQString& validationName )
|
|
{
|
|
TQDomElement element = sheet()->doc()->loadingInfo()->validation( validationName);
|
|
if (d->hasExtra())
|
|
delete d->extra()->validity;
|
|
d->extra()->validity = new Validity;
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "condition" ) )
|
|
{
|
|
TQString valExpression = element.attributeNS( KoXmlNS::table, "condition", TQString() );
|
|
kdDebug()<<" element.attribute( table:condition ) "<<valExpression<<endl;
|
|
//Condition ::= ExtendedTrueCondition | TrueFunction 'and' TrueCondition
|
|
//TrueFunction ::= cell-content-is-whole-number() | cell-content-is-decimal-number() | cell-content-is-date() | cell-content-is-time()
|
|
//ExtendedTrueCondition ::= ExtendedGetFunction | cell-content-text-length() Operator Value
|
|
//TrueCondition ::= GetFunction | cell-content() Operator Value
|
|
//GetFunction ::= cell-content-is-between(Value, Value) | cell-content-is-not-between(Value, Value)
|
|
//ExtendedGetFunction ::= cell-content-text-length-is-between(Value, Value) | cell-content-text-length-is-not-between(Value, Value)
|
|
//Operator ::= '<' | '>' | '<=' | '>=' | '=' | '!='
|
|
//Value ::= NumberValue | String | Formula
|
|
//A Formula is a formula without an equals (=) sign at the beginning. See section 8.1.3 for more information.
|
|
//A String comprises one or more characters surrounded by quotation marks.
|
|
//A NumberValue is a whole or decimal number. It must not contain comma separators for numbers of 1000 or greater.
|
|
|
|
//ExtendedTrueCondition
|
|
if ( valExpression.contains( "cell-content-text-length()" ) )
|
|
{
|
|
//"cell-content-text-length()>45"
|
|
valExpression = valExpression.remove("oooc:cell-content-text-length()" );
|
|
kdDebug()<<" valExpression = :"<<valExpression<<endl;
|
|
d->extra()->validity->m_restriction = Restriction::TextLength;
|
|
|
|
loadOasisValidationCondition( valExpression );
|
|
}
|
|
else if ( valExpression.contains( "cell-content-is-text()" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::Text;
|
|
}
|
|
//cell-content-text-length-is-between(Value, Value) | cell-content-text-length-is-not-between(Value, Value) | cell-content-is-in-list( StringList )
|
|
else if ( valExpression.contains( "cell-content-text-length-is-between" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::TextLength;
|
|
d->extra()->validity->m_cond = Conditional::Between;
|
|
valExpression = valExpression.remove( "oooc:cell-content-text-length-is-between(" );
|
|
kdDebug()<<" valExpression :"<<valExpression<<endl;
|
|
valExpression = valExpression.remove( ")" );
|
|
TQStringList listVal = TQStringList::split( ",", valExpression );
|
|
loadOasisValidationValue( listVal );
|
|
}
|
|
else if ( valExpression.contains( "cell-content-text-length-is-not-between" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::TextLength;
|
|
d->extra()->validity->m_cond = Conditional::Different;
|
|
valExpression = valExpression.remove( "oooc:cell-content-text-length-is-not-between(" );
|
|
kdDebug()<<" valExpression :"<<valExpression<<endl;
|
|
valExpression = valExpression.remove( ")" );
|
|
kdDebug()<<" valExpression :"<<valExpression<<endl;
|
|
TQStringList listVal = TQStringList::split( ",", valExpression );
|
|
loadOasisValidationValue( listVal );
|
|
}
|
|
else if ( valExpression.contains( "cell-content-is-in-list(" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::List;
|
|
valExpression = valExpression.remove( "oooc:cell-content-is-in-list(" );
|
|
kdDebug()<<" valExpression :"<<valExpression<<endl;
|
|
valExpression = valExpression.remove( ")" );
|
|
d->extra()->validity->listValidity = TQStringList::split( ";", valExpression );
|
|
|
|
}
|
|
//TrueFunction ::= cell-content-is-whole-number() | cell-content-is-decimal-number() | cell-content-is-date() | cell-content-is-time()
|
|
else
|
|
{
|
|
if (valExpression.contains( "cell-content-is-whole-number()" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::Number;
|
|
valExpression = valExpression.remove( "oooc:cell-content-is-whole-number() and " );
|
|
}
|
|
else if (valExpression.contains( "cell-content-is-decimal-number()" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::Integer;
|
|
valExpression = valExpression.remove( "oooc:cell-content-is-decimal-number() and " );
|
|
}
|
|
else if (valExpression.contains( "cell-content-is-date()" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::Date;
|
|
valExpression = valExpression.remove( "oooc:cell-content-is-date() and " );
|
|
}
|
|
else if (valExpression.contains( "cell-content-is-time()" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = Restriction::Time;
|
|
valExpression = valExpression.remove( "oooc:cell-content-is-time() and " );
|
|
}
|
|
kdDebug()<<"valExpression :"<<valExpression<<endl;
|
|
|
|
if ( valExpression.contains( "cell-content()" ) )
|
|
{
|
|
valExpression = valExpression.remove( "cell-content()" );
|
|
loadOasisValidationCondition( valExpression );
|
|
}
|
|
//GetFunction ::= cell-content-is-between(Value, Value) | cell-content-is-not-between(Value, Value)
|
|
//for the moment we support just int/double value, not text/date/time :(
|
|
if ( valExpression.contains( "cell-content-is-between(" ) )
|
|
{
|
|
valExpression = valExpression.remove( "cell-content-is-between(" );
|
|
valExpression = valExpression.remove( ")" );
|
|
TQStringList listVal = TQStringList::split( "," , valExpression );
|
|
loadOasisValidationValue( listVal );
|
|
d->extra()->validity->m_cond = Conditional::Between;
|
|
}
|
|
if ( valExpression.contains( "cell-content-is-not-between(" ) )
|
|
{
|
|
valExpression = valExpression.remove( "cell-content-is-not-between(" );
|
|
valExpression = valExpression.remove( ")" );
|
|
TQStringList listVal = TQStringList::split( ",", valExpression );
|
|
loadOasisValidationValue( listVal );
|
|
d->extra()->validity->m_cond = Conditional::Different;
|
|
}
|
|
}
|
|
}
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "allow-empty-cell" ) )
|
|
{
|
|
kdDebug()<<" element.hasAttribute( table:allow-empty-cell ) :"<<element.hasAttributeNS( KoXmlNS::table, "allow-empty-cell" )<<endl;
|
|
d->extra()->validity->allowEmptyCell = ( ( element.attributeNS( KoXmlNS::table, "allow-empty-cell", TQString() )=="true" ) ? true : false );
|
|
}
|
|
if ( element.hasAttributeNS( KoXmlNS::table, "base-cell-address" ) )
|
|
{
|
|
//todo what is it ?
|
|
}
|
|
|
|
TQDomElement help = KoDom::namedItemNS( element, KoXmlNS::table, "help-message" );
|
|
if ( !help.isNull() )
|
|
{
|
|
if ( help.hasAttributeNS( KoXmlNS::table, "title" ) )
|
|
{
|
|
kdDebug()<<"help.attribute( table:title ) :"<<help.attributeNS( KoXmlNS::table, "title", TQString() )<<endl;
|
|
d->extra()->validity->titleInfo = help.attributeNS( KoXmlNS::table, "title", TQString() );
|
|
}
|
|
if ( help.hasAttributeNS( KoXmlNS::table, "display" ) )
|
|
{
|
|
kdDebug()<<"help.attribute( table:display ) :"<<help.attributeNS( KoXmlNS::table, "display", TQString() )<<endl;
|
|
d->extra()->validity->displayValidationInformation = ( ( help.attributeNS( KoXmlNS::table, "display", TQString() )=="true" ) ? true : false );
|
|
}
|
|
TQDomElement attrText = KoDom::namedItemNS( help, KoXmlNS::text, "p" );
|
|
if ( !attrText.isNull() )
|
|
{
|
|
kdDebug()<<"help text :"<<attrText.text()<<endl;
|
|
d->extra()->validity->messageInfo = attrText.text();
|
|
}
|
|
}
|
|
|
|
TQDomElement error = KoDom::namedItemNS( element, KoXmlNS::table, "error-message" );
|
|
if ( !error.isNull() )
|
|
{
|
|
if ( error.hasAttributeNS( KoXmlNS::table, "title" ) )
|
|
d->extra()->validity->title = error.attributeNS( KoXmlNS::table, "title", TQString() );
|
|
if ( error.hasAttributeNS( KoXmlNS::table, "message-type" ) )
|
|
{
|
|
TQString str = error.attributeNS( KoXmlNS::table, "message-type", TQString() );
|
|
if ( str == "warning" )
|
|
d->extra()->validity->m_action = Action::Warning;
|
|
else if ( str == "information" )
|
|
d->extra()->validity->m_action = Action::Information;
|
|
else if ( str == "stop" )
|
|
d->extra()->validity->m_action = Action::Stop;
|
|
else
|
|
kdDebug()<<"validation : message type unknown :"<<str<<endl;
|
|
}
|
|
|
|
if ( error.hasAttributeNS( KoXmlNS::table, "display" ) )
|
|
{
|
|
kdDebug()<<" display message :"<<error.attributeNS( KoXmlNS::table, "display", TQString() )<<endl;
|
|
d->extra()->validity->displayMessage = (error.attributeNS( KoXmlNS::table, "display", TQString() )=="true");
|
|
}
|
|
TQDomElement attrText = KoDom::namedItemNS( error, KoXmlNS::text, "p" );
|
|
if ( !attrText.isNull() )
|
|
d->extra()->validity->message = attrText.text();
|
|
}
|
|
}
|
|
|
|
|
|
void Cell::loadOasisValidationValue( const TQStringList &listVal )
|
|
{
|
|
bool ok = false;
|
|
kdDebug()<<" listVal[0] :"<<listVal[0]<<" listVal[1] :"<<listVal[1]<<endl;
|
|
|
|
if ( d->extra()->validity->m_restriction == Restriction::Date )
|
|
{
|
|
d->extra()->validity->dateMin = TQDate::fromString( listVal[0] );
|
|
d->extra()->validity->dateMax = TQDate::fromString( listVal[1] );
|
|
}
|
|
else if ( d->extra()->validity->m_restriction == Restriction::Time )
|
|
{
|
|
d->extra()->validity->timeMin = TQTime::fromString( listVal[0] );
|
|
d->extra()->validity->timeMax = TQTime::fromString( listVal[1] );
|
|
}
|
|
else
|
|
{
|
|
d->extra()->validity->valMin = listVal[0].toDouble(&ok);
|
|
if ( !ok )
|
|
{
|
|
d->extra()->validity->valMin = listVal[0].toInt(&ok);
|
|
if ( !ok )
|
|
kdDebug()<<" Try to parse this value :"<<listVal[0]<<endl;
|
|
|
|
#if 0
|
|
if ( !ok )
|
|
d->extra()->validity->valMin = listVal[0];
|
|
#endif
|
|
}
|
|
ok=false;
|
|
d->extra()->validity->valMax = listVal[1].toDouble(&ok);
|
|
if ( !ok )
|
|
{
|
|
d->extra()->validity->valMax = listVal[1].toInt(&ok);
|
|
if ( !ok )
|
|
kdDebug()<<" Try to parse this value :"<<listVal[1]<<endl;
|
|
|
|
#if 0
|
|
if ( !ok )
|
|
d->extra()->validity->valMax = listVal[1];
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
void Cell::loadOasisValidationCondition( TQString &valExpression )
|
|
{
|
|
TQString value;
|
|
if (valExpression.find( "<=" )==0 )
|
|
{
|
|
value = valExpression.remove( 0,2 );
|
|
d->extra()->validity->m_cond = Conditional::InferiorEqual;
|
|
}
|
|
else if (valExpression.find( ">=" )==0 )
|
|
{
|
|
value = valExpression.remove( 0,2 );
|
|
d->extra()->validity->m_cond = Conditional::SuperiorEqual;
|
|
}
|
|
else if (valExpression.find( "!=" )==0 )
|
|
{
|
|
//add Differentto attribute
|
|
value = valExpression.remove( 0,2 );
|
|
d->extra()->validity->m_cond = Conditional::DifferentTo;
|
|
}
|
|
else if ( valExpression.find( "<" )==0 )
|
|
{
|
|
value = valExpression.remove( 0,1 );
|
|
d->extra()->validity->m_cond = Conditional::Inferior;
|
|
}
|
|
else if(valExpression.find( ">" )==0 )
|
|
{
|
|
value = valExpression.remove( 0,1 );
|
|
d->extra()->validity->m_cond = Conditional::Superior;
|
|
}
|
|
else if (valExpression.find( "=" )==0 )
|
|
{
|
|
value = valExpression.remove( 0,1 );
|
|
d->extra()->validity->m_cond = Conditional::Equal;
|
|
}
|
|
else
|
|
kdDebug()<<" I don't know how to parse it :"<<valExpression<<endl;
|
|
if ( d->extra()->validity->m_restriction == Restriction::Date )
|
|
{
|
|
d->extra()->validity->dateMin = TQDate::fromString( value );
|
|
}
|
|
else if (d->extra()->validity->m_restriction == Restriction::Date )
|
|
{
|
|
d->extra()->validity->timeMin = TQTime::fromString( value );
|
|
}
|
|
else
|
|
{
|
|
bool ok = false;
|
|
d->extra()->validity->valMin = value.toDouble(&ok);
|
|
if ( !ok )
|
|
{
|
|
d->extra()->validity->valMin = value.toInt(&ok);
|
|
if ( !ok )
|
|
kdDebug()<<" Try to parse this value :"<<value<<endl;
|
|
|
|
#if 0
|
|
if ( !ok )
|
|
d->extra()->validity->valMin = value;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
bool Cell::load( const TQDomElement & cell, int _xshift, int _yshift,
|
|
Paste::Mode pm, Paste::Operation op, bool paste )
|
|
{
|
|
bool ok;
|
|
|
|
//
|
|
// First of all determine in which row and column this
|
|
// cell belongs.
|
|
//
|
|
d->row = cell.attribute( "row" ).toInt( &ok ) + _yshift;
|
|
if ( !ok ) return false;
|
|
d->column = cell.attribute( "column" ).toInt( &ok ) + _xshift;
|
|
if ( !ok ) return false;
|
|
|
|
// Validation
|
|
if ( d->row < 1 || d->row > KS_rowMax )
|
|
{
|
|
kdDebug(36001) << "Cell::load: Value out of Range Cell:row=" << d->row << endl;
|
|
return false;
|
|
}
|
|
if ( d->column < 1 || d->column > KS_colMax )
|
|
{
|
|
kdDebug(36001) << "Cell::load: Value out of Range Cell:column=" << d->column << endl;
|
|
return false;
|
|
}
|
|
|
|
//
|
|
// Load formatting information.
|
|
//
|
|
TQDomElement f = cell.namedItem( "format" ).toElement();
|
|
if ( !f.isNull()
|
|
&& ( (pm == Paste::Normal) || (pm == Paste::Format) || (pm == Paste::NoBorder) ) )
|
|
{
|
|
// send pm parameter. Didn't load Borders if pm==NoBorder
|
|
|
|
if ( !format()->load( f, pm, paste ) )
|
|
return false;
|
|
|
|
if ( f.hasAttribute( "colspan" ) )
|
|
{
|
|
int i = f.attribute("colspan").toInt( &ok );
|
|
if ( !ok ) return false;
|
|
// Validation
|
|
if ( i < 0 || i > KS_spanMax )
|
|
{
|
|
kdDebug(36001) << "Value out of range Cell::colspan=" << i << endl;
|
|
return false;
|
|
}
|
|
if (i || d->hasExtra())
|
|
d->extra()->extraXCells = i;
|
|
if ( i > 0 )
|
|
{
|
|
setFlag(Flag_Merged);
|
|
}
|
|
}
|
|
|
|
if ( f.hasAttribute( "rowspan" ) )
|
|
{
|
|
int i = f.attribute("rowspan").toInt( &ok );
|
|
if ( !ok ) return false;
|
|
// Validation
|
|
if ( i < 0 || i > KS_spanMax )
|
|
{
|
|
kdDebug(36001) << "Value out of range Cell::rowspan=" << i << endl;
|
|
return false;
|
|
}
|
|
if (i || d->hasExtra())
|
|
d->extra()->extraYCells = i;
|
|
if ( i > 0 )
|
|
{
|
|
setFlag(Flag_Merged);
|
|
}
|
|
}
|
|
|
|
if ( testFlag( Flag_Merged ) )
|
|
{
|
|
if (d->hasExtra())
|
|
mergeCells( d->column, d->row, d->extra()->extraXCells, d->extra()->extraYCells );
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// Load the condition section of a cell.
|
|
//
|
|
TQDomElement conditionsElement = cell.namedItem( "condition" ).toElement();
|
|
if ( !conditionsElement.isNull())
|
|
{
|
|
if (d->hasExtra())
|
|
delete d->extra()->conditions;
|
|
d->extra()->conditions = new Conditions( this );
|
|
d->extra()->conditions->loadConditions( conditionsElement );
|
|
d->extra()->conditions->checkMatches();
|
|
}
|
|
else if ((pm == Paste::Normal) || (pm == Paste::NoBorder))
|
|
{
|
|
//clear the conditional formatting
|
|
if (d->hasExtra())
|
|
{
|
|
delete d->extra()->conditions;
|
|
d->extra()->conditions = 0;
|
|
}
|
|
}
|
|
|
|
TQDomElement validity = cell.namedItem( "validity" ).toElement();
|
|
if ( !validity.isNull())
|
|
{
|
|
TQDomElement param = validity.namedItem( "param" ).toElement();
|
|
if(!param.isNull())
|
|
{
|
|
d->extra()->validity = new Validity;
|
|
if ( param.hasAttribute( "cond" ) )
|
|
{
|
|
d->extra()->validity->m_cond = (Conditional::Type) param.attribute("cond").toInt( &ok );
|
|
if ( !ok )
|
|
return false;
|
|
}
|
|
if ( param.hasAttribute( "action" ) )
|
|
{
|
|
d->extra()->validity->m_action = (Action::Type) param.attribute("action").toInt( &ok );
|
|
if ( !ok )
|
|
return false;
|
|
}
|
|
if ( param.hasAttribute( "allow" ) )
|
|
{
|
|
d->extra()->validity->m_restriction = (Restriction::Type) param.attribute("allow").toInt( &ok );
|
|
if ( !ok )
|
|
return false;
|
|
}
|
|
if ( param.hasAttribute( "valmin" ) )
|
|
{
|
|
d->extra()->validity->valMin = param.attribute("valmin").toDouble( &ok );
|
|
if ( !ok )
|
|
return false;
|
|
}
|
|
if ( param.hasAttribute( "valmax" ) )
|
|
{
|
|
d->extra()->validity->valMax = param.attribute("valmax").toDouble( &ok );
|
|
if ( !ok )
|
|
return false;
|
|
}
|
|
if ( param.hasAttribute( "displaymessage" ) )
|
|
{
|
|
d->extra()->validity->displayMessage = ( bool )param.attribute("displaymessage").toInt();
|
|
}
|
|
if ( param.hasAttribute( "displayvalidationinformation" ) )
|
|
{
|
|
d->extra()->validity->displayValidationInformation = ( bool )param.attribute("displayvalidationinformation").toInt();
|
|
}
|
|
if ( param.hasAttribute( "allowemptycell" ) )
|
|
{
|
|
d->extra()->validity->allowEmptyCell = ( bool )param.attribute("allowemptycell").toInt();
|
|
}
|
|
if ( param.hasAttribute("listvalidity") )
|
|
{
|
|
d->extra()->validity->listValidity=TQStringList::split(";", param.attribute("listvalidity") );
|
|
}
|
|
}
|
|
TQDomElement inputTitle = validity.namedItem( "inputtitle" ).toElement();
|
|
if (!inputTitle.isNull())
|
|
{
|
|
d->extra()->validity->titleInfo = inputTitle.text();
|
|
}
|
|
TQDomElement inputMessage = validity.namedItem( "inputmessage" ).toElement();
|
|
if (!inputMessage.isNull())
|
|
{
|
|
d->extra()->validity->messageInfo = inputMessage.text();
|
|
}
|
|
|
|
TQDomElement title = validity.namedItem( "title" ).toElement();
|
|
if (!title.isNull())
|
|
{
|
|
d->extra()->validity->title = title.text();
|
|
}
|
|
TQDomElement message = validity.namedItem( "message" ).toElement();
|
|
if (!message.isNull())
|
|
{
|
|
d->extra()->validity->message = message.text();
|
|
}
|
|
TQDomElement timeMin = validity.namedItem( "timemin" ).toElement();
|
|
if ( !timeMin.isNull() )
|
|
{
|
|
d->extra()->validity->timeMin = toTime(timeMin);
|
|
}
|
|
TQDomElement timeMax = validity.namedItem( "timemax" ).toElement();
|
|
if ( !timeMax.isNull() )
|
|
{
|
|
d->extra()->validity->timeMax = toTime(timeMax);
|
|
}
|
|
TQDomElement dateMin = validity.namedItem( "datemin" ).toElement();
|
|
if ( !dateMin.isNull() )
|
|
{
|
|
d->extra()->validity->dateMin = toDate(dateMin);
|
|
}
|
|
TQDomElement dateMax = validity.namedItem( "datemax" ).toElement();
|
|
if ( !dateMax.isNull() )
|
|
{
|
|
d->extra()->validity->dateMax = toDate(dateMax);
|
|
}
|
|
}
|
|
else if ((pm == Paste::Normal) || (pm == Paste::NoBorder))
|
|
{
|
|
// clear the validity
|
|
removeValidity();
|
|
}
|
|
|
|
//
|
|
// Load the comment
|
|
//
|
|
TQDomElement comment = cell.namedItem( "comment" ).toElement();
|
|
if ( !comment.isNull() && ( pm == Paste::Normal || pm == Paste::Comment || pm == Paste::NoBorder ))
|
|
{
|
|
TQString t = comment.text();
|
|
//t = t.stripWhiteSpace();
|
|
format()->setComment( t );
|
|
}
|
|
|
|
//
|
|
// The real content of the cell is loaded here. It is stored in
|
|
// the "text" tag, which contains either a text or a CDATA section.
|
|
//
|
|
// TODO: make this suck less. We set data twice, in loadCellData, and
|
|
// also here. Not good.
|
|
TQDomElement text = cell.namedItem( "text" ).toElement();
|
|
|
|
if ( !text.isNull() &&
|
|
( pm == Paste::Normal || pm == Paste::Text || pm == Paste::NoBorder || pm == Paste::Result ) )
|
|
{
|
|
/* older versions mistakenly put the datatype attribute on the cell
|
|
instead of the text. Just move it over in case we're parsing
|
|
an old document */
|
|
if ( cell.hasAttribute( "dataType" ) ) // new docs
|
|
text.setAttribute( "dataType", cell.attribute( "dataType" ) );
|
|
|
|
TQDomElement result = cell.namedItem( "result" ).toElement();
|
|
TQString txt = text.text();
|
|
if ((pm == Paste::Result) && (txt[0] == '='))
|
|
// paste text of the element, if we want to paste result
|
|
// and the source cell contains a formula
|
|
// note that we mustn't use setCellValue after this, or else we lose
|
|
// all the formulas ...
|
|
d->strText = result.text();
|
|
else
|
|
//otherwise copy everything
|
|
loadCellData(text, op);
|
|
|
|
if ( !result.isNull() )
|
|
{
|
|
TQString dataType;
|
|
TQString t = result.text();
|
|
|
|
if ( result.hasAttribute( "dataType" ) )
|
|
dataType = result.attribute( "dataType" );
|
|
if ( result.hasAttribute( "outStr" ) )
|
|
{
|
|
d->strOutText = result.attribute( "outStr" );
|
|
if ( !d->strOutText.isEmpty() )
|
|
clearFlag( Flag_TextFormatDirty );
|
|
}
|
|
|
|
bool clear = true;
|
|
// boolean ?
|
|
if( dataType == "Bool" )
|
|
{
|
|
if ( t == "false" )
|
|
setValue( false );
|
|
else if ( t == "true" )
|
|
setValue( true );
|
|
else
|
|
clear = false;
|
|
}
|
|
else if( dataType == "Num" )
|
|
{
|
|
bool ok = false;
|
|
double dd = t.toDouble( &ok );
|
|
if ( ok )
|
|
setValue ( dd );
|
|
else
|
|
clear = false;
|
|
}
|
|
else if( dataType == "Date" )
|
|
{
|
|
bool ok = false;
|
|
double dd = t.toDouble( &ok );
|
|
if ( ok )
|
|
setValue ( dd );
|
|
else
|
|
{
|
|
int pos = t.find( '/' );
|
|
int year = t.mid( 0, pos ).toInt();
|
|
int pos1 = t.find( '/', pos + 1 );
|
|
int month = t.mid( pos + 1, ( ( pos1 - 1 ) - pos ) ).toInt();
|
|
int day = t.right( t.length() - pos1 - 1 ).toInt();
|
|
TQDate date( year, month, day );
|
|
if ( date.isValid() )
|
|
setValue( date );
|
|
else
|
|
clear = false;
|
|
}
|
|
}
|
|
else if( dataType == "Time" )
|
|
{
|
|
bool ok = false;
|
|
double dd = t.toDouble( &ok );
|
|
if ( ok )
|
|
setCellValue( dd );
|
|
else
|
|
{
|
|
int hours = -1;
|
|
int minutes = -1;
|
|
int second = -1;
|
|
int pos, pos1;
|
|
pos = t.find( ':' );
|
|
hours = t.mid( 0, pos ).toInt();
|
|
pos1 = t.find( ':', pos + 1 );
|
|
minutes = t.mid( pos + 1, ( ( pos1 - 1 ) - pos ) ).toInt();
|
|
second = t.right( t.length() - pos1 - 1 ).toInt();
|
|
TQTime time( hours, minutes, second );
|
|
if ( time.isValid() )
|
|
setValue( time );
|
|
else
|
|
clear = false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
setValue( t );
|
|
}
|
|
|
|
// if ( clear )
|
|
// clearFlag( Flag_CalcDirty );
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Cell::loadCellData(const TQDomElement & text, Paste::Operation op )
|
|
{
|
|
//TODO: use converter()->asString() to generate strText
|
|
|
|
TQString t = text.text();
|
|
t = t.stripWhiteSpace();
|
|
|
|
setFlag(Flag_LayoutDirty);
|
|
setFlag(Flag_TextFormatDirty);
|
|
|
|
// A formula like =A1+A2 ?
|
|
if( t[0] == '=' )
|
|
{
|
|
t = decodeFormula( t, d->column, d->row );
|
|
setCellText (pasteOperation( t, d->strText, op ));
|
|
|
|
setFlag(Flag_CalcDirty);
|
|
clearAllErrors();
|
|
|
|
if ( !makeFormula() )
|
|
kdError(36001) << "ERROR: Syntax ERROR" << endl;
|
|
}
|
|
// rich text ?
|
|
else if (t[0] == '!' )
|
|
{
|
|
// KSpread pre 1.4 stores hyperlink as rich text (first char is '!')
|
|
// extract the link and the correspoding text
|
|
// This is a rather dirty hack, but enough for KSpread generated XML
|
|
bool inside_tag = false;
|
|
TQString qml_text;
|
|
TQString tag;
|
|
TQString qml_link;
|
|
|
|
for( unsigned i = 1; i < t.length(); i++ )
|
|
{
|
|
TQChar ch = t[i];
|
|
if( ch == '<' )
|
|
{
|
|
if( !inside_tag )
|
|
{
|
|
inside_tag = true;
|
|
tag = TQString();
|
|
}
|
|
}
|
|
else if( ch == '>' )
|
|
{
|
|
if( inside_tag )
|
|
{
|
|
inside_tag = false;
|
|
if( tag.startsWith( "a href=\"", true ) )
|
|
if( tag.endsWith( "\"" ) )
|
|
qml_link = tag.mid( 8, tag.length()-9 );
|
|
tag = TQString();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( !inside_tag )
|
|
qml_text += ch;
|
|
else
|
|
tag += ch;
|
|
}
|
|
}
|
|
|
|
if( !qml_link.isEmpty() )
|
|
d->extra()->link = qml_link;
|
|
d->strText = qml_text;
|
|
setValue( d->strText );
|
|
}
|
|
else
|
|
{
|
|
bool newStyleLoading = true;
|
|
TQString dataType;
|
|
|
|
if ( text.hasAttribute( "dataType" ) ) // new docs
|
|
{
|
|
dataType = text.attribute( "dataType" );
|
|
}
|
|
else // old docs: do the ugly solution of calling checkTextInput to parse the text
|
|
{
|
|
// ...except for date/time
|
|
if (isDate() && ( t.contains('/') == 2 ))
|
|
dataType = "Date";
|
|
else if (isTime() && ( t.contains(':') == 2 ) )
|
|
dataType = "Time";
|
|
else
|
|
{
|
|
d->strText = pasteOperation( t, d->strText, op );
|
|
checkTextInput();
|
|
//kdDebug(36001) << "Cell::load called checkTextInput, got dataType=" << dataType << " t=" << t << endl;
|
|
newStyleLoading = false;
|
|
}
|
|
}
|
|
|
|
if ( newStyleLoading )
|
|
{
|
|
d->value = Value::empty();
|
|
clearAllErrors();
|
|
|
|
// boolean ?
|
|
if( dataType == "Bool" )
|
|
{
|
|
bool val = (t.lower() == "true");
|
|
setCellValue (val);
|
|
}
|
|
|
|
// number ?
|
|
else if( dataType == "Num" )
|
|
{
|
|
bool ok = false;
|
|
if (t.contains('.'))
|
|
setValue ( Value( t.toDouble(&ok) ) ); // We save in non-localized format
|
|
else
|
|
setValue ( Value( t.toLong(&ok) ) );
|
|
if ( !ok )
|
|
{
|
|
kdWarning(36001) << "Couldn't parse '" << t << "' as number." << endl;
|
|
}
|
|
/* We will need to localize the text version of the number */
|
|
TDELocale* locale = format()->sheet()->doc()->locale();
|
|
|
|
/* TDELocale::formatNumber requires the precision we want to return.
|
|
*/
|
|
int precision = t.length() - t.find('.') - 1;
|
|
|
|
if ( formatType() == Percentage_format )
|
|
{
|
|
if (value().isInteger())
|
|
t = locale->formatNumber( value().asInteger() * 100 );
|
|
else
|
|
t = locale->formatNumber( value().asFloat() * 100.0, precision );
|
|
d->strText = pasteOperation( t, d->strText, op );
|
|
d->strText += '%';
|
|
}
|
|
else
|
|
{
|
|
if (value().isInteger())
|
|
t = locale->formatLong(value().asInteger());
|
|
else
|
|
t = locale->formatNumber(value().asFloat(), precision);
|
|
d->strText = pasteOperation( t, d->strText, op );
|
|
}
|
|
}
|
|
|
|
// date ?
|
|
else if( dataType == "Date" )
|
|
{
|
|
int pos = t.find('/');
|
|
int year = t.mid(0,pos).toInt();
|
|
int pos1 = t.find('/',pos+1);
|
|
int month = t.mid(pos+1,((pos1-1)-pos)).toInt();
|
|
int day = t.right(t.length()-pos1-1).toInt();
|
|
setValue( TQDate(year,month,day) );
|
|
if ( value().asDate().isValid() ) // Should always be the case for new docs
|
|
d->strText = locale()->formatDate( value().asDate(), true );
|
|
else // This happens with old docs, when format is set wrongly to date
|
|
{
|
|
d->strText = pasteOperation( t, d->strText, op );
|
|
checkTextInput();
|
|
}
|
|
}
|
|
|
|
// time ?
|
|
else if( dataType == "Time" )
|
|
{
|
|
int hours = -1;
|
|
int minutes = -1;
|
|
int second = -1;
|
|
int pos, pos1;
|
|
pos = t.find(':');
|
|
hours = t.mid(0,pos).toInt();
|
|
pos1 = t.find(':',pos+1);
|
|
minutes = t.mid(pos+1,((pos1-1)-pos)).toInt();
|
|
second = t.right(t.length()-pos1-1).toInt();
|
|
setValue( TQTime(hours,minutes,second) );
|
|
if ( value().asTime().isValid() ) // Should always be the case for new docs
|
|
d->strText = locale()->formatTime( value().asTime(), true );
|
|
else // This happens with old docs, when format is set wrongly to time
|
|
{
|
|
d->strText = pasteOperation( t, d->strText, op );
|
|
checkTextInput();
|
|
}
|
|
}
|
|
|
|
else
|
|
{
|
|
// Set the cell's text
|
|
d->strText = pasteOperation( t, d->strText, op );
|
|
setValue( d->strText );
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( text.hasAttribute( "outStr" ) ) // very new docs
|
|
{
|
|
d->strOutText = text.attribute( "outStr" );
|
|
if ( !d->strOutText.isEmpty() )
|
|
clearFlag( Flag_TextFormatDirty );
|
|
}
|
|
|
|
if ( !format()->sheet()->isLoading() )
|
|
setCellText( d->strText );
|
|
|
|
if ( d->hasExtra() && d->extra()->conditions )
|
|
d->extra()->conditions->checkMatches();
|
|
|
|
return true;
|
|
}
|
|
|
|
TQTime Cell::toTime(const TQDomElement &element)
|
|
{
|
|
//TODO: can't we use tryParseTime (after modification) instead?
|
|
TQString t = element.text();
|
|
t = t.stripWhiteSpace();
|
|
int hours = -1;
|
|
int minutes = -1;
|
|
int second = -1;
|
|
int pos, pos1;
|
|
pos = t.find(':');
|
|
hours = t.mid(0,pos).toInt();
|
|
pos1 = t.find(':',pos+1);
|
|
minutes = t.mid(pos+1,((pos1-1)-pos)).toInt();
|
|
second = t.right(t.length()-pos1-1).toInt();
|
|
setValue( Value( TQTime(hours,minutes,second)) );
|
|
return value().asTime();
|
|
}
|
|
|
|
TQDate Cell::toDate(const TQDomElement &element)
|
|
{
|
|
TQString t = element.text();
|
|
int pos;
|
|
int pos1;
|
|
int year = -1;
|
|
int month = -1;
|
|
int day = -1;
|
|
pos = t.find('/');
|
|
year = t.mid(0,pos).toInt();
|
|
pos1 = t.find('/',pos+1);
|
|
month = t.mid(pos+1,((pos1-1)-pos)).toInt();
|
|
day = t.right(t.length()-pos1-1).toInt();
|
|
setValue( Value( TQDate(year,month,day) ) );
|
|
return value().asDate();
|
|
}
|
|
|
|
TQString Cell::pasteOperation( const TQString &new_text, const TQString &old_text, Paste::Operation op )
|
|
{
|
|
if ( op == Paste::OverWrite )
|
|
return new_text;
|
|
|
|
TQString tmp_op;
|
|
TQString tmp;
|
|
TQString old;
|
|
|
|
if( !new_text.isEmpty() && new_text[0] == '=' )
|
|
{
|
|
tmp = new_text.right( new_text.length() - 1 );
|
|
}
|
|
else
|
|
{
|
|
tmp = new_text;
|
|
}
|
|
|
|
if ( old_text.isEmpty() &&
|
|
( op == Paste::Add || op == Paste::Mul || op == Paste::Sub || op == Paste::Div ) )
|
|
{
|
|
old = "=0";
|
|
}
|
|
|
|
if( !old_text.isEmpty() && old_text[0] == '=' )
|
|
{
|
|
old = old_text.right( old_text.length() - 1 );
|
|
}
|
|
else
|
|
{
|
|
old = old_text;
|
|
}
|
|
|
|
bool b1, b2;
|
|
tmp.toDouble( &b1 );
|
|
old.toDouble( &b2 );
|
|
if (b1 && !b2 && old.length() == 0)
|
|
{
|
|
old = "0";
|
|
b2 = true;
|
|
}
|
|
|
|
if( b1 && b2 )
|
|
{
|
|
switch( op )
|
|
{
|
|
case Paste::Add:
|
|
tmp_op = TQString::number(old.toDouble()+tmp.toDouble());
|
|
break;
|
|
case Paste::Mul :
|
|
tmp_op = TQString::number(old.toDouble()*tmp.toDouble());
|
|
break;
|
|
case Paste::Sub:
|
|
tmp_op = TQString::number(old.toDouble()-tmp.toDouble());
|
|
break;
|
|
case Paste::Div:
|
|
tmp_op = TQString::number(old.toDouble()/tmp.toDouble());
|
|
break;
|
|
default:
|
|
Q_ASSERT( 0 );
|
|
}
|
|
|
|
setFlag(Flag_LayoutDirty);
|
|
clearAllErrors();
|
|
|
|
return tmp_op;
|
|
}
|
|
else if ( ( new_text[0] == '=' && old_text[0] == '=' ) ||
|
|
( b1 && old_text[0] == '=' ) || ( new_text[0] == '=' && b2 ) )
|
|
{
|
|
switch( op )
|
|
{
|
|
case Paste::Add :
|
|
tmp_op="=("+old+")+"+"("+tmp+")";
|
|
break;
|
|
case Paste::Mul :
|
|
tmp_op="=("+old+")*"+"("+tmp+")";
|
|
break;
|
|
case Paste::Sub:
|
|
tmp_op="=("+old+")-"+"("+tmp+")";
|
|
break;
|
|
case Paste::Div:
|
|
tmp_op="=("+old+")/"+"("+tmp+")";
|
|
break;
|
|
default :
|
|
Q_ASSERT( 0 );
|
|
}
|
|
|
|
tmp_op = decodeFormula( tmp_op, d->column, d->row );
|
|
setFlag(Flag_LayoutDirty);
|
|
clearAllErrors();
|
|
|
|
return tmp_op;
|
|
}
|
|
|
|
tmp = decodeFormula( new_text, d->column, d->row );
|
|
setFlag(Flag_LayoutDirty);
|
|
clearAllErrors();
|
|
|
|
return tmp;
|
|
}
|
|
|
|
TQString Cell::testAnchor( int x, int y ) const
|
|
{
|
|
if( link().isEmpty() )
|
|
return TQString();
|
|
|
|
const Doc* doc = format()->sheet()->doc();
|
|
int x1 = doc->zoomItX( d->textX );
|
|
int y1 = doc->zoomItX( d->textY - d->textHeight );
|
|
int x2 = doc->zoomItX( d->textX + d->textWidth );
|
|
int y2 = doc->zoomItX( d->textY );
|
|
|
|
if( x > x1 ) if( x < x2 )
|
|
if( y > y1 ) if( y < y2 )
|
|
return link();
|
|
|
|
return TQString();
|
|
}
|
|
|
|
void Cell::sheetDies()
|
|
{
|
|
// Avoid unobscuring the cells in the destructor.
|
|
if (d->hasExtra())
|
|
{
|
|
d->extra()->extraXCells = 0;
|
|
d->extra()->extraYCells = 0;
|
|
d->extra()->mergedXCells = 0;
|
|
d->extra()->mergedYCells = 0;
|
|
}
|
|
|
|
//d->nextCell = 0;
|
|
//d->previousCell = 0;
|
|
}
|
|
|
|
Cell::~Cell()
|
|
{
|
|
if ( d->nextCell )
|
|
d->nextCell->setPreviousCell( d->previousCell );
|
|
if ( d->previousCell )
|
|
d->previousCell->setNextCell( d->nextCell );
|
|
|
|
if (d->hasExtra())
|
|
{
|
|
delete d->extra()->validity;
|
|
}
|
|
|
|
// Unobscure cells.
|
|
int extraXCells = d->hasExtra() ? d->extra()->extraXCells : 0;
|
|
int extraYCells = d->hasExtra() ? d->extra()->extraYCells : 0;
|
|
for( int x = 0; x <= extraXCells; ++x )
|
|
for( int y = (x == 0) ? 1 : 0; // avoid looking at (+0,+0)
|
|
y <= extraYCells; ++y )
|
|
{
|
|
Cell* cell = format()->sheet()->cellAt( d->column + x, d->row + y );
|
|
if ( cell )
|
|
cell->unobscure(this);
|
|
}
|
|
|
|
d->value = Value::empty();
|
|
|
|
if (!isDefault())
|
|
valueChanged (); //our value has been changed (is now null), but only if we aren't default
|
|
|
|
delete d->format;
|
|
delete d;
|
|
}
|
|
|
|
bool Cell::operator > ( const Cell & cell ) const
|
|
{
|
|
if ( value().isNumber() ) // ### what about bools ?
|
|
{
|
|
if ( cell.value().isNumber() )
|
|
return value().asFloat() > cell.value().asFloat();
|
|
else
|
|
return false; // numbers are always < than texts
|
|
}
|
|
else if(isDate())
|
|
{
|
|
if( cell.isDate() )
|
|
return value().asDate() > cell.value().asDate();
|
|
else if (cell.value().isNumber())
|
|
return true;
|
|
else
|
|
return false; //date are always < than texts and time
|
|
}
|
|
else if(isTime())
|
|
{
|
|
if( cell.isTime() )
|
|
return value().asTime() > cell.value().asTime();
|
|
else if( cell.isDate())
|
|
return true; //time are always > than date
|
|
else if( cell.value().isNumber())
|
|
return true;
|
|
else
|
|
return false; //time are always < than texts
|
|
}
|
|
else
|
|
{
|
|
if ( Map::respectCase )
|
|
return value().asString().compare(cell.value().asString()) > 0;
|
|
else
|
|
return ( value().asString() ).lower().compare(cell.value().asString().lower()) > 0;
|
|
}
|
|
}
|
|
|
|
bool Cell::operator < ( const Cell & cell ) const
|
|
{
|
|
if ( value().isNumber() )
|
|
{
|
|
if ( cell.value().isNumber() )
|
|
return value().asFloat() < cell.value().asFloat();
|
|
else
|
|
return true; // numbers are always < than texts
|
|
}
|
|
else if(isDate())
|
|
{
|
|
if( cell.isDate() )
|
|
return value().asDateTime().date() < cell.value().asDateTime().date();
|
|
else if( cell.value().isNumber())
|
|
return false;
|
|
else
|
|
return true; //date are always < than texts and time
|
|
}
|
|
else if(isTime())
|
|
{
|
|
if( cell.isTime() )
|
|
return value().asDateTime().time() < cell.value().asDateTime().time();
|
|
else if(cell.isDate())
|
|
return false; //time are always > than date
|
|
else if( cell.value().isNumber())
|
|
return false;
|
|
else
|
|
return true; //time are always < than texts
|
|
}
|
|
else
|
|
{
|
|
if ( Map::respectCase )
|
|
return value().asString().compare(cell.value().asString()) < 0;
|
|
else
|
|
return ( value().asString() ).lower().compare(cell.value().asString().lower()) < 0;
|
|
}
|
|
}
|
|
|
|
bool Cell::operator==( const Cell& other ) const
|
|
{
|
|
if ( d->strText != other.d->strText )
|
|
return false;
|
|
if ( d->value != other.d->value )
|
|
return false;
|
|
if ( *d->format != *other.d->format )
|
|
return false;
|
|
if ( d->hasExtra() )
|
|
{
|
|
if ( !other.d->hasExtra() )
|
|
return false;
|
|
if ( d->extra()->link != other.d->extra()->link )
|
|
return false;
|
|
if ( d->extra()->mergedXCells != other.d->extra()->mergedXCells )
|
|
return false;
|
|
if ( d->extra()->mergedYCells != other.d->extra()->mergedYCells )
|
|
return false;
|
|
if ( d->extra()->conditions )
|
|
{
|
|
if ( !other.d->extra()->conditions )
|
|
return false;
|
|
if ( *d->extra()->conditions != *other.d->extra()->conditions )
|
|
return false;
|
|
}
|
|
if ( d->extra()->validity )
|
|
{
|
|
if ( !other.d->extra()->validity )
|
|
return false;
|
|
if ( *d->extra()->validity != *other.d->extra()->validity )
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
TQRect Cell::cellRect()
|
|
{
|
|
Q_ASSERT(!isDefault());
|
|
return TQRect(TQPoint(d->column, d->row), TQPoint(d->column, d->row));
|
|
}
|
|
|
|
TQValueList<Conditional> Cell::conditionList() const
|
|
{
|
|
if ( !d->hasExtra() || !d->extra()->conditions )
|
|
{
|
|
TQValueList<Conditional> emptyList;
|
|
return emptyList;
|
|
}
|
|
|
|
return d->extra()->conditions->conditionList();
|
|
}
|
|
|
|
void Cell::setConditionList( const TQValueList<Conditional> & newList )
|
|
{
|
|
if (d->hasExtra())
|
|
delete d->extra()->conditions;
|
|
d->extra()->conditions = new Conditions( this );
|
|
d->extra()->conditions->setConditionList( newList );
|
|
d->extra()->conditions->checkMatches();
|
|
}
|
|
|
|
bool Cell::hasError() const
|
|
{
|
|
return ( testFlag(Flag_ParseError) ||
|
|
testFlag(Flag_CircularCalculation) ||
|
|
testFlag(Flag_DependancyError));
|
|
}
|
|
|
|
void Cell::clearAllErrors()
|
|
{
|
|
clearFlag( Flag_ParseError );
|
|
clearFlag( Flag_CircularCalculation );
|
|
clearFlag( Flag_DependancyError );
|
|
}
|
|
|
|
bool Cell::calcDirtyFlag()
|
|
{
|
|
return isFormula() ? testFlag( Flag_CalcDirty ) : false;
|
|
}
|
|
|
|
bool Cell::layoutDirtyFlag() const
|
|
{
|
|
return testFlag( Flag_LayoutDirty );
|
|
}
|
|
|
|
void Cell::clearDisplayDirtyFlag()
|
|
{
|
|
clearFlag( Flag_DisplayDirty );
|
|
}
|
|
|
|
void Cell::setDisplayDirtyFlag()
|
|
{
|
|
setFlag( Flag_DisplayDirty );
|
|
}
|
|
|
|
bool Cell::doesMergeCells() const
|
|
{
|
|
return testFlag( Flag_Merged );
|
|
}
|
|
|
|
void Cell::clearFlag( CellFlags flag )
|
|
{
|
|
d->flags &= ~(TQ_UINT32)flag;
|
|
}
|
|
|
|
void Cell::setFlag( CellFlags flag )
|
|
{
|
|
d->flags |= (TQ_UINT32)flag;
|
|
}
|
|
|
|
bool Cell::testFlag( CellFlags flag ) const
|
|
{
|
|
return ( d->flags & (TQ_UINT32)flag );
|
|
}
|
|
|
|
|
|
void Cell::checkForNamedAreas( TQString & formula ) const
|
|
{
|
|
KSPLoadingInfo* loadinginfo = sheet()->doc()->loadingInfo();
|
|
if(! loadinginfo) {
|
|
kdDebug() << "Cell::checkForNamedAreas loadinginfo is NULL" << endl;
|
|
return;
|
|
}
|
|
|
|
int l = formula.length();
|
|
int i = 0;
|
|
TQString word;
|
|
int start = 0;
|
|
while ( i < l )
|
|
{
|
|
if ( formula[i].isLetterOrNumber() )
|
|
{
|
|
word += formula[i];
|
|
++i;
|
|
continue;
|
|
}
|
|
if ( !word.isEmpty() )
|
|
{
|
|
if ( loadinginfo->findWordInAreaList(word) )
|
|
{
|
|
formula = formula.replace( start, word.length(), "'" + word + "'" );
|
|
l = formula.length();
|
|
++i;
|
|
kdDebug() << "Formula: " << formula << ", L: " << l << ", i: " << i + 1 <<endl;
|
|
}
|
|
}
|
|
|
|
++i;
|
|
word = "";
|
|
start = i;
|
|
}
|
|
if ( !word.isEmpty() )
|
|
{
|
|
if ( loadinginfo->findWordInAreaList(word) )
|
|
{
|
|
formula = formula.replace( start, word.length(), "'" + word + "'" );
|
|
l = formula.length();
|
|
++i;
|
|
kdDebug() << "Formula: " << formula << ", L: " << l << ", i: " << i + 1 <<endl;
|
|
}
|
|
}
|
|
}
|