/*************************************************************************** * Copyright (c) 2015 Stefan Tröger * * * * 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 #endif #include "ShapeBinder.h" #include #ifndef M_PI #define M_PI 3.14159265358979323846 #endif using namespace PartDesign; // ============================================================================ PROPERTY_SOURCE(PartDesign::ShapeBinder, Part::Feature) ShapeBinder::ShapeBinder() { ADD_PROPERTY_TYPE(Support, (0,0), "",(App::PropertyType)(App::Prop_None),"Support of the geometry"); } ShapeBinder::~ShapeBinder() { } short int ShapeBinder::mustExecute(void) const { if(Support.isTouched()) return 1; return Part::Feature::mustExecute(); } App::DocumentObjectExecReturn* ShapeBinder::execute(void) { if(! this->isRestoring()){ Part::Feature* obj = nullptr; std::vector subs; ShapeBinder::getFilterdReferences(&Support, obj, subs); Shape.setValue(ShapeBinder::buildShapeFromReferences(obj, subs)._Shape); } return Part::Feature::execute(); } void ShapeBinder::getFilterdReferences(App::PropertyLinkSubList* prop, Part::Feature*& obj, std::vector< std::string >& subobjects) { obj = nullptr; subobjects.clear(); auto objs = prop->getValues(); auto subs = prop->getSubValues(); if(objs.empty()) { return; } //we only allow one part feature, so get the first one we find int index = 0; while(!objs[index]->isDerivedFrom(Part::Feature::getClassTypeId()) && index < objs.size()) index++; //do we have any part feature? if(index >= objs.size()) return; obj = static_cast(objs[index]); //if we have no subshpape we use the whole shape if(subs[index].empty()) { return; } //collect all subshapes for the object index = 0; for(std::string sub : subs) { //we only allow subshapes from a single Part::Feature if(objs[index] != obj) continue; //in this mode the full shape is not allowed, as we already started the subshape //processing if(sub.empty()) continue; subobjects.push_back(sub); } } TopoShape ShapeBinder::buildShapeFromReferences( Part::Feature* obj, std::vector< std::string > subs) { if(!obj) return TopoDS_Shape(); if(subs.empty()) return obj->Shape.getShape(); //if we use multiple subshapes we build a shape from them by fusing them together TopoShape base; std::vector operators; for(std::string sub : subs) { if(base.isNull()) base = obj->Shape.getShape().getSubShape(sub.c_str()); else operators.push_back(obj->Shape.getShape().getSubShape(sub.c_str())); } try { if(!operators.empty() && !base.isNull()) return base.multiFuse(operators); } catch(...) { return base; } return base; } PROPERTY_SOURCE(PartDesign::ShapeBinder2D, Part::Part2DObject) ShapeBinder2D::ShapeBinder2D() { } ShapeBinder2D::~ShapeBinder2D() { } App::DocumentObjectExecReturn* ShapeBinder2D::execute(void) { if(! this->isRestoring()){ Part::Feature* obj = nullptr; std::vector subs; ShapeBinder::getFilterdReferences(&Support, obj, subs); Shape.setValue(ShapeBinder::buildShapeFromReferences(obj, subs)._Shape); } return Part::Part2DObject::execute(); }