/*************************************************************************** * Copyright (c) 2007 Werner Mayer * * * * 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 # include # include #endif #include #include #include #include #include #include #include "SoFCDB.h" #include "ViewProvider.h" #include "PythonWrapper.h" #include // inclusion of the generated files (generated out of ViewProviderPy2.xml) #include #include #include #include #include #include #include #include #include #include #include using namespace Gui; // returns a string which represent the object e.g. when printed in python std::string ViewProviderPy::representation(void) const { return ""; } PyObject* ViewProviderPy::addProperty(PyObject *args) { char *sType,*sName=0,*sGroup=0,*sDoc=0; short attr=0; std::string sDocStr; PyObject *ro = Py_False, *hd = Py_False; if (!PyArg_ParseTuple(args, "s|ssethO!O!", &sType,&sName,&sGroup,"utf-8",&sDoc,&attr, &PyBool_Type, &ro, &PyBool_Type, &hd)) // convert args: Python->C return NULL; // NULL triggers exception if (sDoc) { sDocStr = sDoc; PyMem_Free(sDoc); } App::Property* prop=0; try { prop = getViewProviderPtr()->addDynamicProperty(sType,sName,sGroup,sDocStr.c_str(),attr, PyObject_IsTrue(ro) ? true : false, PyObject_IsTrue(hd) ? true : false); } catch (const Base::Exception& e) { throw Py::RuntimeError(e.what()); } if (!prop) { std::stringstream str; str << "No property found of type '" << sType << "'" << std::ends; throw Py::Exception(Base::BaseExceptionFreeCADError,str.str()); } return Py::new_reference_to(this); } PyObject* ViewProviderPy::removeProperty(PyObject *args) { char *sName; if (!PyArg_ParseTuple(args, "s", &sName)) return NULL; try { bool ok = getViewProviderPtr()->removeDynamicProperty(sName); return Py_BuildValue("O", (ok ? Py_True : Py_False)); } catch (const Base::Exception& e) { throw Py::RuntimeError(e.what()); } } PyObject* ViewProviderPy::supportedProperties(PyObject *args) { if (!PyArg_ParseTuple(args, "")) // convert args: Python->C return NULL; // NULL triggers exception std::vector ary; Base::Type::getAllDerivedFrom(App::Property::getClassTypeId(), ary); Py::List res; for (std::vector::iterator it = ary.begin(); it != ary.end(); ++it) { Base::BaseClass *data = static_cast(it->createInstance()); if (data) { delete data; res.append(Py::String(it->getName())); } } return Py::new_reference_to(res); } PyObject* ViewProviderPy::show(PyObject *args) { if (!PyArg_ParseTuple(args, "")) // convert args: Python->C return NULL; // NULL triggers exception PY_TRY { getViewProviderPtr()->show(); Py_Return; } PY_CATCH; } PyObject* ViewProviderPy::hide(PyObject *args) { if (!PyArg_ParseTuple(args, "")) // convert args: Python->C return NULL; // NULL triggers exception PY_TRY { getViewProviderPtr()->hide(); Py_Return; } PY_CATCH; } PyObject* ViewProviderPy::isVisible(PyObject *args) { if (!PyArg_ParseTuple(args, "")) // convert args: Python->C return NULL; // NULL triggers exception PY_TRY { return Py_BuildValue("O", (getViewProviderPtr()->isShow() ? Py_True : Py_False)); } PY_CATCH; } PyObject* ViewProviderPy::canDragObject(PyObject *args) { PyObject *obj = Py_None; if (!PyArg_ParseTuple(args, "|O", &obj)) return NULL; PY_TRY { bool ret; if(obj == Py_None) ret = getViewProviderPtr()->canDragObjects(); else if(!PyObject_TypeCheck(obj,&App::DocumentObjectPy::Type)) { PyErr_SetString(PyExc_TypeError, "exepcting a type of DocumentObject"); return 0; }else ret = getViewProviderPtr()->canDragObject( static_cast(obj)->getDocumentObjectPtr()); return Py::new_reference_to(Py::Boolean(ret)); } PY_CATCH; } PyObject* ViewProviderPy::canDropObject(PyObject *args) { PyObject *obj = Py_None; PyObject *owner = Py_None; PyObject *pyElements = Py_None; const char *subname = 0; if (!PyArg_ParseTuple(args, "|OOsO", &obj,&owner,&subname,&pyElements)) return NULL; PY_TRY { bool ret; if(obj == Py_None) ret = getViewProviderPtr()->canDropObjects(); else if(!PyObject_TypeCheck(obj,&App::DocumentObjectPy::Type)) { PyErr_SetString(PyExc_TypeError, "exepcting 'obj' to be of type DocumentObject"); return 0; } auto pcObject = static_cast(obj)->getDocumentObjectPtr(); App::DocumentObject *pcOwner = 0; if(owner!=Py_None) { if(!PyObject_TypeCheck(owner,&App::DocumentObjectPy::Type)) { PyErr_SetString(PyExc_TypeError, "exepcting 'owner' to be of type DocumentObject"); return NULL; } pcOwner = static_cast(owner)->getDocumentObjectPtr(); } App::PropertyStringList elements; if(pyElements!=Py_None) { try { elements.setPyObject(pyElements); }catch(...) { PyErr_SetString(PyExc_TypeError, "exepcting the forth argument to be of type sequence of strings"); return 0; } } ret = getViewProviderPtr()->canDropObjectEx(pcObject,pcOwner,subname,elements.getValues()); return Py::new_reference_to(Py::Boolean(ret)); } PY_CATCH; } PyObject* ViewProviderPy::canDragAndDropObject(PyObject *args) { PyObject *obj; if (!PyArg_ParseTuple(args, "O!", &App::DocumentObjectPy::Type,&obj)) return NULL; PY_TRY { bool ret = getViewProviderPtr()->canDragAndDropObject( static_cast(obj)->getDocumentObjectPtr()); return Py::new_reference_to(Py::Boolean(ret)); } PY_CATCH; } PyObject* ViewProviderPy::dropObject(PyObject *args) { PyObject *obj; PyObject *owner = Py_None; PyObject *pyElements = Py_None; const char *subname = 0; if (!PyArg_ParseTuple(args, "O!|OsO", &App::DocumentObjectPy::Type,&obj,&owner,&subname,&pyElements)) return NULL; PY_TRY { App::DocumentObject *pcOwner = 0; if(owner!=Py_None) { if(!PyObject_TypeCheck(owner,&App::DocumentObjectPy::Type)) { PyErr_SetString(PyExc_TypeError, "exepcting 'owner' to be of type DocumentObject"); return NULL; } pcOwner = static_cast(owner)->getDocumentObjectPtr(); } App::PropertyStringList elements; if(pyElements!=Py_None) { try { elements.setPyObject(pyElements); }catch(...) { PyErr_SetString(PyExc_TypeError, "exepcting the forth argument to be of type sequence of strings"); return 0; } } auto ret = getViewProviderPtr()->dropObjectEx( static_cast(obj)->getDocumentObjectPtr(), pcOwner, subname,elements.getValues()); return Py::new_reference_to(Py::String(ret)); } PY_CATCH; } PyObject* ViewProviderPy::dragObject(PyObject *args) { PyObject *obj; if (!PyArg_ParseTuple(args, "O!", &App::DocumentObjectPy::Type,&obj)) return NULL; PY_TRY { getViewProviderPtr()->dragObject( static_cast(obj)->getDocumentObjectPtr()); Py_Return; } PY_CATCH; } PyObject* ViewProviderPy::replaceObject(PyObject *args) { PyObject *oldObj; PyObject *newObj; if (!PyArg_ParseTuple(args, "O!O!", &App::DocumentObjectPy::Type,&oldObj, &App::DocumentObjectPy::Type,&newObj)) return NULL; PY_TRY { int ret = getViewProviderPtr()->replaceObject( static_cast(oldObj)->getDocumentObjectPtr(), static_cast(newObj)->getDocumentObjectPtr()); return Py::new_reference_to(Py::Int(ret)); } PY_CATCH; } PyObject* ViewProviderPy::addDisplayMode(PyObject * args) { char* mode; PyObject* obj; if (!PyArg_ParseTuple(args, "Os", &obj, &mode)) return NULL; void* ptr = 0; try { Base::Interpreter().convertSWIGPointerObj("pivy.coin","_p_SoNode", obj, &ptr, 0); } catch (const Base::Exception& e) { PyErr_SetString(PyExc_RuntimeError, e.what()); return 0; } PY_TRY { SoNode* node = reinterpret_cast(ptr); getViewProviderPtr()->addDisplayMaskMode(node,mode); Py_Return; } PY_CATCH; } PyObject* ViewProviderPy::listDisplayModes(PyObject *args) { if (!PyArg_ParseTuple(args, "")) // convert args: Python->C return NULL; // NULL triggers exception PY_TRY { std::vector modes = getViewProviderPtr()->getDisplayModes(); PyObject* pyList = PyList_New(modes.size()); int i=0; for ( std::vector::iterator it = modes.begin(); it != modes.end(); ++it ) { PyObject* str = PyUnicode_FromString(it->c_str()); PyList_SetItem(pyList, i++, str); } return pyList; } PY_CATCH; } PyObject* ViewProviderPy::toString(PyObject *args) { if (!PyArg_ParseTuple(args, "")) // convert args: Python->C return NULL; // NULL triggers exception PY_TRY { std::string buffer = getViewProviderPtr()->toString(); return Py::new_reference_to(Py::String(buffer)); } PY_CATCH; } PyObject* ViewProviderPy::setTransformation(PyObject *args) { PyObject* p; Base::Matrix4D mat; if (PyArg_ParseTuple(args, "O!",&(Base::MatrixPy::Type),&p)) { mat = *static_cast(p)->getMatrixPtr(); getViewProviderPtr()->setTransformation(mat); Py_Return; } PyErr_Clear(); if (PyArg_ParseTuple(args, "O!",&(Base::PlacementPy::Type),&p)) { Base::PlacementPy* plc = static_cast(p); getViewProviderPtr()->setTransformation(plc->getPlacementPtr()->toMatrix()); Py_Return; } PyErr_SetString(Base::BaseExceptionFreeCADError, "Either set matrix or placement to set transformation"); return 0; } PyObject* ViewProviderPy::claimChildren(PyObject* args) { if (!PyArg_ParseTuple(args, "")) return NULL; // NULL triggers exception std::vector children = this->getViewProviderPtr()->claimChildren(); Py::List ret; for(App::DocumentObject* child: children){ if (child) ret.append(Py::asObject(child->getPyObject())); else ret.append(Py::None()); } return Py::new_reference_to(ret); } PyObject* ViewProviderPy::partialRender(PyObject* args) { PyObject *value = Py_None; PyObject *clear = Py_False; if (!PyArg_ParseTuple(args, "|OO",&value,&clear)) return NULL; // NULL triggers exception std::vector values; if(value != Py_None) { PyObject *item = 0; Py_ssize_t nSize; if (PyList_Check(value) || PyTuple_Check(value)) nSize = PySequence_Size(value); else { item = value; value = 0; nSize = 1; } values.resize(nSize); for (Py_ssize_t i = 0; i < nSize; ++i) { if(value) item = PySequence_GetItem(value, i); if (PyUnicode_Check(item)) { values[i] = PyUnicode_AsUTF8(item); } else { std::string error = std::string("type must be str or unicode"); error += " not, "; error += item->ob_type->tp_name; throw Base::TypeError(error); } } } Py::Int ret(getViewProviderPtr()->partialRender(values,PyObject_IsTrue(clear))); return Py::new_reference_to(ret); } PyObject* ViewProviderPy::getElementColors(PyObject* args) { const char *element = 0; if (!PyArg_ParseTuple(args, "|s", &element)) return 0; Py::Dict dict; for(auto &v : getViewProviderPtr()->getElementColors(element)) { auto &c = v.second; dict.setItem(Py::String(v.first), Py::TupleN(Py::Float(c.r),Py::Float(c.g),Py::Float(c.b),Py::Float(c.a))); } return Py::new_reference_to(dict); } PyObject* ViewProviderPy::setElementColors(PyObject* args) { PyObject *pyObj; if (!PyArg_ParseTuple(args, "O", &pyObj)) return 0; if(!PyDict_Check(pyObj)) throw Py::TypeError("Expect a dict"); std::map colors; Py::Dict dict(pyObj); for(auto it=dict.begin();it!=dict.end();++it) { const auto &value = *it; if(!value.first.isString() || !value.second.isSequence()) throw Py::TypeError("Expect the dictionary to contain items of type elementName:(r,g,b,a)"); App::PropertyColor prop; prop.setPyObject(value.second.ptr()); colors[value.first.as_string()] = prop.getValue(); } getViewProviderPtr()->setElementColors(colors); Py_Return; } PyObject* ViewProviderPy::getElementPicked(PyObject* args) { PyObject *obj; if (!PyArg_ParseTuple(args, "O",&obj)) return NULL; void *ptr = 0; Base::Interpreter().convertSWIGPointerObj("pivy.coin", "_p_SoPickedPoint", obj, &ptr, 0); SoPickedPoint *pp = reinterpret_cast(ptr); if(!pp) throw Base::TypeError("type must be of coin.SoPickedPoint"); std::string name; if(!getViewProviderPtr()->getElementPicked(pp,name)) Py_Return; return Py::new_reference_to(Py::String(name)); } PyObject* ViewProviderPy::getDetailPath(PyObject* args) { const char *sub; PyObject *path; PyObject *append = Py_True; if (!PyArg_ParseTuple(args, "sO|O",&sub,&path,&append)) return NULL; void *ptr = 0; Base::Interpreter().convertSWIGPointerObj("pivy.coin", "_p_SoPath", path, &ptr, 0); SoPath *pPath = reinterpret_cast(ptr); if(!pPath) throw Base::TypeError("type must be of coin.SoPath"); SoDetail *det = 0; if(!getViewProviderPtr()->getDetailPath( sub,static_cast(pPath),PyObject_IsTrue(append),det)) { if(det) delete det; Py_Return; } if(!det) return Py::new_reference_to(Py::True()); return Base::Interpreter().createSWIGPointerObj("pivy.coin", "_p_SoDetail", (void*)det, 0); } PyObject *ViewProviderPy::signalChangeIcon(PyObject *args) { if (!PyArg_ParseTuple(args, "")) return NULL; getViewProviderPtr()->signalChangeIcon(); Py_Return; } PyObject *ViewProviderPy::getBoundingBox(PyObject *args) { PyObject *transform=Py_True; PyObject *pyView = 0; const char *subname = 0; if (!PyArg_ParseTuple(args, "|sOO!", &subname,&transform,View3DInventorPy::type_object(),&pyView)) return NULL; PY_TRY { View3DInventor *view = 0; if(pyView) view = static_cast(pyView)->getView3DIventorPtr(); auto bbox = getViewProviderPtr()->getBoundingBox(subname,PyObject_IsTrue(transform),view); Py::Object ret(new Base::BoundBoxPy(new Base::BoundBox3d(bbox))); return Py::new_reference_to(ret); } PY_CATCH; } PyObject *ViewProviderPy::doubleClicked(PyObject *args) { if(!PyArg_ParseTuple(args, "")) return 0; PY_TRY { return Py::new_reference_to(Py::Boolean(getViewProviderPtr()->doubleClicked())); }PY_CATCH; } PyObject *ViewProviderPy::getCustomAttributes(const char* attr) const { // search for dynamic property App::Property* prop = getViewProviderPtr()->getDynamicPropertyByName(attr); if (prop) return prop->getPyObject(); else return 0; } int ViewProviderPy::setCustomAttributes(const char* attr, PyObject* value) { // search for dynamic property try { App::Property* prop = getViewProviderPtr()->getDynamicPropertyByName(attr); if (prop) { prop->setPyObject(value); return 1; } return 0; } catch (Base::Exception &exc) { PyErr_Format(PyExc_AttributeError, "Attribute (Name: %s) error: '%s' ", attr, exc.what()); return -1; } catch (...) { PyErr_Format(PyExc_AttributeError, "Unknown error in attribute %s", attr); return -1; } } Py::Object ViewProviderPy::getAnnotation(void) const { try { auto node = getViewProviderPtr()->getAnnotation(); PyObject* Ptr = Base::Interpreter().createSWIGPointerObj("pivy.coin", "_p_SoSeparator", node, 1); node->ref(); return Py::Object(Ptr, true); } catch (const Base::Exception& e) { throw Py::RuntimeError(e.what()); } } void ViewProviderPy::setAnnotation(Py::Object) { } Py::Object ViewProviderPy::getRootNode(void) const { try { SoSeparator* node = getViewProviderPtr()->getRoot(); PyObject* Ptr = Base::Interpreter().createSWIGPointerObj("pivy.coin","_p_SoSeparator", node, 1); node->ref(); return Py::Object(Ptr, true); } catch (const Base::Exception& e) { throw Py::RuntimeError(e.what()); } } void ViewProviderPy::setRootNode(Py::Object) { } Py::Object ViewProviderPy::getSwitchNode(void) const { try { SoSwitch* node = getViewProviderPtr()->getModeSwitch(); PyObject* Ptr = Base::Interpreter().createSWIGPointerObj("pivy.coin","_p_SoSwitch", node, 1); node->ref(); return Py::Object(Ptr, true); } catch (const Base::Exception& e) { throw Py::RuntimeError(e.what()); } } void ViewProviderPy::setSwitchNode(Py::Object) { } Py::String ViewProviderPy::getIV(void) const { std::string buf = Gui::SoFCDB::writeNodesToString(getViewProviderPtr()->getRoot()); return Py::String(buf); } Py::Object ViewProviderPy::getIcon(void) const { #if 0 QByteArray ba; QDataStream str(&ba, QIODevice::WriteOnly); QIcon icon = getViewProviderPtr()->getIcon(); str << icon; return Py::String(ba.constData(), ba.size()); #else PythonWrapper wrap; wrap.loadGuiModule(); wrap.loadWidgetsModule(); QIcon icon = getViewProviderPtr()->getIcon(); return wrap.fromQIcon(new QIcon(icon)); #endif } Py::Int ViewProviderPy::getDefaultMode(void) const { return Py::Int((long)getViewProviderPtr()->getDefaultMode()); } void ViewProviderPy::setDefaultMode(Py::Int arg) { return getViewProviderPtr()->setDefaultMode(arg); } Py::Boolean ViewProviderPy::getCanRemoveChildrenFromRoot() const { return Py::Boolean(getViewProviderPtr()->canRemoveChildrenFromRoot()); } Py::Boolean ViewProviderPy::getLinkVisibility() const { return Py::Boolean(getViewProviderPtr()->isLinkVisible()); } void ViewProviderPy::setLinkVisibility(Py::Boolean arg) { getViewProviderPtr()->setLinkVisible(arg); } Py::String ViewProviderPy::getDropPrefix() const { return Py::String(getViewProviderPtr()->getDropPrefix()); }