/*************************************************************************** * Copyright (c) Jürgen Riegel * * * * 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" #ifndef _PreComp_ # include #endif #include #include "UnitsApi.h" #include "Quantity.h" #include "QuantityPy.h" using namespace Base; //************************************************************************** // Python stuff of UnitsApi // UnitsApi Methods PyMethodDef UnitsApi::Methods[] = { {"parseQuantity", UnitsApi::sParseQuantity, METH_VARARGS, "parseQuantity(string) -> Base.Quantity()\n\n" "calculate a mathematical expression with units to a quantity object. \n" "can be used for simple unit translation like: \n" "parseQuantity('10m')\n" "or for more complex espressions:\n" "parseQuantity('sin(pi)/50.0 m/s^2')\n" }, {"listSchemas", UnitsApi::sListSchemas, METH_VARARGS, "listSchemas() -> a tuple of schemas\n\n" "listSchemas(int) -> description of the given schema\n\n" }, {"getSchema", UnitsApi::sGetSchema, METH_VARARGS, "getSchema() -> int\n\n" "The int is the position of the tuple returned by listSchemas" }, {"setSchema", UnitsApi::sSetSchema, METH_VARARGS, "setSchema(int) -> None\n\n" "Sets the current schema to the given number, if possible" }, {"schemaTranslate", UnitsApi::sSchemaTranslate, METH_VARARGS, "schemaTranslate(Quantity, int) -> tuple\n\n" "Translate a quantity to a given schema" }, {"toNumber", UnitsApi::sToNumber, METH_VARARGS, "toNumber(Quantity or float, [format='g', decimals=-1]) -> str\n\n" "Convert a quantity or float to a string" }, {nullptr, nullptr, 0, nullptr} /* Sentinel */ }; PyObject* UnitsApi::sParseQuantity(PyObject * /*self*/, PyObject *args) { char *pstr; if (!PyArg_ParseTuple(args, "et", "utf-8", &pstr)) return nullptr; Quantity rtn; QString qstr = QString::fromUtf8(pstr); PyMem_Free(pstr); try { rtn = Quantity::parse(qstr); } catch (const Base::ParserError&) { PyErr_Format(PyExc_ValueError, "invalid unit expression \n"); return nullptr; } return new QuantityPy(new Quantity(rtn)); } PyObject* UnitsApi::sListSchemas(PyObject * /*self*/, PyObject *args) { if (PyArg_ParseTuple(args, "")) { int num = static_cast(UnitSystem::NumUnitSystemTypes); Py::Tuple tuple(num); for (int i=0; i(i)))); } return Py::new_reference_to(tuple); } PyErr_Clear(); int index; if (PyArg_ParseTuple(args, "i", &index)) { int num = static_cast(UnitSystem::NumUnitSystemTypes); if (index < 0 || index >= num) { PyErr_SetString(PyExc_ValueError, "invalid schema value"); return nullptr; } return Py_BuildValue("s", UnitsApi::getDescription(static_cast(index))); } PyErr_SetString(PyExc_TypeError, "int or empty argument list expected"); return nullptr; } PyObject* UnitsApi::sGetSchema(PyObject * /*self*/, PyObject *args) { if (!PyArg_ParseTuple(args, "")) return nullptr; return Py_BuildValue("i", static_cast(currentSystem)); } PyObject* UnitsApi::sSetSchema(PyObject * /*self*/, PyObject *args) { PyErr_Clear(); int index; if (PyArg_ParseTuple(args, "i", &index)) { int num = static_cast(UnitSystem::NumUnitSystemTypes); if (index < 0 || index >= num) { PyErr_SetString(PyExc_ValueError, "invalid schema value"); return nullptr; } setSchema(static_cast(index)); } Py_Return; } PyObject* UnitsApi::sSchemaTranslate(PyObject * /*self*/, PyObject *args) { PyObject* q; int index; if (!PyArg_ParseTuple(args, "O!i", &(QuantityPy::Type), &q, &index)) return nullptr; Quantity quant; quant = *static_cast(q)->getQuantityPtr(); std::unique_ptr schema(createSchema(static_cast(index))); if (!schema.get()) { PyErr_SetString(PyExc_ValueError, "invalid schema value"); return nullptr; } double factor; QString uus; QString uss = schema->schemaTranslate(quant, factor, uus); Py::Tuple res(3); res[0] = Py::String(uss.toUtf8(),"utf-8"); res[1] = Py::Float(factor); res[2] = Py::String(uus.toUtf8(),"utf-8"); return Py::new_reference_to(res); } PyObject* UnitsApi::sToNumber(PyObject * /*self*/, PyObject *args) { double value; char* format = "g"; int decimals; do { PyObject* q; if (PyArg_ParseTuple(args, "O!|si", &(QuantityPy::Type), &q, &format, &decimals)) { value = static_cast(q)->getQuantityPtr()->getValue(); break; } PyErr_Clear(); if (PyArg_ParseTuple(args, "d|si", &value, &format, &decimals)) { break; } PyErr_SetString(PyExc_TypeError, "toNumber(Quantity or float, [format='g', decimals=-1])"); return nullptr; } while (false); if (strlen(format) != 1) { PyErr_SetString(PyExc_ValueError, "Format string hasn't length of 1"); return nullptr; } bool ok; QuantityFormat qf; qf.format = QuantityFormat::toFormat(format[0], &ok); qf.precision = decimals; if (!ok) { PyErr_SetString(PyExc_ValueError, "Invalid format string"); return nullptr; } QString string = toNumber(value, qf); return Py::new_reference_to(Py::String(string.toStdString())); }