XLKit  0.1.0
 All Classes Files Functions Typedefs Macros Groups
xlOperand.hpp
Go to the documentation of this file.
1 
6 // Copyright (c) 2014 Edward Lam
7 //
8 // All rights reserved. This software is distributed under the
9 // Mozilla Public License, v. 2.0 ( http://www.mozilla.org/MPL/2.0/ ).
10 //
11 // Redistributions of source code must retain the above copyright
12 // and license notice and the following restrictions and disclaimer.
13 //
14 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
15 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
16 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
17 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
18 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY INDIRECT, INCIDENTAL,
19 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
20 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
21 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
22 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
23 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
24 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
25 
26 #ifndef XLKIT_XLOPERAND_HPP
27 #define XLKIT_XLOPERAND_HPP
28 
29 #include <xlkit/xlcall.hpp>
30 #include <xlkit/xlException.hpp>
31 #include <xlkit/xlutil.hpp>
32 #include <xlkit/xlversion.hpp>
33 
34 #include <boost/lexical_cast.hpp>
35 
36 #include <utility>
37 #include <string>
38 
39 #include <stddef.h>
40 #include <stdint.h>
41 #include <string.h>
42 
43 namespace xlkit {
45 namespace XLKIT_VERSION_NAME {
46 
47 namespace detail {
48 
49 template<typename T>
50 struct unimplemented : std::false_type {};
51 
52 
53 } // namespace detail
54 
56 struct xlError {
57  xlError() : num(xlerrNull) { }
59  xlError(int x) : num(x) { }
61  operator int() {
62  return num;
63  }
65  std::string str() const {
66  if (num == xlerrNull)
67  return std::string("xlerrNull");
68  if (num == xlerrDiv0)
69  return std::string("xlerrDiv0");
70  if (num == xlerrValue)
71  return std::string("xlerrValue");
72  if (num == xlerrRef)
73  return std::string("xlerrRef");
74  if (num == xlerrName)
75  return std::string("xlerrName");
76  if (num == xlerrNum)
77  return std::string("xlerrNum");
78  if (num == xlerrNA)
79  return std::string("xlerrNA");
80  if (num == xlerrGettingData)
81  return std::string("xlerrGettingData");
82  return boost::lexical_cast<std::string>(num);
83  }
84  int num;
85 };
86 
87 inline std::string
88 xltypeString(unsigned int xltype) {
89 
90  std::string xlfree;
91  if (xltype & xlbitXLFree)
92  xlfree += "|xlbitXLFree";
93  if (xltype & xlbitDLLFree)
94  xlfree += "|xlbitDLLFree";
95 
96  // Do big data first as it has both Str and Int bits enabled
97  if (xltype & xltypeBigData)
98  return std::string("xltypeBigData") + xlfree;
99 
100  if (xltype & xltypeNum)
101  return std::string("xltypeNum") + xlfree;
102  if (xltype & xltypeStr)
103  return std::string("xltypeStr") + xlfree;
104  if (xltype & xltypeBool)
105  return std::string("xltypeBool") + xlfree;
106  if (xltype & xltypeRef)
107  return std::string("xltypeRef") + xlfree;
108  if (xltype & xltypeErr)
109  return std::string("xltypeErr") + xlfree;
110  if (xltype & xltypeFlow)
111  return std::string("xltypeFlow") + xlfree;
112  if (xltype & xltypeMulti)
113  return std::string("xltypeMulti") + xlfree;
114  if (xltype & xltypeMissing)
115  return std::string("xltypeMissing") + xlfree;
116  if (xltype & xltypeNil)
117  return std::string("xltypeNil") + xlfree;
118  if (xltype & xltypeSRef)
119  return std::string("xltypeSRef") + xlfree;
120  if (xltype & xltypeInt)
121  return std::string("xltypeInt") + xlfree;
122 
123  return std::string("Unknown xltype") + xlfree;
124 }
125 
130 class xlOper4 : private XLOPER {
131 
132  public:
133 
137  public:
138 
140  int rows() const {
141  return myOperand->val.array.rows;
142  }
144  int cols() const {
145  return myOperand->val.array.columns;
146  }
147 
150  const xlOper4& operator()(int i, int j) const {
151  return *((xlOper4*)myOperand->val.array.lparray
152  + (i * myOperand->val.array.columns) + j);
153  }
154  xlOper4& operator()(int i, int j) {
155  return *((xlOper4*)myOperand->val.array.lparray
156  + (i * myOperand->val.array.columns) + j);
157  }
159 
160  private:
161  explicit CellMatrixRef(xlOper4* operand)
162  : myOperand(operand) { }
163 
164  xlOper4* myOperand;
165  friend class xlOper4;
166  friend class ConstCellMatrixRef;
167  };
170  public:
173  : myOperand(ref.myOperand) {
174  }
175 
177  int rows() const {
178  return myOperand->val.array.rows;
179  }
181  int cols() const {
182  return myOperand->val.array.columns;
183  }
184 
186  const xlOper4& operator()(int i, int j) const {
187  return *((xlOper4*)myOperand->val.array.lparray
188  + (i * myOperand->val.array.columns) + j);
189  }
190 
191  private:
192  explicit ConstCellMatrixRef(const xlOper4* operand)
193  : myOperand(operand) { }
194 
195  const xlOper4* myOperand;
196  friend class xlOper4;
197  };
198  friend class xlOper4::CellMatrixRef;
199  friend class xlOper4::ConstCellMatrixRef;
201 
204  init();
205  }
206  ~xlOper4() {
207  reset();
208  }
209  xlOper4(const xlOper4& other) {
210  init();
211  *this = other;
212  }
213  xlOper4(xlOper4&& other) {
214  init();
215  *this = std::move(other);
216  }
217 
219  explicit xlOper4(double v) {
220  init();
221  set(v);
222  }
224  explicit xlOper4(int v) {
225  init();
226  set(v);
227  }
229  explicit xlOper4(const std::string& v) {
230  init();
231  set(v);
232  }
234  explicit xlOper4(const char* v) {
235  init();
236  set(v);
237  }
239  explicit xlOper4(bool v) {
240  init();
241  set(v);
242  }
245  explicit xlOper4(xlError num) {
246  init();
247  set(num);
248  }
252  explicit xlOper4(int rows, int cols, xlOper4* init_val = NULL) {
253  init();
254  setMatrix(rows, cols, init_val);
255  }
257  explicit xlOper4(ConstCellMatrixRef cell_ref) {
258  init();
259  set(cell_ref);
260  }
261 
263  inline void reset() {
264  if (xltype & xltypeStr) {
265  if (xltype & xlbitXLFree)
266  XLKIT_THROW("Cannot reset memory allocated by Excel!");
267  else if (xltype & xlbitDLLFree)
268  ::free(val.str);
269  } else if (xltype & xltypeMulti) {
270  if (xltype & xlbitXLFree)
271  XLKIT_THROW("Cannot reset memory allocated by Excel!");
272  else if (xltype & xlbitDLLFree)
273  ::free(reinterpret_cast<void*>(val.array.lparray));
274  }
275  init();
276  }
277 
279  xlOper4& operator=(const xlOper4& other) {
280  if (this != &other) {
281  if (other.isString()) {
282  set(other.get<std::string>());
283  } else if (other.isCellMatrix()) {
284  set(other.get<ConstCellMatrixRef>());
285  } else {
286  reset();
287  ::memcpy(this, &other, sizeof(*this));
288  }
289  }
290  return *this;
291  }
294  if (this != &other) {
295  reset();
296  ::memcpy(this, &other, sizeof(*this));
297  other.init();
298  }
299  return *this;
300  }
301 
304  inline bool isDouble() const {
305  return (xltype == xltypeNum);
306  }
307  inline bool isInteger() const {
308  return (xltype == xltypeInt);
309  }
310  inline bool isString() const {
311  return ( xltype == xltypeStr
312  || xltype == (xltypeStr|xlbitXLFree)
313  || xltype == (xltypeStr|xlbitDLLFree));
314  }
315  inline bool isBool() const {
316  return (xltype == xltypeBool);
317  }
318  inline bool isError() const {
319  return (xltype == xltypeErr);
320  }
321  inline bool isMissing() const {
322  return (xltype == xltypeMissing);
323  }
324  inline bool isCellMatrix() const {
325  return ( xltype == xltypeMulti
326  || xltype == (xltypeMulti|xlbitXLFree)
327  || xltype == (xltypeMulti|xlbitDLLFree));
328  }
330 
331  template <typename T> T get() const {
332  static_assert( detail::unimplemented<T>::value
333  , "Only specializations of get<>() const may be used" );
334  }
335 
336  template <>
337  double get<double>() const {
338  if (!isDouble())
339  return castValue<double>();
340  return val.num;
341  }
342  template <>
343  int get<int>() const {
344  if (!isInteger())
345  return castValue<int>();
346  return val.w;
347  }
348  template <>
349  std::string get<std::string>() const {
350  if (!isString())
351  return castValue<std::string>();
352  uint8_t* blen = reinterpret_cast<uint8_t*>(&val.str[0]);
353  return std::string(reinterpret_cast<char *>(val.str + 1), *blen);
354  }
355  template <>
356  bool get<bool>() const {
357  if (!isBool())
358  return castValue<bool>();
359  return (val.xbool != 0);
360  }
361  template <>
362  xlError get<xlError>() const {
363  if (!isError())
364  XLKIT_THROW("Cannot cast to xlError from " + xltypeString(xltype));
365  return xlError(val.err);
366  }
367  template <>
368  ConstCellMatrixRef get<ConstCellMatrixRef>() const {
369  if (!isCellMatrix())
370  XLKIT_THROW("Cannot cast to ConstCellMatrixRef from " + xltypeString(xltype));
371  return ConstCellMatrixRef(this);
372  }
373 
376  CellMatrixRef
377  setMatrix(int rows, int cols, xlOper4* init_val = NULL) {
378  reset();
379  xltype = xltypeMulti | xlbitDLLFree;
380  val.array.rows = rows;
381  val.array.columns = cols;
382  val.array.lparray = reinterpret_cast<XLOPER*>(
383  ::malloc(rows * cols * sizeof(xlOper4)));
384  CellMatrixRef dst(this);
385  if (init_val) {
386  for (int i = 0; i < rows; ++i) {
387  for (int j = 0; j < cols; ++j) {
388  dst(i, j) = *init_val;
389  }
390  }
391  } else {
392  for (int i = 0; i < rows; ++i) {
393  for (int j = 0; j < cols; ++j) {
394  dst(i, j).init();
395  }
396  }
397  }
398  return CellMatrixRef(this);
399  }
403  if (!isCellMatrix())
404  XLKIT_THROW("Cannot cast to CellMatrixRef from " + xltypeString(xltype));
405  return CellMatrixRef(this);
406  }
407  ConstCellMatrixRef asCellMatrixRef() const {
408  if (!isCellMatrix())
409  XLKIT_THROW("Cannot cast to ConstCellMatrixRef from " + xltypeString(xltype));
410  return ConstCellMatrixRef(this);
411  }
413 
415  int stringLength() const {
416  if (!isString())
417  XLKIT_THROW("Not a string");
418  return *(reinterpret_cast<uint8_t*>(&val.str[0]));
419  }
421  int cellMatrixRows() const {
422  if (!isCellMatrix())
423  XLKIT_THROW("Not a cell matrix");
424  return val.array.rows;
425  }
427  int cellMatrixCols() const {
428  if (!isCellMatrix())
429  XLKIT_THROW("Not a cell matrix");
430  return val.array.columns;
431  }
432 
435  void set(double v) {
436  reset();
437  xltype = xltypeNum;
438  val.num = v;
439  }
440  void set(int v) {
441  reset();
442  xltype = xltypeInt;
443  val.w = (int16_t)v;
444  }
445  void set(const std::string& v) {
446  reset();
447  xltype = xltypeStr | xlbitDLLFree;
448  size_t len = v.size();
449  val.str = reinterpret_cast<char*>(::malloc((len+1) * sizeof(uint8_t)));
450  uint8_t* blen = reinterpret_cast<uint8_t*>(&val.str[0]);
451  *blen = uint8_t(len < 255 ? len : 255);
452  XLKIT_PUSH_DISABLE_WARN_DEPRECATION
453  ::strncpy(reinterpret_cast<char *>(val.str + 1), v.data(), len);
454  XLKIT_POP_DISABLE_WARN_DEPRECATION
455  }
456  void set(const char* v) {
457  reset();
458  xltype = xltypeStr | xlbitDLLFree;
459  size_t len = ::strlen(v);
460  val.str = reinterpret_cast<char*>(::malloc((len+1) * sizeof(uint8_t)));
461  uint8_t* blen = reinterpret_cast<uint8_t*>(&val.str[0]);
462  *blen = uint8_t(len < 255 ? len : 255);
463  XLKIT_PUSH_DISABLE_WARN_DEPRECATION
464  ::strncpy(reinterpret_cast<char *>(val.str + 1), v, len);
465  XLKIT_POP_DISABLE_WARN_DEPRECATION
466  }
467  void set(bool v) {
468  reset();
469  xltype = xltypeBool;
470  val.xbool = v;
471  }
472  void set(xlError num) {
473  reset();
474  xltype = xltypeErr;
475  val.err = num;
476  }
477  void set(ConstCellMatrixRef src) {
478  CellMatrixRef dst(setMatrix(src.rows(), src.cols()));
479  for (int i = 0, rows = src.rows(); i < rows; ++i) {
480  for (int j = 0, cols = src.cols(); j < cols; ++j) {
481  dst(i, j) = src(i, j);
482  }
483  }
484  }
486 
487  private: // methods
488 
489  // Mimic default ctor behaviour, assumes we're uninitialized
490  inline void init() {
491  xltype = xltypeMissing;
492  val.num = 0;
493  }
494 
495  template <typename T>
496  T castValue() const {
497  // Casting to a number
498  if (isDouble())
499  return (T)(get<double>());
500  if (isInteger())
501  return (T)(get<int>());
502  if (isString())
503  return boost::lexical_cast<T>(get<std::string>());
504  if (isBool())
505  return (T)(get<bool>());
506  XLKIT_THROW("Unsupported conversion from " + xltypeString(xltype));
507  }
508  template <>
509  std::string castValue<std::string>() const {
510  // Casting to a string
511  if (isDouble())
512  return boost::lexical_cast<std::string>(get<double>());
513  if (isInteger())
514  return boost::lexical_cast<std::string>(get<int>());
515  if (isBool())
516  return boost::lexical_cast<std::string>(get<bool>());
517  if (isError())
518  return xlError(val.err).str();
519  XLKIT_THROW("Cannot cast to string from " + xltypeString(xltype));
520  }
521  template <>
522  bool castValue<bool>() const {
523  // Casting to a bool
524  if (isDouble())
525  return (get<double>() != 0);
526  if (isInteger())
527  return (get<int>() != 0);
528  if (isString())
529  return (stringLength() != 0);
530  if (isBool())
531  return get<bool>();
532  XLKIT_THROW("Cannot cast to bool from " + xltypeString(xltype));
533  }
534 
535 };
536 typedef xlOper4 xlOperand;
537 typedef xlOper4::CellMatrixRef xlCellMatrixRef;
538 typedef xlOper4::ConstCellMatrixRef xlConstCellMatrixRef;
539 
540 } // namespace XLKIT_VERSION_NAME
541 } // namespace xlkit
542 
545 
548 
551 
554 
556 
557 #endif // XLKIT_XLOPERAND_HPP
xlOper4(xlError num)
Construct an error number (xlerrValue, xlerrNA, etc..)
Definition: xlOperand.hpp:245
std::string str() const
Return enum value as a string.
Definition: xlOperand.hpp:65
CellMatrixRef asCellMatrixRef()
Obtain cell matrix reference.
Definition: xlOperand.hpp:402
bool isDouble() const
Test operand type.
Definition: xlOperand.hpp:304
int cols() const
Columns in matrix.
Definition: xlOperand.hpp:144
xlOper4(int rows, int cols, xlOper4 *init_val=NULL)
Construct a cell matrix of given size.
Definition: xlOperand.hpp:252
#define XLKIT_THROW(MSG)
Throws an xlException with the given string literal.
Definition: xlException.hpp:47
const xlOper4 & operator()(int i, int j) const
(row,col) value in matrix
Definition: xlOperand.hpp:150
xlOper4()
Default constructor, initializes as xltypeMissing.
Definition: xlOperand.hpp:203
void reset()
Free allocated memory and reset to initial state (xltypeMissing)
Definition: xlOperand.hpp:263
xlkit::xlOperand xlOperand
C++ wrapper for XLOPER struct. See xlOper4.
Definition: xlOperand.hpp:547
int cellMatrixCols() const
For a cell matrix, return its number of columns.
Definition: xlOperand.hpp:427
xlOper4(int v)
Construct an integer.
Definition: xlOperand.hpp:224
xlOper4(ConstCellMatrixRef cell_ref)
Construct a copy of a cell matrix.
Definition: xlOperand.hpp:257
Proxy class into an operand's cell matrix (non-mutable)
Definition: xlOperand.hpp:169
xlOper4 & operator=(xlOper4 &&other)
Assignment move operator.
Definition: xlOperand.hpp:293
CellMatrixRef setMatrix(int rows, int cols, xlOper4 *init_val=NULL)
Make a matrix of the given size and return a ref to it.
Definition: xlOperand.hpp:377
const xlOper4 & operator()(int i, int j) const
(row,col) value in matrix
Definition: xlOperand.hpp:186
ConstCellMatrixRef(CellMatrixRef ref)
Construct from non-const CellMatrixRef.
Definition: xlOperand.hpp:172
void set(double v)
Set a new value.
Definition: xlOperand.hpp:435
XLKit utilities.
xlOper4(const char *v)
Construct from a C-style null-terminated string.
Definition: xlOperand.hpp:234
int rows() const
Rows in matrix.
Definition: xlOperand.hpp:140
xlkit::xlCellMatrixRef xlCellMatrixRef
Proxy class into an operand's cell matrix (mutable). See CellMatrixRef.
Definition: xlOperand.hpp:550
#define XLKIT_VERSION_NAME
Version namespace for this library.
Definition: xlversion.hpp:33
xlOper4 & operator=(const xlOper4 &other)
Assignment operator.
Definition: xlOperand.hpp:279
xlError(int x)
Construct from xlerr* enum.
Definition: xlOperand.hpp:59
xlOper4(double v)
Construct a double precision floating point number.
Definition: xlOperand.hpp:219
C++ methods that operate on top of an XLOPER struct.
Definition: xlOperand.hpp:130
#define XLKIT_USE_VERSION_NAMESPACE
Macro used to pull the versioned namespace into the main xlkit namepsace.
Definition: xlversion.hpp:52
int cellMatrixRows() const
For a cell matrix, return its number of rows.
Definition: xlOperand.hpp:421
xlOper4(bool v)
Construct a bool.
Definition: xlOperand.hpp:239
int cols() const
Columns in matrix.
Definition: xlOperand.hpp:181
int rows() const
Rows in matrix.
Definition: xlOperand.hpp:177
int stringLength() const
For a string, return its length.
Definition: xlOperand.hpp:415
Proxy class into an operand's cell matrix (mutable)
Definition: xlOperand.hpp:136
XLKit version defines.
Wrapper for XLCALL.H so that we don't need #include
xlOper4(const std::string &v)
Construct from an std::string.
Definition: xlOperand.hpp:229
xlkit::xlConstCellMatrixRef xlConstCellMatrixRef
Proxy class into an operand's cell matrix (non-mutable). See ConstCellMatrixRef.
Definition: xlOperand.hpp:553
An error number value.
Definition: xlOperand.hpp:56
Provides XLKIT_THROW() macro and xlException class.