/*************************************************************************** * Copyright (c) 2015 Victor Titov (DeepSOIC) * * * * 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 #include "AttachExtension.h" #include "AttachExtensionPy.h" using namespace Part; using namespace Attacher; EXTENSION_PROPERTY_SOURCE(Part::AttachExtension, App::DocumentObjectExtension) AttachExtension::AttachExtension() { EXTENSION_ADD_PROPERTY_TYPE(AttacherType, ("Attacher::AttachEngine3D"), "Attachment",(App::PropertyType)(App::Prop_None),"Class name of attach engine object driving the attachment."); this->AttacherType.setStatus(App::Property::Status::Hidden, true); EXTENSION_ADD_PROPERTY_TYPE(AttachmentSupport, (nullptr,nullptr), "Attachment",(App::PropertyType)(App::Prop_None),"Support of the 2D geometry"); EXTENSION_ADD_PROPERTY_TYPE(MapMode, (mmDeactivated), "Attachment", App::Prop_None, "Mode of attachment to other object"); MapMode.setEditorName("PartGui::PropertyEnumAttacherItem"); MapMode.setEnums(AttachEngine::eMapModeStrings); //a rough test if mode string list in Attacher.cpp is in sync with eMapMode enum. assert(MapMode.getEnumVector().size() == mmDummy_NumberOfModes); EXTENSION_ADD_PROPERTY_TYPE(MapReversed, (false), "Attachment", App::Prop_None, "Reverse Z direction (flip sketch upside down)"); EXTENSION_ADD_PROPERTY_TYPE(MapPathParameter, (0.0), "Attachment", App::Prop_None, "Sets point of curve to map the sketch to. 0..1 = start..end"); EXTENSION_ADD_PROPERTY_TYPE(AttachmentOffset, (Base::Placement()), "Attachment", App::Prop_None, "Extra placement to apply in addition to attachment (in local coordinates)"); // Only show these properties when applicable. Controlled by extensionOnChanged this->MapPathParameter.setStatus(App::Property::Status::Hidden, true); this->MapReversed.setStatus(App::Property::Status::Hidden, true); this->AttachmentOffset.setStatus(App::Property::Status::Hidden, true); setAttacher(new AttachEngine3D);//default attacher initExtensionType(AttachExtension::getExtensionClassTypeId()); } AttachExtension::~AttachExtension() { if(_attacher) delete _attacher; } void AttachExtension::setAttacher(AttachEngine* attacher) { if (_attacher) delete _attacher; _attacher = attacher; if (_attacher){ const char* typeName = attacher->getTypeId().getName(); if(strcmp(this->AttacherType.getValue(),typeName)!=0) //make sure we need to change, to break recursive onChange->changeAttacherType->onChange... this->AttacherType.setValue(typeName); updateAttacherVals(); } else { if (strlen(AttacherType.getValue()) != 0){ //make sure we need to change, to break recursive onChange->changeAttacherType->onChange... this->AttacherType.setValue(""); } } } bool AttachExtension::changeAttacherType(const char* typeName) { //check if we need to actually change anything if (_attacher){ if (strcmp(_attacher->getTypeId().getName(),typeName)==0){ return false; } } else if (strlen(typeName) == 0){ return false; } if (strlen(typeName) == 0){ setAttacher(nullptr); return true; } Base::Type t = Base::Type::fromName(typeName); if (t.isDerivedFrom(AttachEngine::getClassTypeId())){ AttachEngine* pNewAttacher = static_cast(Base::Type::createInstanceByName(typeName)); this->setAttacher(pNewAttacher); return true; } std::stringstream errMsg; errMsg << "Object if this type is not derived from AttachEngine: " << typeName; throw AttachEngineException(errMsg.str()); } bool AttachExtension::positionBySupport() { _active = 0; if (!_attacher) throw Base::RuntimeError("AttachExtension: can't positionBySupport, because no AttachEngine is set."); updateAttacherVals(); try { if (_attacher->mapMode == mmDeactivated) return false; getPlacement().setValue(_attacher->calculateAttachedPlacement(getPlacement().getValue())); _active = 1; return true; } catch (ExceptionCancel&) { //disabled, don't do anything return false; }; } bool AttachExtension::isAttacherActive() const { if(_active < 0) { _active = 0; try { _attacher->calculateAttachedPlacement(getPlacement().getValue()); _active = 1; } catch (ExceptionCancel&) { } } return _active!=0; } short int AttachExtension::extensionMustExecute() { return DocumentObjectExtension::extensionMustExecute(); } App::DocumentObjectExecReturn *AttachExtension::extensionExecute() { if(this->isTouched_Mapping()) { try{ positionBySupport(); // we let all Base::Exceptions thru, so that App:DocumentObject can take appropriate action /*} catch (Base::Exception &e) { return new App::DocumentObjectExecReturn(e.what());*/ // Convert OCC exceptions to Base::Exception } catch (Standard_Failure &e){ throw Base::RuntimeError(e.GetMessageString()); // return new App::DocumentObjectExecReturn(e.GetMessageString()); } } return App::DocumentObjectExtension::extensionExecute(); } void AttachExtension::extensionOnChanged(const App::Property* prop) { if(! getExtendedObject()->isRestoring()){ if ((prop == &AttachmentSupport || prop == &MapMode || prop == &MapPathParameter || prop == &MapReversed || prop == &AttachmentOffset)){ bool bAttached = false; try{ bAttached = positionBySupport(); } catch (Base::Exception &e) { getExtendedObject()->setStatus(App::Error, true); Base::Console().Error("PositionBySupport: %s\n",e.what()); //set error message - how? } catch (Standard_Failure &e){ getExtendedObject()->setStatus(App::Error, true); Base::Console().Error("PositionBySupport: %s\n",e.GetMessageString()); } // Hide properties when not applicable to reduce user confusion eMapMode mmode = eMapMode(this->MapMode.getValue()); bool modeIsPointOnCurve = mmode == mmNormalToPath || mmode == mmFrenetNB || mmode == mmFrenetTN || mmode == mmFrenetTB || mmode == mmRevolutionSection || mmode == mmConcentric; // MapPathParameter is only used if there is a reference to one edge and not edge + vertex bool hasOneRef = false; if (_attacher && _attacher->references.getSubValues().size() == 1) { hasOneRef = true; } this->MapPathParameter.setStatus(App::Property::Status::Hidden, !bAttached || !(modeIsPointOnCurve && hasOneRef)); this->MapReversed.setStatus(App::Property::Status::Hidden, !bAttached); this->AttachmentOffset.setStatus(App::Property::Status::Hidden, !bAttached); getPlacement().setReadOnly(bAttached && mmode != mmTranslate); //for mmTranslate, orientation should remain editable even when attached. } } if(prop == &(this->AttacherType)){ this->changeAttacherType(this->AttacherType.getValue()); } App::DocumentObjectExtension::extensionOnChanged(prop); } bool AttachExtension::extensionHandleChangedPropertyName(Base::XMLReader &reader, const char* TypeName, const char *PropName) { // superPlacement -> AttachmentOffset Base::Type type = Base::Type::fromName(TypeName); if (strcmp(PropName, "superPlacement") == 0 && AttachmentOffset.getClassTypeId() == type) { AttachmentOffset.Restore(reader); return true; } // Support -> AttachmentSupport else if (strcmp(PropName, "Support") == 0) { // At one point, the type of Support changed from PropertyLinkSub to its present type of PropertyLinkSubList. // Later, the property name changed to AttachmentSupport App::PropertyLinkSub tmp; if (0 == strcmp(tmp.getTypeId().getName(),TypeName)) { tmp.setContainer(this->getExtendedContainer()); tmp.Restore(reader); AttachmentSupport.setValue(tmp.getValue(), tmp.getSubValues()); this->MapMode.setValue(Attacher::mmFlatFace); return true; } else if (AttachmentSupport.getClassTypeId() == type) { AttachmentSupport.Restore(reader); return true; } } return App::DocumentObjectExtension::extensionHandleChangedPropertyName(reader, TypeName, PropName); } void AttachExtension::onExtendedDocumentRestored() { try { bool bAttached = positionBySupport(); // Hide properties when not applicable to reduce user confusion eMapMode mmode = eMapMode(this->MapMode.getValue()); bool modeIsPointOnCurve = (mmode == mmNormalToPath || mmode == mmFrenetNB || mmode == mmFrenetTN || mmode == mmFrenetTB || mmode == mmRevolutionSection || mmode == mmConcentric); // MapPathParameter is only used if there is a reference to one edge and not edge + vertex bool hasOneRef = false; if (_attacher && _attacher->references.getSubValues().size() == 1) { hasOneRef = true; } this->MapPathParameter.setStatus(App::Property::Status::Hidden, !bAttached || !(modeIsPointOnCurve && hasOneRef)); this->MapReversed.setStatus(App::Property::Status::Hidden, !bAttached); this->AttachmentOffset.setStatus(App::Property::Status::Hidden, !bAttached); getPlacement().setReadOnly(bAttached && mmode != mmTranslate); //for mmTranslate, orientation should remain editable even when attached. } catch (Base::Exception&) { } catch (Standard_Failure &) { } } void AttachExtension::updateAttacherVals() { if (!_attacher) return; _attacher->setUp(this->AttachmentSupport, eMapMode(this->MapMode.getValue()), this->MapReversed.getValue(), this->MapPathParameter.getValue(), 0.0,0.0, this->AttachmentOffset.getValue()); } App::PropertyPlacement& AttachExtension::getPlacement() const { auto pla = Base::freecad_dynamic_cast( getExtendedObject()->getPropertyByName("Placement")); if(!pla) throw Base::RuntimeError("AttachExtension cannot find placement property"); return *pla; } PyObject* AttachExtension::getExtensionPyObject() { if (ExtensionPythonObject.is(Py::_None())){ // ref counter is set to 1 ExtensionPythonObject = Py::Object(new AttachExtensionPy(this),true); } return Py::new_reference_to(ExtensionPythonObject); } // ------------------------------------------------ AttachEngineException::AttachEngineException() : Base::Exception() { } AttachEngineException::AttachEngineException(const char * sMessage) : Base::Exception(sMessage) { } AttachEngineException::AttachEngineException(const std::string& sMessage) : Base::Exception(sMessage) { } namespace App { /// @cond DOXERR EXTENSION_PROPERTY_SOURCE_TEMPLATE(Part::AttachExtensionPython, Part::AttachExtension) /// @endcond // explicit template instantiation template class PartExport ExtensionPythonT; }