331 lines
13 KiB
C++
331 lines
13 KiB
C++
/***************************************************************************
|
|
* Copyright (c) 2015 Victor Titov (DeepSOIC) <vv.titov@gmail.com> *
|
|
* *
|
|
* 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/Console.h>
|
|
|
|
#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<Attacher::AttachEngine*>(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<App::PropertyPlacement>(
|
|
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<Part::AttachExtension>;
|
|
}
|
|
|