513 lines
18 KiB
C++
513 lines
18 KiB
C++
/***************************************************************************
|
|
* Copyright (c) 2002 Jürgen Riegel <juergen.riegel@web.de> *
|
|
* *
|
|
* 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 <sstream>
|
|
# include <iostream>
|
|
# include <iterator>
|
|
#include <Precision.hxx>
|
|
#include <cmath>
|
|
#endif
|
|
|
|
#include <Base/Exception.h>
|
|
#include <Base/Console.h>
|
|
#include <Base/FileInfo.h>
|
|
#include <Base/Parameter.h>
|
|
#include <Base/UnitsApi.h>
|
|
|
|
#include <App/Application.h>
|
|
#include <App/Document.h>
|
|
#include <boost/regex.hpp>
|
|
|
|
#include "DrawPage.h"
|
|
#include "DrawView.h"
|
|
#include "DrawProjGroup.h"
|
|
#include "DrawViewClip.h"
|
|
#include "DrawTemplate.h"
|
|
#include "DrawViewCollection.h"
|
|
#include "DrawViewPart.h"
|
|
#include "DrawViewDimension.h"
|
|
#include "DrawViewBalloon.h"
|
|
#include "DrawLeaderLine.h"
|
|
#include "Preferences.h"
|
|
|
|
#include <Mod/TechDraw/App/DrawPagePy.h> // generated from DrawPagePy.xml
|
|
|
|
using namespace TechDraw;
|
|
using namespace std;
|
|
|
|
|
|
//===========================================================================
|
|
// DrawPage
|
|
//===========================================================================
|
|
|
|
App::PropertyFloatConstraint::Constraints DrawPage::scaleRange = {Precision::Confusion(),
|
|
std::numeric_limits<double>::max(),
|
|
(0.1)}; // increment by 0.1
|
|
|
|
PROPERTY_SOURCE(TechDraw::DrawPage, App::DocumentObject)
|
|
|
|
const char* DrawPage::ProjectionTypeEnums[] = { "First Angle",
|
|
"Third Angle",
|
|
nullptr };
|
|
|
|
DrawPage::DrawPage(void)
|
|
{
|
|
static const char *group = "Page";
|
|
nowUnsetting = false;
|
|
forceRedraw(false);
|
|
|
|
ADD_PROPERTY_TYPE(KeepUpdated, (Preferences::keepPagesUpToDate()),
|
|
group, (App::PropertyType)(App::Prop_Output), "Keep page in sync with model");
|
|
ADD_PROPERTY_TYPE(Template, (nullptr), group, (App::PropertyType)(App::Prop_None), "Attached Template");
|
|
Template.setScope(App::LinkScope::Global);
|
|
ADD_PROPERTY_TYPE(Views, (nullptr), group, (App::PropertyType)(App::Prop_None), "Attached Views");
|
|
Views.setScope(App::LinkScope::Global);
|
|
|
|
// Projection Properties
|
|
ProjectionType.setEnums(ProjectionTypeEnums);
|
|
ADD_PROPERTY(ProjectionType, ((long)Preferences::projectionAngle()));
|
|
|
|
Base::Reference<ParameterGrp> hGrp = App::GetApplication().GetUserParameter().
|
|
GetGroup("BaseApp")->GetGroup("Preferences")->
|
|
GetGroup("Mod/TechDraw/General");
|
|
double defScale = hGrp->GetFloat("DefaultScale",1.0);
|
|
ADD_PROPERTY_TYPE(Scale, (defScale), group, (App::PropertyType)(App::Prop_None), "Scale factor for this Page");
|
|
|
|
ADD_PROPERTY_TYPE(NextBalloonIndex, (1), group, (App::PropertyType)(App::Prop_None),
|
|
"Auto-numbering for Balloons");
|
|
|
|
Scale.setConstraints(&scaleRange);
|
|
balloonParent = nullptr;
|
|
}
|
|
|
|
DrawPage::~DrawPage()
|
|
{
|
|
}
|
|
|
|
void DrawPage::onBeforeChange(const App::Property* prop)
|
|
{
|
|
App::DocumentObject::onBeforeChange(prop);
|
|
}
|
|
|
|
void DrawPage::onChanged(const App::Property* prop)
|
|
{
|
|
if ((prop == &KeepUpdated) &&
|
|
KeepUpdated.getValue()) {
|
|
if (!isRestoring() &&
|
|
!isUnsetting()) {
|
|
//would be nice if this message was displayed immediately instead of after the recomputeFeature
|
|
Base::Console().Message("Rebuilding Views for: %s/%s\n",getNameInDocument(),Label.getValue());
|
|
updateAllViews();
|
|
purgeTouched();
|
|
}
|
|
} else if (prop == &Template) {
|
|
if (!isRestoring() &&
|
|
!isUnsetting()) {
|
|
//nothing to page to do??
|
|
}
|
|
} else if(prop == &Scale) {
|
|
// touch all views in the Page as they may be dependent on this scale
|
|
// WF: not sure this loop is required. Views figure out their scale as required. but maybe
|
|
// this is needed just to mark the Views to recompute??
|
|
if (!isRestoring()) {
|
|
const std::vector<App::DocumentObject*> &vals = Views.getValues();
|
|
for(std::vector<App::DocumentObject *>::const_iterator it = vals.begin(); it < vals.end(); ++it) {
|
|
TechDraw::DrawView *view = dynamic_cast<TechDraw::DrawView *>(*it);
|
|
if (view && view->ScaleType.isValue("Page")) {
|
|
if(std::abs(view->Scale.getValue() - Scale.getValue()) > FLT_EPSILON) {
|
|
view->Scale.setValue(Scale.getValue());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else if (prop == &ProjectionType) {
|
|
// touch all ortho views in the Page as they may be dependent on Projection Type //(is this true?)
|
|
const std::vector<App::DocumentObject*> &vals = Views.getValues();
|
|
for(std::vector<App::DocumentObject *>::const_iterator it = vals.begin(); it < vals.end(); ++it) {
|
|
TechDraw::DrawProjGroup *view = dynamic_cast<TechDraw::DrawProjGroup *>(*it);
|
|
if (view && view->ProjectionType.isValue("Default")) {
|
|
view->ProjectionType.touch();
|
|
}
|
|
}
|
|
|
|
// TODO: Also update Template graphic.
|
|
|
|
}
|
|
App::DocumentObject::onChanged(prop);
|
|
}
|
|
|
|
//Page is just a container. It doesn't "do" anything.
|
|
App::DocumentObjectExecReturn *DrawPage::execute(void)
|
|
{
|
|
return App::DocumentObject::execute();
|
|
}
|
|
|
|
// this is now irrelevant, b/c DP::execute doesn't do anything.
|
|
short DrawPage::mustExecute() const
|
|
{
|
|
short result = 0;
|
|
if (!isRestoring()) {
|
|
result = (Views.isTouched() ||
|
|
Scale.isTouched() ||
|
|
ProjectionType.isTouched() ||
|
|
Template.isTouched());
|
|
if (result) {
|
|
return result;
|
|
}
|
|
}
|
|
return App::DocumentObject::mustExecute();
|
|
}
|
|
|
|
PyObject *DrawPage::getPyObject(void)
|
|
{
|
|
if (PythonObject.is(Py::_None())){
|
|
// ref counter is set to 1
|
|
PythonObject = Py::Object(new DrawPagePy(this),true);
|
|
}
|
|
|
|
return Py::new_reference_to(PythonObject);
|
|
}
|
|
|
|
bool DrawPage::hasValidTemplate() const
|
|
{
|
|
App::DocumentObject *obj = nullptr;
|
|
obj = Template.getValue();
|
|
|
|
if(obj && obj->isDerivedFrom(TechDraw::DrawTemplate::getClassTypeId())) {
|
|
TechDraw::DrawTemplate *templ = static_cast<TechDraw::DrawTemplate *>(obj);
|
|
if (templ->getWidth() > 0. &&
|
|
templ->getHeight() > 0.) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
double DrawPage::getPageWidth() const
|
|
{
|
|
App::DocumentObject *obj = nullptr;
|
|
obj = Template.getValue();
|
|
|
|
if( obj && obj->isDerivedFrom(TechDraw::DrawTemplate::getClassTypeId()) ) {
|
|
TechDraw::DrawTemplate *templ = static_cast<TechDraw::DrawTemplate *>(obj);
|
|
return templ->getWidth();
|
|
}
|
|
|
|
throw Base::RuntimeError("Template not set for Page");
|
|
}
|
|
|
|
double DrawPage::getPageHeight() const
|
|
{
|
|
App::DocumentObject *obj = nullptr;
|
|
obj = Template.getValue();
|
|
|
|
if(obj) {
|
|
if(obj->isDerivedFrom(TechDraw::DrawTemplate::getClassTypeId())) {
|
|
TechDraw::DrawTemplate *templ = static_cast<TechDraw::DrawTemplate *>(obj);
|
|
return templ->getHeight();
|
|
}
|
|
}
|
|
|
|
throw Base::RuntimeError("Template not set for Page");
|
|
}
|
|
|
|
const char * DrawPage::getPageOrientation() const
|
|
{
|
|
App::DocumentObject *obj;
|
|
obj = Template.getValue();
|
|
|
|
if(obj) {
|
|
if(obj->isDerivedFrom(TechDraw::DrawTemplate::getClassTypeId())) {
|
|
TechDraw::DrawTemplate *templ = static_cast<TechDraw::DrawTemplate *>(obj);
|
|
|
|
return templ->Orientation.getValueAsString();
|
|
}
|
|
}
|
|
throw Base::RuntimeError("Template not set for Page");
|
|
}
|
|
|
|
int DrawPage::addView(App::DocumentObject *docObj)
|
|
{
|
|
if(!docObj->isDerivedFrom(TechDraw::DrawView::getClassTypeId()))
|
|
return -1;
|
|
DrawView* view = static_cast<DrawView*>(docObj);
|
|
|
|
//position all new views in center of Page (exceptDVDimension)
|
|
if (!docObj->isDerivedFrom(TechDraw::DrawViewDimension::getClassTypeId()) &&
|
|
!docObj->isDerivedFrom(TechDraw::DrawViewBalloon::getClassTypeId())) {
|
|
view->X.setValue(getPageWidth()/2.0);
|
|
view->Y.setValue(getPageHeight()/2.0);
|
|
}
|
|
|
|
//add view to list
|
|
const std::vector<App::DocumentObject *> currViews = Views.getValues();
|
|
std::vector<App::DocumentObject *> newViews(currViews);
|
|
newViews.push_back(docObj);
|
|
Views.setValues(newViews);
|
|
|
|
//check if View fits on Page
|
|
if ( !view->checkFit(this) ) {
|
|
Base::Console().Warning("%s is larger than page. Will be scaled.\n",view->getNameInDocument());
|
|
view->ScaleType.setValue("Automatic");
|
|
}
|
|
|
|
view->checkScale();
|
|
|
|
return Views.getSize();
|
|
}
|
|
|
|
//Note Views might be removed from document elsewhere so need to check if a View is still in Document here
|
|
int DrawPage::removeView(App::DocumentObject *docObj)
|
|
{
|
|
if(!docObj->isDerivedFrom(TechDraw::DrawView::getClassTypeId()))
|
|
return -1;
|
|
|
|
App::Document* doc = docObj->getDocument();
|
|
if (doc == nullptr) {
|
|
return -1;
|
|
}
|
|
|
|
const char* name = docObj->getNameInDocument();
|
|
if (!name) {
|
|
return -1;
|
|
}
|
|
const std::vector<App::DocumentObject*> currViews = Views.getValues();
|
|
std::vector<App::DocumentObject*> newViews;
|
|
std::vector<App::DocumentObject*>::const_iterator it = currViews.begin();
|
|
for (; it != currViews.end(); it++) {
|
|
App::Document* viewDoc = (*it)->getDocument();
|
|
if (viewDoc == nullptr) {
|
|
continue;
|
|
}
|
|
|
|
std::string viewName = name;
|
|
if (viewName.compare((*it)->getNameInDocument()) != 0) {
|
|
newViews.push_back((*it));
|
|
}
|
|
}
|
|
Views.setValues(newViews);
|
|
return Views.getSize();
|
|
}
|
|
|
|
void DrawPage::requestPaint(void)
|
|
{
|
|
signalGuiPaint(this);
|
|
}
|
|
|
|
//this doesn't work right because there is no guaranteed of the restoration order
|
|
void DrawPage::onDocumentRestored()
|
|
{
|
|
if (canUpdate()) {
|
|
updateAllViews();
|
|
}
|
|
|
|
App::DocumentObject::onDocumentRestored();
|
|
}
|
|
|
|
void DrawPage::redrawCommand()
|
|
{
|
|
// Base::Console().Message("DP::redrawCommand()\n");
|
|
forceRedraw(true);
|
|
updateAllViews();
|
|
forceRedraw(false);
|
|
}
|
|
//should really be called "updateMostViews". can still be problems to due execution order.
|
|
void DrawPage::updateAllViews()
|
|
{
|
|
// Base::Console().Message("DP::updateAllViews()\n");
|
|
std::vector<App::DocumentObject*> featViews = getAllViews();
|
|
std::vector<App::DocumentObject*>::iterator it = featViews.begin();
|
|
//first, make sure all the Parts have been executed so GeometryObjects exist
|
|
for(; it != featViews.end(); ++it) {
|
|
TechDraw::DrawViewPart *part = dynamic_cast<TechDraw::DrawViewPart *>(*it);
|
|
TechDraw::DrawViewCollection *collect = dynamic_cast<TechDraw::DrawViewCollection*>(*it);
|
|
if (part) {
|
|
part->recomputeFeature();
|
|
} else if (collect) {
|
|
collect->recomputeFeature();
|
|
}
|
|
}
|
|
//second, make sure all the Dimensions have been executed so Measurements have References
|
|
for(it = featViews.begin(); it != featViews.end(); ++it) {
|
|
TechDraw::DrawViewDimension *dim = dynamic_cast<TechDraw::DrawViewDimension *>(*it);
|
|
if (dim) {
|
|
dim->recomputeFeature();
|
|
}
|
|
}
|
|
|
|
//third, try to execute all leader lines. may not work if parent DVP isn't ready.
|
|
for(it = featViews.begin(); it != featViews.end(); ++it) {
|
|
TechDraw::DrawLeaderLine *line = dynamic_cast<TechDraw::DrawLeaderLine *>(*it);
|
|
if (line) {
|
|
line->recomputeFeature();
|
|
}
|
|
}
|
|
|
|
//fourth, try to execute all spreadsheets.
|
|
for (it = featViews.begin(); it != featViews.end(); ++it) {
|
|
TechDraw::DrawViewSpreadsheet *sheet = dynamic_cast<TechDraw::DrawViewSpreadsheet *>(*it);
|
|
if (sheet) {
|
|
sheet->recomputeFeature();
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
std::vector<App::DocumentObject*> DrawPage::getAllViews(void)
|
|
{
|
|
auto views = Views.getValues(); //list of docObjects
|
|
std::vector<App::DocumentObject*> allViews;
|
|
for (auto& v: views) {
|
|
allViews.push_back(v);
|
|
if (v->isDerivedFrom(TechDraw::DrawProjGroup::getClassTypeId())) {
|
|
TechDraw::DrawProjGroup* dpg = static_cast<TechDraw::DrawProjGroup*>(v);
|
|
if (dpg) { //can't really happen!
|
|
std::vector<App::DocumentObject*> pgViews = dpg->Views.getValues();
|
|
allViews.insert(allViews.end(),pgViews.begin(),pgViews.end());
|
|
}
|
|
}
|
|
}
|
|
return allViews;
|
|
}
|
|
|
|
void DrawPage::unsetupObject()
|
|
{
|
|
nowUnsetting = true;
|
|
|
|
// Remove the Page's views & template from document
|
|
App::Document* doc = getDocument();
|
|
std::string docName = doc->getName();
|
|
std::string pageName = getNameInDocument();
|
|
|
|
try {
|
|
const std::vector<App::DocumentObject*> currViews = Views.getValues();
|
|
for (auto& v: currViews) {
|
|
//NOTE: the order of objects in Page.Views does not reflect the object hierarchy
|
|
// this means that a ProjGroup could be deleted before it's child ProjGroupItems.
|
|
// this causes problems when removing objects from document
|
|
if (v->isAttachedToDocument()) {
|
|
std::string viewName = v->getNameInDocument();
|
|
Base::Interpreter().runStringArg("App.getDocument(\"%s\").removeObject(\"%s\")",
|
|
docName.c_str(), viewName.c_str());
|
|
} else {
|
|
Base::Console().Log("DP::unsetupObject - v(%s) is not in document. skipping\n", pageName.c_str());
|
|
}
|
|
}
|
|
std::vector<App::DocumentObject*> emptyViews; //probably superfluous
|
|
Views.setValues(emptyViews);
|
|
|
|
}
|
|
catch (...) {
|
|
Base::Console().Warning("DP::unsetupObject - %s - error while deleting children\n", getNameInDocument());
|
|
}
|
|
|
|
App::DocumentObject* tmp = Template.getValue();
|
|
if (tmp) {
|
|
std::string templateName = Template.getValue()->getNameInDocument();
|
|
Base::Interpreter().runStringArg("App.getDocument(\"%s\").removeObject(\"%s\")",
|
|
docName.c_str(), templateName.c_str());
|
|
}
|
|
Template.setValue(nullptr);
|
|
}
|
|
|
|
int DrawPage::getNextBalloonIndex(void)
|
|
{
|
|
int result = NextBalloonIndex.getValue();
|
|
int newValue = result + 1;
|
|
NextBalloonIndex.setValue(newValue);
|
|
return result;
|
|
}
|
|
|
|
void DrawPage::handleChangedPropertyType(
|
|
Base::XMLReader &reader, const char * TypeName, App::Property * prop)
|
|
{
|
|
if (prop == &Scale) {
|
|
App::PropertyFloat tmp;
|
|
if (strcmp(tmp.getTypeId().getName(),TypeName)==0) { //property in file is Float
|
|
tmp.setContainer(this);
|
|
tmp.Restore(reader);
|
|
double tmpValue = tmp.getValue();
|
|
if (tmpValue > 0.0) {
|
|
Scale.setValue(tmpValue);
|
|
} else {
|
|
Scale.setValue(1.0);
|
|
}
|
|
} else {
|
|
// has Scale prop that isn't Float!
|
|
Base::Console().Log("DrawPage::Restore - old Document Scale is Not Float!\n");
|
|
// no idea
|
|
}
|
|
}
|
|
}
|
|
|
|
bool DrawPage::canUpdate() const
|
|
{
|
|
bool result = false;
|
|
if (GlobalUpdateDrawings() &&
|
|
KeepUpdated.getValue()) {
|
|
result = true;
|
|
} else if (!GlobalUpdateDrawings() &&
|
|
AllowPageOverride() &&
|
|
KeepUpdated.getValue()) {
|
|
result = true;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
//allow/prevent drawing updates for all Pages
|
|
bool DrawPage::GlobalUpdateDrawings(void)
|
|
{
|
|
Base::Reference<ParameterGrp> hGrp = App::GetApplication().GetUserParameter()
|
|
.GetGroup("BaseApp")->GetGroup("Preferences")->GetGroup("Mod/TechDraw/General");
|
|
bool result = hGrp->GetBool("GlobalUpdateDrawings", true);
|
|
return result;
|
|
}
|
|
|
|
//allow/prevent a single page to update despite GlobalUpdateDrawings setting
|
|
bool DrawPage::AllowPageOverride(void)
|
|
{
|
|
Base::Reference<ParameterGrp> hGrp = App::GetApplication().GetUserParameter()
|
|
.GetGroup("BaseApp")->GetGroup("Preferences")->GetGroup("Mod/TechDraw/General");
|
|
bool result = hGrp->GetBool("AllowPageOverride", true);
|
|
return result;
|
|
}
|
|
|
|
|
|
// Python Drawing feature ---------------------------------------------------------
|
|
|
|
namespace App {
|
|
/// @cond DOXERR
|
|
PROPERTY_SOURCE_TEMPLATE(TechDraw::DrawPagePython, TechDraw::DrawPage)
|
|
template<> const char* TechDraw::DrawPagePython::getViewProviderName(void) const {
|
|
return "TechDrawGui::ViewProviderPage";
|
|
}
|
|
/// @endcond
|
|
|
|
// explicit template instantiation
|
|
template class TechDrawExport FeaturePythonT<TechDraw::DrawPage>;
|
|
}
|