summaryrefslogtreecommitdiffstats
path: root/kspread/valueformatter.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kspread/valueformatter.cpp')
-rw-r--r--kspread/valueformatter.cpp649
1 files changed, 649 insertions, 0 deletions
diff --git a/kspread/valueformatter.cpp b/kspread/valueformatter.cpp
new file mode 100644
index 00000000..94b6b57c
--- /dev/null
+++ b/kspread/valueformatter.cpp
@@ -0,0 +1,649 @@
+/* This file is part of the KDE project
+ Copyright 2004 Tomas Mecir <mecirt@gmail.com>
+ Copyright (C) 1998-2004 KSpread Team <koffice-devel@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 "valueformatter.h"
+
+#include "kspread_cell.h"
+#include "kspread_locale.h"
+#include "kspread_util.h"
+#include "valueconverter.h"
+
+#include <kcalendarsystem.h>
+#include <kdebug.h>
+#include <tdelocale.h>
+
+#include <float.h>
+#include <math.h>
+
+using namespace KSpread;
+
+ValueFormatter::ValueFormatter (ValueConverter *conv) : converter( conv )
+{
+}
+
+TQString ValueFormatter::formatText (Cell *cell, FormatType fmtType)
+{
+ if (cell->hasError ())
+ return errorFormat (cell);
+
+ TQString str;
+
+ Format::FloatFormat floatFormat =
+ cell->format()->floatFormat (cell->column(), cell->row());
+ int precision = cell->format()->precision (cell->column(), cell->row());
+ TQString prefix = cell->format()->prefix (cell->column(), cell->row());
+ TQString postfix = cell->format()->postfix (cell->column(), cell->row());
+ Format::Currency currency;
+ bool valid = cell->format()->currencyInfo(currency);
+ TQString currencySymbol = valid ? currency.symbol : TQString();
+
+ return formatText (cell->value(), fmtType, precision,
+ floatFormat, prefix, postfix, currencySymbol);
+}
+
+TQString ValueFormatter::formatText (const Value &value,
+ FormatType fmtType, int precision, Format::FloatFormat floatFormat,
+ const TQString &prefix, const TQString &postfix, const TQString &currencySymbol)
+{
+ //if we have an array, use its first element
+ if (value.isArray())
+ return formatText (value.element (0, 0), fmtType, precision,
+ floatFormat, prefix, postfix, currencySymbol);
+
+ TQString str;
+
+ //step 1: determine formatting that will be used
+ fmtType = determineFormatting (value, fmtType);
+
+ //step 2: format the value !
+
+ //text
+ if (fmtType == Text_format)
+ {
+ str = converter->asString (value).asString();
+ if (!str.isEmpty() && str[0]=='\'' )
+ str = str.mid(1);
+ }
+
+ //date
+ else if (formatIsDate (fmtType))
+ str = dateFormat (value.asDate(), fmtType);
+
+ //time
+ else if (formatIsTime (fmtType))
+ str = timeFormat (value.asDateTime(), fmtType);
+
+ //fraction
+ else if (formatIsFraction (fmtType))
+ str = fractionFormat (value.asFloat(), fmtType);
+
+ //another
+ else
+ {
+ //some cell parameters ...
+ double v = converter->asFloat (value).asFloat();
+
+ // Always unsigned ?
+ if ((floatFormat == Format::AlwaysUnsigned) && (v < 0.0))
+ v *= -1.0;
+
+ // Make a string out of it.
+ str = createNumberFormat (v, precision, fmtType,
+ (floatFormat == Format::AlwaysSigned), currencySymbol);
+
+ // Remove trailing zeros and the decimal point if necessary
+ // unless the number has no decimal point
+ if (precision == -1)
+ {
+ TQChar decimal_point = converter->locale()->decimalSymbol()[0];
+ if ( decimal_point.isNull() )
+ decimal_point = '.';
+
+ removeTrailingZeros (str, decimal_point);
+ }
+ }
+
+ if (!prefix.isEmpty())
+ str = prefix + ' ' + str;
+
+ if( !postfix.isEmpty())
+ str += ' ' + postfix;
+
+ //kdDebug() << "ValueFormatter says: " << str << endl;
+ return str;
+}
+
+FormatType ValueFormatter::determineFormatting (const Value &value,
+ FormatType fmtType)
+{
+ //if the cell value is a string, then we want to display it as-is,
+ //no matter what, same if the cell is empty
+ if (value.isString () || (value.format() == Value::fmt_None))
+ return Text_format;
+ //same if we're supposed to display string, no matter what we actually got
+ if (fmtType == Text_format)
+ return Text_format;
+
+ //now, everything depends on whether the formatting is Generic or not
+ if (fmtType == Generic_format)
+ {
+ //here we decide based on value's format...
+ Value::Format fmt = value.format();
+ switch (fmt) {
+ case Value::fmt_None:
+ fmtType = Text_format;
+ break;
+ case Value::fmt_Boolean:
+ fmtType = Text_format;
+ break;
+ case Value::fmt_Number:
+ if (value.asFloat() > 1e+10)
+ fmtType = Scientific_format;
+ else
+ fmtType = Number_format;
+ break;
+ case Value::fmt_Percent:
+ fmtType = Percentage_format;
+ break;
+ case Value::fmt_Money:
+ fmtType = Money_format;
+ break;
+ case Value::fmt_DateTime:
+ fmtType = TextDate_format;
+ break;
+ case Value::fmt_Date:
+ fmtType = ShortDate_format;
+ break;
+ case Value::fmt_Time:
+ fmtType = Time_format;
+ break;
+ case Value::fmt_String:
+ //this should never happen
+ fmtType = Text_format;
+ break;
+ };
+ return fmtType;
+ }
+ else
+ {
+ //we'll mostly want to use the given formatting, the only exception
+ //being Boolean values
+
+ //TODO: is this correct? We may also want to convert bools to 1s and 0s
+ //if we want to display a number...
+
+ //TODO: what to do about Custom formatting? We don't support it as of now,
+ // but we'll have it ... one day, that is ...
+ if (value.isBoolean())
+ return Text_format;
+ else
+ return fmtType;
+ }
+}
+
+
+void ValueFormatter::removeTrailingZeros (TQString &str, TQChar decimal_point)
+{
+ if (str.find (decimal_point) < 0)
+ //no decimal point -> nothing to do
+ return;
+
+ int start = 0;
+ int cslen = converter->locale()->currencySymbol().length();
+ if (str.find ('%') != -1)
+ start = 2;
+ else if (str.find (converter->locale()->currencySymbol()) ==
+ ((int) (str.length() - cslen)))
+ start = cslen + 1;
+ else if ((start = str.find ('E')) != -1)
+ start = str.length() - start;
+ else
+ start = 0;
+
+ int i = str.length() - start;
+ bool bFinished = false;
+ while ( !bFinished && i > 0 )
+ {
+ TQChar ch = str[i - 1];
+ if (ch == '0')
+ str.remove (--i,1);
+ else
+ {
+ bFinished = true;
+ if (ch == decimal_point)
+ str.remove (--i, 1);
+ }
+ }
+}
+
+TQString ValueFormatter::createNumberFormat ( double value, int precision,
+ FormatType fmt, bool alwaysSigned, const TQString& currencySymbol)
+{
+ // if precision is -1, ask for a huge number of decimals, we'll remove
+ // the zeros later. Is 8 ok ?
+ // Stefan: No. Use maximum possible decimal precision (DBL_DIG) instead.
+ int p = (precision == -1) ? 8 : precision;
+ TQString localizedNumber;
+ int pos = 0;
+
+ //multiply value by 100 for percentage format
+ if (fmt == Percentage_format)
+ value *= 100;
+
+ // this will avoid displaying negative zero, i.e "-0.0000"
+ if( fabs( value ) < DBL_EPSILON ) value = 0.0;
+
+ // round the number, based on desired precision if not scientific is chosen
+ //(scientific has relative precision)
+ if( fmt != Scientific_format )
+ {
+ double m[] = { 1, 10, 100, 1e3, 1e4, 1e5, 1e6, 1e7, 1e8, 1e9, 1e10 };
+ double mm = (p > 10) ? pow(10.0,p) : m[p];
+ bool neg = value < 0;
+ value = floor( fabs(value)*mm + 0.5 ) / mm;
+ if( neg ) value = -value;
+ }
+
+ TQChar decimal_point;
+ switch (fmt)
+ {
+ case Number_format:
+ localizedNumber = converter->locale()->formatNumber(value, p);
+ break;
+ case Percentage_format:
+ localizedNumber = converter->locale()->formatNumber (value, p)+ " %";
+ break;
+ case Money_format:
+ localizedNumber = converter->locale()->formatMoney (value,
+ currencySymbol.isEmpty() ? converter->locale()->currencySymbol() : currencySymbol, p );
+ break;
+ case Scientific_format:
+ decimal_point = converter->locale()->decimalSymbol()[0];
+ localizedNumber = TQString::number (value, 'E', p);
+ if ((pos = localizedNumber.find ('.')) != -1)
+ localizedNumber = localizedNumber.replace (pos, 1, decimal_point);
+ break;
+ default :
+ //other formatting?
+ // This happens with Custom_format...
+ kdDebug(36001)<<"Wrong usage of ValueFormatter::createNumberFormat fmt=" << fmt << "\n";
+ break;
+ }
+
+ //prepend positive sign if needed
+ if (alwaysSigned && value >= 0 )
+ if (converter->locale()->positiveSign().isEmpty())
+ localizedNumber='+'+localizedNumber;
+
+ return localizedNumber;
+}
+
+TQString ValueFormatter::fractionFormat (double value, FormatType fmtType)
+{
+ double result = value - floor(value);
+ int index;
+ int limit = 0;
+
+ /* return w/o fraction part if not necessary */
+ if (result == 0)
+ return TQString::number(value);
+
+ switch (fmtType) {
+ case fraction_half:
+ index = 2;
+ break;
+ case fraction_quarter:
+ index = 4;
+ break;
+ case fraction_eighth:
+ index = 8;
+ break;
+ case fraction_sixteenth:
+ index = 16;
+ break;
+ case fraction_tenth:
+ index = 10;
+ break;
+ case fraction_hundredth:
+ index = 100;
+ break;
+ case fraction_one_digit:
+ index = 3;
+ limit = 9;
+ break;
+ case fraction_two_digits:
+ index = 4;
+ limit = 99;
+ break;
+ case fraction_three_digits:
+ index = 5;
+ limit = 999;
+ break;
+ default:
+ kdDebug(36001) << "Error in Fraction format\n";
+ return TQString::number(value);
+ break;
+ } /* switch */
+
+
+ /* handle halves, quarters, tenths, ... */
+
+ if (fmtType != fraction_three_digits
+ && fmtType != fraction_two_digits
+ && fmtType != fraction_one_digit) {
+ double calc = 0;
+ int index1 = 0;
+ double diff = result;
+ for (int i = 1; i <= index; i++) {
+ calc = i * 1.0 / index;
+ if (fabs(result - calc) < diff) {
+ index1 = i;
+ diff = fabs(result - calc);
+ }
+ }
+ if( index1 == 0 ) return TQString("%1").arg( floor(value) );
+ if( index1 == index ) return TQString("%1").arg( floor(value)+1 );
+ if( floor(value) == 0)
+ return TQString("%1/%2").arg( index1 ).arg( index );
+
+ return TQString("%1 %2/%3")
+ .arg( floor(value) )
+ .arg( index1 )
+ .arg( index );
+ }
+
+
+ /* handle fraction_one_digit, fraction_two_digit
+ * and fraction_three_digit style */
+
+ double precision, denominator, numerator;
+
+ do {
+ double val1 = result;
+ double val2 = rint(result);
+ double inter2 = 1;
+ double inter4, p, q;
+ inter4 = p = q = 0;
+
+ precision = pow(10.0, -index);
+ numerator = val2;
+ denominator = 1;
+
+ while (fabs(numerator/denominator - result) > precision) {
+ val1 = (1 / (val1 - val2));
+ val2 = rint(val1);
+ p = val2 * numerator + inter2;
+ q = val2 * denominator + inter4;
+ inter2 = numerator;
+ inter4 = denominator;
+ numerator = p;
+ denominator = q;
+ }
+ index--;
+ } while (fabs(denominator) > limit);
+
+ denominator = fabs(denominator);
+ numerator = fabs(numerator);
+
+ if (denominator == numerator)
+ return TQString().setNum(floor(value + 1));
+ else
+ {
+ if ( floor(value) == 0 )
+ return TQString("%1/%2").arg(numerator).arg(denominator);
+ else
+ return TQString("%1 %2/%3")
+ .arg(floor(value))
+ .arg(numerator)
+ .arg(denominator);
+ }
+}
+
+TQString ValueFormatter::timeFormat (const TQDateTime &dt, FormatType fmtType)
+{
+ if (fmtType == Time_format)
+ return converter->locale()->formatTime(dt.time(), false);
+
+ if (fmtType == SecondeTime_format)
+ return converter->locale()->formatTime(dt.time(), true);
+
+ int h = dt.time().hour();
+ int m = dt.time().minute();
+ int s = dt.time().second();
+
+ TQString hour = ( h < 10 ? "0" + TQString::number(h) : TQString::number(h) );
+ TQString minute = ( m < 10 ? "0" + TQString::number(m) : TQString::number(m) );
+ TQString second = ( s < 10 ? "0" + TQString::number(s) : TQString::number(s) );
+ bool pm = (h > 12);
+ TQString AMPM( pm ? i18n("PM"):i18n("AM") );
+
+ if (fmtType == Time_format1) { // 9 : 01 AM
+ return TQString("%1:%2 %3")
+ .arg((pm ? h - 12 : h),2)
+ .arg(minute,2)
+ .arg(AMPM);
+ }
+
+ if (fmtType == Time_format2) { //9:01:05 AM
+ return TQString("%1:%2:%3 %4")
+ .arg((pm ? h-12 : h),2)
+ .arg(minute,2)
+ .arg(second,2)
+ .arg(AMPM);
+ }
+
+ if (fmtType == Time_format3) {
+ return TQString("%1 %2 %3 %4 %5 %6") // 9 h 01 min 28 s
+ .arg(hour,2)
+ .arg(i18n("h"))
+ .arg(minute,2)
+ .arg(i18n("min"))
+ .arg(second,2)
+ .arg(i18n("s"));
+ }
+
+ if (fmtType == Time_format4) { // 9:01
+ return TQString("%1:%2").arg(hour, 2).arg(minute, 2);
+ }
+
+ if (fmtType == Time_format5) { // 9:01:12
+ return TQString("%1:%2:%3").arg(hour, 2).arg(minute, 2).arg(second, 2);
+ }
+
+ TQDate d1(dt.date());
+ TQDate d2( 1899, 12, 31 );
+ int d = d2.daysTo( d1 ) + 1;
+
+ h += d * 24;
+
+ if (fmtType == Time_format6)
+ { // [mm]:ss
+ m += (h * 60);
+ return TQString("%1:%2").arg(m, 1).arg(second, 2);
+ }
+ if (fmtType == Time_format7) { // [h]:mm:ss
+ return TQString("%1:%2:%3").arg(h, 1).arg(minute, 2).arg(second, 2);
+ }
+ if (fmtType == Time_format8)
+ { // [h]:mm
+ m += (h * 60);
+ return TQString("%1:%2").arg(h, 1).arg(minute, 2);
+ }
+
+ return converter->locale()->formatTime( dt.time(), false );
+}
+
+TQString ValueFormatter::dateFormat (const TQDate &date, FormatType fmtType)
+{
+ TQString tmp;
+ if (fmtType == ShortDate_format) {
+ tmp = converter->locale()->formatDate(date, true);
+ }
+ else if (fmtType == TextDate_format) {
+ tmp = converter->locale()->formatDate(date, false);
+ }
+ else if (fmtType == date_format1) { /*18-Feb-99 */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "-" + converter->locale()->calendar()->monthString(date, true) + "-";
+ tmp += TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format2) { /*18-Feb-1999 */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "-" + converter->locale()->calendar()->monthString(date, true) + "-";
+ tmp += TQString::number(date.year());
+ }
+ else if (fmtType == date_format3) { /*18-Feb */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "-" + converter->locale()->calendar()->monthString(date, true);
+ }
+ else if (fmtType == date_format4) { /*18-05 */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "-" + TQString().sprintf("%02d", date.month() );
+ }
+ else if (fmtType == date_format5) { /*18/05/00 */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "/" + TQString().sprintf("%02d", date.month()) + "/";
+ tmp += TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format6) { /*18/05/1999 */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "/" + TQString().sprintf("%02d", date.month()) + "/";
+ tmp += TQString::number(date.year());
+ }
+ else if (fmtType == date_format7) { /*Feb-99 */
+ tmp = converter->locale()->calendar()->monthString(date, true) + "-";
+ tmp += TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format8) { /*February-99 */
+ tmp = converter->locale()->calendar()->monthString(date, false) + "-";
+ tmp += TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format9) { /*February-1999 */
+ tmp = converter->locale()->calendar()->monthString(date, false) + "-";
+ tmp += TQString::number(date.year());
+ }
+ else if (fmtType == date_format10) { /*F-99 */
+ tmp = converter->locale()->calendar()->monthString(date, false).at(0) + "-";
+ tmp += TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format11) { /*18/Feb */
+ tmp = TQString().sprintf("%02d", date.day()) + "/";
+ tmp += converter->locale()->calendar()->monthString(date, true);
+ }
+ else if (fmtType == date_format12) { /*18/02 */
+ tmp = TQString().sprintf("%02d", date.day()) + "/";
+ tmp += TQString().sprintf("%02d", date.month());
+ }
+ else if (fmtType == date_format13) { /*18/Feb/1999 */
+ tmp = TQString().sprintf("%02d", date.day());
+ tmp += "/" + converter->locale()->calendar()->monthString(date, true) + "/";
+ tmp += TQString::number(date.year());
+ }
+ else if (fmtType == date_format14) { /*2000/Feb/18 */
+ tmp = TQString::number(date.year());
+ tmp += "/" + converter->locale()->calendar()->monthString(date, true) + "/";
+ tmp += TQString().sprintf("%02d", date.day());
+ }
+ else if (fmtType == date_format15) { /*2000-Feb-18 */
+ tmp = TQString::number(date.year());
+ tmp += "-" + converter->locale()->calendar()->monthString(date, true) + "-";
+ tmp += TQString().sprintf("%02d", date.day());
+ }
+ else if (fmtType == date_format16) { /*2000-02-18 */
+ tmp = TQString::number(date.year());
+ tmp += "-" + TQString().sprintf("%02d", date.month()) + "-";
+ tmp += TQString().sprintf("%02d", date.day());
+ }
+ else if (fmtType == date_format17) { /*2 february 2000 */
+ tmp = TQString().sprintf("%d", date.day());
+ tmp += " " + converter->locale()->calendar()->monthString(date, false) + " ";
+ tmp += TQString::number(date.year());
+ }
+ else if (fmtType == date_format18) { /*02/18/1999 */
+ tmp = TQString().sprintf("%02d", date.month());
+ tmp += "/" + TQString().sprintf("%02d", date.day());
+ tmp += "/" + TQString::number(date.year());
+ }
+ else if (fmtType == date_format19) { /*02/18/99 */
+ tmp = TQString().sprintf("%02d", date.month());
+ tmp += "/" + TQString().sprintf("%02d", date.day());
+ tmp += "/" + TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format20) { /*Feb/18/99 */
+ tmp = converter->locale()->calendar()->monthString(date, true);
+ tmp += "/" + TQString().sprintf("%02d", date.day());
+ tmp += "/" + TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format21) { /*Feb/18/1999 */
+ tmp = converter->locale()->calendar()->monthString(date, true);
+ tmp += "/" + TQString().sprintf("%02d", date.day());
+ tmp += "/" + TQString::number(date.year());
+ }
+ else if (fmtType == date_format22) { /*Feb-1999 */
+ tmp = converter->locale()->calendar()->monthString(date, true) + "-";
+ tmp += TQString::number(date.year());
+ }
+ else if (fmtType == date_format23) { /*1999 */
+ tmp = TQString::number(date.year());
+ }
+ else if (fmtType == date_format24) { /*99 */
+ tmp = TQString::number(date.year()).right(2);
+ }
+ else if (fmtType == date_format25) { /*2000/02/18 */
+ tmp = TQString::number(date.year());
+ tmp += "/" + TQString().sprintf("%02d", date.month());
+ tmp += "/" + TQString().sprintf("%02d", date.day());
+ }
+ else if (fmtType == date_format26) { /*2000/Feb/18 */
+ tmp = TQString::number(date.year());
+ tmp += "/" + converter->locale()->calendar()->monthString(date, true);
+ tmp += "/" + TQString().sprintf("%02d", date.day());
+ }
+ else
+ tmp = converter->locale()->formatDate(date, true);
+
+ // Missing compared with gnumeric:
+ // "m/d/yy h:mm", /* 20 */
+ // "m/d/yyyy h:mm", /* 21 */
+ // "mmm/ddd/yy", /* 12 */
+ // "mmm/ddd/yyyy", /* 13 */
+ // "mm/ddd/yy", /* 14 */
+ // "mm/ddd/yyyy", /* 15 */
+
+ return tmp;
+}
+
+TQString ValueFormatter::errorFormat (Cell *cell)
+{
+ TQString err;
+ if (cell->testFlag (Cell::Flag_ParseError))
+ err = "#" + i18n ("Parse") + "!";
+ else if ( cell->testFlag (Cell::Flag_CircularCalculation))
+ err = "#" + i18n ("Circle") + "!";
+ else if ( cell->testFlag (Cell::Flag_DependancyError))
+ err = "#" + i18n ("Depend") + "!";
+ else
+ {
+ err = "####";
+ kdDebug(36001) << "Unhandled error type." << endl;
+ }
+ return err;
+}
+