summaryrefslogtreecommitdiffstats
path: root/kspread/kspread_functions_reference.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'kspread/kspread_functions_reference.cpp')
-rw-r--r--kspread/kspread_functions_reference.cpp342
1 files changed, 342 insertions, 0 deletions
diff --git a/kspread/kspread_functions_reference.cpp b/kspread/kspread_functions_reference.cpp
new file mode 100644
index 00000000..7dacb054
--- /dev/null
+++ b/kspread/kspread_functions_reference.cpp
@@ -0,0 +1,342 @@
+/* This file is part of the KDE project
+ Copyright (C) 1998-2002 The KSpread Team
+ www.koffice.org/kspread
+ Copyright (C) 2005 Tomas Mecir <mecirt@gmail.com>
+
+ 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.
+
+ 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.
+*/
+
+// built-in reference functions
+
+#include "kspread_cell.h"
+#include "kspread_sheet.h"
+#include "kspread_util.h"
+#include "kspread_value.h"
+
+#include "functions.h"
+#include "valuecalc.h"
+#include "valueconverter.h"
+
+using namespace KSpread;
+
+// prototypes (sorted alphabetically)
+Value func_address (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_areas (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_choose (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_column (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_columns (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_index (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_indirect (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_lookup (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_row (valVector args, ValueCalc *calc, FuncExtra *);
+Value func_rows (valVector args, ValueCalc *calc, FuncExtra *);
+
+// registers all reference functions
+void RegisterReferenceFunctions()
+{
+ FunctionRepository* repo = FunctionRepository::self();
+ Function *f;
+
+ f = new Function ("ADDRESS", func_address);
+ f->setParamCount (2, 5);
+ repo->add (f);
+ f = new Function ("AREAS", func_areas);
+ f->setParamCount (1);
+ f->setNeedsExtra (true);
+ repo->add (f);
+ f = new Function ("CHOOSE", func_choose);
+ f->setParamCount (2, -1);
+ f->setAcceptArray ();
+ repo->add (f);
+ f = new Function ("COLUMN", func_column);
+ f->setParamCount (0, 1);
+ repo->add (f);
+ f = new Function ("COLUMNS", func_columns);
+ f->setParamCount (1);
+ f->setAcceptArray ();
+ f->setNeedsExtra (true);
+ repo->add (f);
+ f = new Function ("INDEX", func_index);
+ f->setParamCount (3);
+ f->setAcceptArray ();
+ repo->add (f);
+ f = new Function ("INDIRECT", func_indirect);
+ f->setParamCount (1, 2);
+ f->setNeedsExtra (true);
+ repo->add (f);
+ f = new Function ("LOOKUP", func_lookup);
+ f->setParamCount (3);
+ f->setAcceptArray ();
+ repo->add (f);
+ f = new Function ("ROW", func_row);
+ f->setParamCount (0, 1);
+ repo->add (f);
+ f = new Function ("ROWS", func_rows);
+ f->setParamCount (1);
+ f->setAcceptArray ();
+ f->setNeedsExtra (true);
+ repo->add (f);
+}
+
+// Function: ADDRESS
+Value func_address (valVector args, ValueCalc *calc, FuncExtra *)
+{
+ bool r1c1 = false;
+ TQString sheetName;
+ int absNum = 1;
+ if (args.count() > 2)
+ absNum = calc->conv()->asInteger (args[2]).asInteger();
+ if (args.count() > 3)
+ r1c1 = !(calc->conv()->asBoolean (args[3]).asBoolean());
+ if (args.count() == 5)
+ sheetName = calc->conv()->asString (args[4]).asString();
+
+ TQString result;
+ int row = calc->conv()->asInteger (args[0]).asInteger();
+ int col = calc->conv()->asInteger (args[1]).asInteger();
+
+ if ( !sheetName.isEmpty() )
+ {
+ result += sheetName;
+ result += "!";
+ }
+
+ if ( r1c1 )
+ {
+ // row first
+ bool abs = false;
+ if ( absNum == 1 || absNum == 2 )
+ abs = true;
+
+ result += 'R';
+ if ( !abs )
+ result += '[';
+ result += TQString::number( row );
+
+ if ( !abs )
+ result += ']';
+
+ // column
+ abs = false;
+ if ( absNum == 1 || absNum == 3 )
+ abs = true;
+
+ result += 'C';
+ if ( !abs )
+ result += '[';
+ result += TQString::number( col );
+
+ if ( !abs )
+ result += ']';
+ }
+ else
+ {
+ bool abs = false;
+ if ( absNum == 1 || absNum == 3 )
+ abs = true;
+
+ if ( abs )
+ result += '$';
+
+ result += Cell::columnName( col );
+
+ abs = false;
+ if ( absNum == 1 || absNum == 2 )
+ abs = true;
+
+ if ( abs )
+ result += '$';
+
+ result += TQString::number( row );
+ }
+
+ return Value (result);
+}
+
+bool checkRef( TQString const & ref )
+{
+ Range r( ref );
+ if ( !r.isValid() )
+ {
+ Point p( ref );
+ if ( !p.isValid() )
+ return false;
+ }
+ return true;
+}
+
+// Function: AREAS
+Value func_areas (valVector args, ValueCalc *calc, FuncExtra *e)
+{
+ if (e) {
+ if ((e->ranges[0].col1 != -1) && (e->ranges[0].row1 != -1) &&
+ (e->ranges[0].col2 != -1) && (e->ranges[0].row2 != -1))
+ // we have a range reference - return 1
+ return 1;
+ }
+
+ TQString s = calc->conv()->asString (args[0]).asString();
+ if ( s[0] != '(' || s[s.length() - 1] != ')' )
+ return Value::errorVALUE();
+
+ int l = s.length();
+
+ int num = 0;
+ TQString ref;
+ for ( int i = 1; i < l; ++i )
+ {
+ if ( s[i] == ',' || s[i] == ')' )
+ {
+ if ( !checkRef( ref ) )
+ return Value::errorVALUE();
+ else
+ {
+ ++num;
+ ref = "";
+ }
+ }
+ else
+ ref += s[i];
+ }
+
+ return Value (num);
+}
+
+// Function: CHOOSE
+Value func_choose (valVector args, ValueCalc *calc, FuncExtra *)
+{
+ int cnt = args.count () - 1;
+ int num = calc->conv()->asInteger (args[0]).asInteger();
+ if ((num <= 0) || (num > cnt))
+ return Value::errorVALUE();
+ return args[num];
+}
+
+// Function: INDEX
+Value func_index (valVector args, ValueCalc *calc, FuncExtra *)
+{
+ // first argument can be either a range, then we return a given cell's
+ // value, or a single cell containing an array - then we return the array
+ // element. In any case, this function can assume that the given value
+ // is the same. Because it is.
+
+ Value val = args[0];
+ unsigned row = calc->conv()->asInteger (args[1]).asInteger() - 1;
+ unsigned col = calc->conv()->asInteger (args[2]).asInteger() - 1;
+ if ((row >= val.rows()) || (col >= val.columns()))
+ return Value::errorREF();
+ return val.element (col, row);
+}
+
+// Function: LOOKUP
+Value func_lookup (valVector args, ValueCalc *calc, FuncExtra *)
+{
+ Value num = calc->conv()->asNumeric (args[0]);
+ if (num.isArray())
+ return Value::errorVALUE();
+ Value lookup = args[1];
+ Value rr = args[2];
+ unsigned cols = lookup.columns();
+ unsigned rows = lookup.rows();
+ if ((cols != rr.columns()) || (rows != rr.rows()))
+ return Value::errorVALUE();
+ Value res = Value::errorNA();
+
+ // now traverse the array and perform comparison
+ for (unsigned r = 0; r < rows; ++r)
+ for (unsigned c = 0; c < cols; ++c)
+ {
+ // update the result, return if we cross the line
+ Value le = lookup.element (c, r);
+ if (calc->lower (le, num) || calc->equal (num, le))
+ res = rr.element (c, r);
+ else
+ return res;
+ }
+ return res;
+}
+
+// Function: COLUMN
+Value func_column (valVector args, ValueCalc *, FuncExtra *e)
+{
+ int col = e ? e->mycol : 0;
+ if (e && args.count())
+ col = e->ranges[0].col1;
+ if (col > 0)
+ return Value (col);
+ return Value::errorVALUE();
+}
+
+// Function: ROW
+Value func_row (valVector args, ValueCalc *, FuncExtra *e)
+{
+ int row = e ? e->myrow : 0;
+ if (e && args.count())
+ row = e->ranges[0].row1;
+ if (row > 0)
+ return Value (row);
+ return Value::errorVALUE();
+}
+
+// Function: COLUMNS
+Value func_columns (valVector, ValueCalc *, FuncExtra *e)
+{
+ int col1 = e->ranges[0].col1;
+ int col2 = e->ranges[0].col2;
+ if ((col1 == -1) || (col2 == -1))
+ return Value::errorVALUE();
+ return Value (col2 - col1 + 1);
+}
+
+// Function: ROWS
+Value func_rows (valVector, ValueCalc *, FuncExtra *e)
+{
+ int row1 = e->ranges[0].row1;
+ int row2 = e->ranges[0].row2;
+ if ((row1 == -1) || (row2 == -1))
+ return Value::errorVALUE();
+ return Value (row2 - row1 + 1);
+}
+
+
+// Function: INDIRECT
+Value func_indirect (valVector args, ValueCalc *calc, FuncExtra *e)
+{
+ bool r1c1 = false;
+ TQString ref = calc->conv()->asString (args[0]).asString();
+ if (args.count() == 2)
+ r1c1 = !(calc->conv()->asBoolean (args[1]).asBoolean());
+
+ if (ref.isEmpty())
+ return Value::errorVALUE();
+
+ if ( r1c1 )
+ {
+ // TODO: translate the r1c1 style to a1 style
+ ref = ref;
+ }
+
+ Point p (ref, e->sheet->workbook(), e->sheet);
+
+ if ( !p.isValid() )
+ return Value::errorVALUE();
+
+ Cell * cell = p.cell();
+ if (cell)
+ return cell->value();
+ return Value::errorVALUE();
+}
+