/*************************************************************************** * Copyright (c) Jürgen Riegel (juergen.riegel@web.de) 2013 * * * * This file is part of the FreeCAD CAx development system. * * * * 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., 59 Temple Place, * * Suite 330, Boston, MA 02111-1307, USA * * * ***************************************************************************/ #include "PreCompiled.h" #include "Base/Quantity.h" #include "Base/Vector3D.h" // inclusion of the generated files (generated out of QuantityPy.xml) #include "QuantityPy.h" #include "UnitPy.h" #include "QuantityPy.cpp" using namespace Base; // returns a string which represents the object e.g. when printed in python std::string QuantityPy::representation(void) const { std::stringstream ret; ret << getQuantityPtr()->getValue() << " "; ret << getQuantityPtr()->getUnit().getString().toLatin1().constData(); return ret.str(); } PyObject *QuantityPy::PyMake(struct _typeobject *, PyObject *, PyObject *) // Python wrapper { // create a new instance of QuantityPy and the Twin object return new QuantityPy(new Quantity); } // constructor method int QuantityPy::PyInit(PyObject* args, PyObject* kwd) { Quantity *self = getQuantityPtr(); double f = DOUBLE_MAX; int i1=0; int i2=0; int i3=0; int i4=0; int i5=0; int i6=0; int i7=0; int i8=0; if (PyArg_ParseTuple(args, "|diiiiiiii", &f,&i1,&i2,&i3,&i4,&i5,&i6,&i7,&i8)) { if(f!=DOUBLE_MAX) *self = Quantity(f,Unit(i1,i2,i3,i4,i5,i6,i7,i8)); return 0; } PyErr_Clear(); // set by PyArg_ParseTuple() PyObject *object; if (PyArg_ParseTuple(args,"O!",&(Base::QuantityPy::Type), &object)) { // Note: must be static_cast, not reinterpret_cast *self = *(static_cast(object)->getQuantityPtr()); return 0; } PyErr_Clear(); // set by PyArg_ParseTuple() if (PyArg_ParseTuple(args,"dO!",&f,&(Base::UnitPy::Type), &object)) { // Note: must be static_cast, not reinterpret_cast *self = Quantity(f,*(static_cast(object)->getUnitPtr())); return 0; } PyErr_Clear(); // set by PyArg_ParseTuple() const char* string; if (PyArg_ParseTuple(args,"s", &string)) { try { *self = Quantity::parse(QString::fromLatin1(string)); }catch(const Base::Exception& e) { PyErr_SetString(PyExc_ImportError, e.what()); return-1; } return 0; } PyErr_SetString(PyExc_TypeError, "Either three floats, tuple or Vector expected"); return -1; } PyObject* QuantityPy::getUserPreferred(PyObject *args) { QString uus; double factor; Py::Tuple res(3); QString uss = getQuantityPtr()->getUserString(factor,uus); res[0] = Py::String(uss.toLatin1()); res[1] = Py::Float(factor); res[2] = Py::String(uus.toLatin1()); return Py::new_reference_to(res); } PyObject* QuantityPy::getValueAs(PyObject *args) { Quantity quant; double f = DOUBLE_MAX; int i1=0; int i2=0; int i3=0; int i4=0; int i5=0; int i6=0; int i7=0; int i8=0; if (PyArg_ParseTuple(args, "d|iiiiiiii", &f,&i1,&i2,&i3,&i4,&i5,&i6,&i7,&i8)) { if(f!=DOUBLE_MAX) quant = Quantity(f,Unit(i1,i2,i3,i4,i5,i6,i7,i8)); }else{ PyErr_Clear(); // set by PyArg_ParseTuple() PyObject *object; if (PyArg_ParseTuple(args,"O!",&(Base::QuantityPy::Type), &object)) { // Note: must be static_cast, not reinterpret_cast quant = * static_cast(object)->getQuantityPtr(); }else{ PyErr_Clear(); // set by PyArg_ParseTuple() const char* string; if (PyArg_ParseTuple(args,"s", &string)) { quant = Quantity::parse(QString::fromLatin1(string)); }else{ PyErr_SetString(PyExc_TypeError, "Either three floats, tuple or Vector expected"); return 0; } } } quant = getQuantityPtr()->getValueAs(quant); return new QuantityPy(new Quantity(quant) ); } PyObject * QuantityPy::number_float_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Arg must be Quantity"); return 0; } QuantityPy* q = static_cast(self); return PyFloat_FromDouble(q->getValue()); } PyObject * QuantityPy::number_int_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Arg must be Quantity"); return 0; } QuantityPy* q = static_cast(self); return PyInt_FromLong((long)q->getValue()); } PyObject * QuantityPy::number_long_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Arg must be Quantity"); return 0; } QuantityPy* q = static_cast(self); return PyInt_FromLong((long)q->getValue()); } PyObject * QuantityPy::number_negative_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Arg must be Quantity"); return 0; } Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = -1; return new QuantityPy(new Quantity(*a * b)); } PyObject * QuantityPy::number_positive_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Arg must be Quantity"); return 0; } Base::Quantity *a = static_cast(self) ->getQuantityPtr(); return new QuantityPy(new Quantity(*a)); } PyObject * QuantityPy::number_absolute_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Arg must be Quantity"); return 0; } Base::Quantity *a = static_cast(self) ->getQuantityPtr(); return new QuantityPy(new Quantity(fabs(a->getValue()), a->getUnit())); } PyObject* QuantityPy::number_add_handler(PyObject *self, PyObject *other) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "First arg must be Quantity"); return 0; } if (!PyObject_TypeCheck(other, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Second arg must be Quantity"); return 0; } Base::Quantity *a = static_cast(self)->getQuantityPtr(); Base::Quantity *b = static_cast(other)->getQuantityPtr(); return new QuantityPy(new Quantity(*a+*b) ); } PyObject* QuantityPy::number_subtract_handler(PyObject *self, PyObject *other) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "First arg must be Quantity"); return 0; } if (!PyObject_TypeCheck(other, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "Second arg must be Quantity"); return 0; } Base::Quantity *a = static_cast(self)->getQuantityPtr(); Base::Quantity *b = static_cast(other)->getQuantityPtr(); return new QuantityPy(new Quantity(*a-*b) ); } PyObject* QuantityPy::number_multiply_handler(PyObject *self, PyObject *other) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "First arg must be Quantity"); return 0; } if (PyObject_TypeCheck(other, &(QuantityPy::Type))) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); Base::Quantity *b = static_cast(other)->getQuantityPtr(); return new QuantityPy(new Quantity(*a * *b) ); } else if (PyFloat_Check(other)) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = PyFloat_AsDouble(other); return new QuantityPy(new Quantity(*a*b) ); } else if (PyInt_Check(other)) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = (double)PyInt_AsLong(other); return new QuantityPy(new Quantity(*a*b) ); } else { PyErr_SetString(PyExc_TypeError, "A Quantity can only be multiplied by Quantity or number"); return 0; } } PyObject * QuantityPy::number_divide_handler (PyObject *self, PyObject *other) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "First arg must be Quantity"); return 0; } if (PyObject_TypeCheck(other, &(QuantityPy::Type))) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); Base::Quantity *b = static_cast(other)->getQuantityPtr(); return new QuantityPy(new Quantity(*a / *b) ); } else if (PyFloat_Check(other)) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = PyFloat_AsDouble(other); return new QuantityPy(new Quantity(*a / b) ); } else if (PyInt_Check(other)) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = (double)PyInt_AsLong(other); return new QuantityPy(new Quantity(*a / b) ); } else { PyErr_SetString(PyExc_TypeError, "A Quantity can only be divided by Quantity or number"); return 0; } } PyObject * QuantityPy::number_remainder_handler (PyObject *self, PyObject *other) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "First arg must be Quantity"); return 0; } double d1, d2; Base::Quantity *a = static_cast(self) ->getQuantityPtr(); d1 = a->getValue(); if (PyObject_TypeCheck(other, &(QuantityPy::Type))) { Base::Quantity *b = static_cast(other)->getQuantityPtr(); d2 = b->getValue(); } else if (PyFloat_Check(other)) { d2 = PyFloat_AsDouble(other); } else if (PyInt_Check(other)) { d2 = (double)PyInt_AsLong(other); } else { PyErr_SetString(PyExc_TypeError, "Expected quantity or number"); return 0; } PyObject* p1 = PyFloat_FromDouble(d1); PyObject* p2 = PyFloat_FromDouble(d2); PyObject* r = PyNumber_Remainder(p1, p2); Py_DECREF(p1); Py_DECREF(p2); if (!r) return 0; double q = PyFloat_AsDouble(r); Py_DECREF(r); return new QuantityPy(new Quantity(q,a->getUnit())); } PyObject * QuantityPy::number_divmod_handler (PyObject *self, PyObject *other) { //PyNumber_Divmod(); PyErr_SetString(PyExc_NotImplementedError, "Not implemented"); return 0; } PyObject * QuantityPy::number_power_handler (PyObject *self, PyObject *other, PyObject *modulo) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { PyErr_SetString(PyExc_TypeError, "First arg must be Quantity"); return 0; } if (PyObject_TypeCheck(other, &(QuantityPy::Type))) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); Base::Quantity *b = static_cast(other)->getQuantityPtr(); return new QuantityPy(new Quantity(a->pow(*b))); } else if (PyFloat_Check(other)) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = PyFloat_AsDouble(other); return new QuantityPy(new Quantity(a->pow(b)) ); } else if (PyInt_Check(other)) { Base::Quantity *a = static_cast(self) ->getQuantityPtr(); double b = (double)PyInt_AsLong(other); return new QuantityPy(new Quantity(a->pow(b))); } else { PyErr_SetString(PyExc_TypeError, "Expected quantity or number"); return 0; } } int QuantityPy::number_nonzero_handler (PyObject *self) { if (!PyObject_TypeCheck(self, &(QuantityPy::Type))) { return 1; } Base::Quantity *a = static_cast(self) ->getQuantityPtr(); return a->getValue() != 0; } PyObject* QuantityPy::richCompare(PyObject *v, PyObject *w, int op) { if (PyObject_TypeCheck(v, &(QuantityPy::Type)) && PyObject_TypeCheck(w, &(QuantityPy::Type))) { const Quantity * u1 = static_cast(v)->getQuantityPtr(); const Quantity * u2 = static_cast(w)->getQuantityPtr(); PyObject *res=0; if (op == Py_NE) { res = (!(*u1 == *u2)) ? Py_True : Py_False; Py_INCREF(res); return res; } else if (op == Py_LT) { res = (*u1 < *u2) ? Py_True : Py_False; Py_INCREF(res); return res; } else if (op == Py_LE) { res = (*u1 < *u2)||(*u1 == *u2) ? Py_True : Py_False; Py_INCREF(res); return res; } else if (op == Py_GT) { res = (!(*u1 < *u2))&&(!(*u1 == *u2)) ? Py_True : Py_False; Py_INCREF(res); return res; } else if (op == Py_GE) { res = (!(*u1 < *u2)) ? Py_True : Py_False; Py_INCREF(res); return res; } else if (op == Py_EQ) { res = (*u1 == *u2) ? Py_True : Py_False; Py_INCREF(res); return res; } } // This always returns False Py_INCREF(Py_NotImplemented); return Py_NotImplemented; } Py::Float QuantityPy::getValue(void) const { return Py::Float(getQuantityPtr()->getValue()); } void QuantityPy::setValue(Py::Float arg) { getQuantityPtr()->setValue(arg); } Py::Object QuantityPy::getUnit(void) const { return Py::Object(new UnitPy(new Unit(getQuantityPtr()->getUnit()))); } void QuantityPy::setUnit(Py::Object arg) { union PyType_Object pyType = {&(Base::UnitPy::Type)}; Py::Type UnitType(pyType.o); if(!arg.isType(UnitType)) throw Py::AttributeError("Not yet implemented"); getQuantityPtr()->setUnit(*static_cast((*arg))->getUnitPtr()); } Py::String QuantityPy::getUserString(void) const { return Py::String(getQuantityPtr()->getUserString().toLatin1()); } PyObject *QuantityPy::getCustomAttributes(const char* /*attr*/) const { return 0; } int QuantityPy::setCustomAttributes(const char* /*attr*/, PyObject* /*obj*/) { return 0; } PyObject * QuantityPy::number_invert_handler (PyObject *self) { PyErr_SetString(PyExc_TypeError, "bad operand type for unary ~"); return 0; } PyObject * QuantityPy::number_lshift_handler (PyObject *self, PyObject *other) { PyErr_SetString(PyExc_TypeError, "unsupported operand type(s) for <<"); return 0; } PyObject * QuantityPy::number_rshift_handler (PyObject *self, PyObject *other) { PyErr_SetString(PyExc_TypeError, "unsupported operand type(s) for >>"); return 0; } PyObject * QuantityPy::number_and_handler (PyObject *self, PyObject *other) { PyErr_SetString(PyExc_TypeError, "unsupported operand type(s) for &"); return 0; } PyObject * QuantityPy::number_xor_handler (PyObject *self, PyObject *other) { PyErr_SetString(PyExc_TypeError, "unsupported operand type(s) for ^"); return 0; } PyObject * QuantityPy::number_or_handler (PyObject *self, PyObject *other) { PyErr_SetString(PyExc_TypeError, "unsupported operand type(s) for |"); return 0; } int QuantityPy::number_coerce_handler (PyObject **self, PyObject **other) { return 1; } PyObject * QuantityPy::number_oct_handler (PyObject *self) { PyErr_SetString(PyExc_TypeError, "oct() argument can't be converted to oct"); return 0; } PyObject * QuantityPy::number_hex_handler (PyObject *self) { PyErr_SetString(PyExc_TypeError, "hex() argument can't be converted to hex"); return 0; }