26 #ifndef XLKIT_XLOPERAND_HPP
27 #define XLKIT_XLOPERAND_HPP
34 #include <boost/lexical_cast.hpp>
50 struct unimplemented : std::false_type {};
65 std::string
str()
const {
67 return std::string(
"xlerrNull");
69 return std::string(
"xlerrDiv0");
70 if (num == xlerrValue)
71 return std::string(
"xlerrValue");
73 return std::string(
"xlerrRef");
75 return std::string(
"xlerrName");
77 return std::string(
"xlerrNum");
79 return std::string(
"xlerrNA");
80 if (num == xlerrGettingData)
81 return std::string(
"xlerrGettingData");
82 return boost::lexical_cast<std::string>(num);
88 xltypeString(
unsigned int xltype) {
91 if (xltype & xlbitXLFree)
92 xlfree +=
"|xlbitXLFree";
93 if (xltype & xlbitDLLFree)
94 xlfree +=
"|xlbitDLLFree";
97 if (xltype & xltypeBigData)
98 return std::string(
"xltypeBigData") + xlfree;
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;
123 return std::string(
"Unknown xltype") + xlfree;
141 return myOperand->val.array.rows;
145 return myOperand->val.array.columns;
151 return *((
xlOper4*)myOperand->val.array.lparray
152 + (i * myOperand->val.array.columns) + j);
155 return *((
xlOper4*)myOperand->val.array.lparray
156 + (i * myOperand->val.array.columns) + j);
161 explicit CellMatrixRef(
xlOper4* operand)
162 : myOperand(operand) { }
165 friend class xlOper4;
166 friend class ConstCellMatrixRef;
173 : myOperand(ref.myOperand) {
178 return myOperand->val.array.rows;
182 return myOperand->val.array.columns;
187 return *((
xlOper4*)myOperand->val.array.lparray
188 + (i * myOperand->val.array.columns) + j);
193 : myOperand(operand) { }
198 friend class xlOper4::CellMatrixRef;
199 friend class xlOper4::ConstCellMatrixRef;
215 *
this = std::move(other);
264 if (xltype & xltypeStr) {
265 if (xltype & xlbitXLFree)
266 XLKIT_THROW(
"Cannot reset memory allocated by Excel!");
267 else if (xltype & xlbitDLLFree)
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));
280 if (
this != &other) {
281 if (other.isString()) {
282 set(other.get<std::string>());
283 }
else if (other.isCellMatrix()) {
287 ::memcpy(
this, &other,
sizeof(*
this));
294 if (
this != &other) {
296 ::memcpy(
this, &other,
sizeof(*
this));
305 return (xltype == xltypeNum);
307 inline bool isInteger()
const {
308 return (xltype == xltypeInt);
310 inline bool isString()
const {
311 return ( xltype == xltypeStr
312 || xltype == (xltypeStr|xlbitXLFree)
313 || xltype == (xltypeStr|xlbitDLLFree));
315 inline bool isBool()
const {
316 return (xltype == xltypeBool);
318 inline bool isError()
const {
319 return (xltype == xltypeErr);
321 inline bool isMissing()
const {
322 return (xltype == xltypeMissing);
324 inline bool isCellMatrix()
const {
325 return ( xltype == xltypeMulti
326 || xltype == (xltypeMulti|xlbitXLFree)
327 || xltype == (xltypeMulti|xlbitDLLFree));
331 template <
typename T> T
get()
const {
332 static_assert( detail::unimplemented<T>::value
333 ,
"Only specializations of get<>() const may be used" );
337 double get<double>()
const {
339 return castValue<double>();
343 int get<int>()
const {
345 return castValue<int>();
349 std::string get<std::string>()
const {
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);
356 bool get<bool>()
const {
358 return castValue<bool>();
359 return (val.xbool != 0);
362 xlError get<xlError>()
const {
364 XLKIT_THROW(
"Cannot cast to xlError from " + xltypeString(xltype));
365 return xlError(val.err);
368 ConstCellMatrixRef get<ConstCellMatrixRef>()
const {
370 XLKIT_THROW(
"Cannot cast to ConstCellMatrixRef from " + xltypeString(xltype));
371 return ConstCellMatrixRef(
this);
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)));
386 for (
int i = 0; i < rows; ++i) {
387 for (
int j = 0; j < cols; ++j) {
388 dst(i, j) = *init_val;
392 for (
int i = 0; i < rows; ++i) {
393 for (
int j = 0; j < cols; ++j) {
404 XLKIT_THROW(
"Cannot cast to CellMatrixRef from " + xltypeString(xltype));
409 XLKIT_THROW(
"Cannot cast to ConstCellMatrixRef from " + xltypeString(xltype));
410 return ConstCellMatrixRef(
this);
418 return *(
reinterpret_cast<uint8_t*
>(&val.str[0]));
424 return val.array.rows;
430 return val.array.columns;
445 void set(
const std::string& v) {
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
456 void set(
const char* v) {
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
472 void set(xlError num) {
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);
491 xltype = xltypeMissing;
495 template <
typename T>
496 T castValue()
const {
499 return (T)(get<double>());
501 return (T)(get<int>());
503 return boost::lexical_cast<T>(get<std::string>());
505 return (T)(get<bool>());
506 XLKIT_THROW(
"Unsupported conversion from " + xltypeString(xltype));
509 std::string castValue<std::string>()
const {
512 return boost::lexical_cast<std::string>(get<double>());
514 return boost::lexical_cast<std::string>(get<int>());
516 return boost::lexical_cast<std::string>(get<bool>());
518 return xlError(val.err).str();
519 XLKIT_THROW(
"Cannot cast to string from " + xltypeString(xltype));
522 bool castValue<bool>()
const {
525 return (get<double>() != 0);
527 return (get<int>() != 0);
532 XLKIT_THROW(
"Cannot cast to bool from " + xltypeString(xltype));
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
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
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.