/*************************************************************************** * Copyright (c) 2014 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 # include #endif #include #include #include #include #include "BitmapFactory.h" #include "DlgPropertyLink.h" #include "Application.h" #include "ViewProviderDocumentObject.h" #include "ui_DlgPropertyLink.h" using namespace Gui::Dialog; /* TRANSLATOR Gui::Dialog::DlgPropertyLink */ DlgPropertyLink::DlgPropertyLink(const QStringList& list, QWidget* parent, Qt::WindowFlags fl, bool xlink) : QDialog(parent, fl), link(list), ui(new Ui_DlgPropertyLink) { #ifdef FC_DEBUG assert(list.size() >= 4); #endif // populate inList to filter out any objects that contains the owner object // of the editing link property auto doc = App::GetApplication().getDocument(qPrintable(link[0])); if(doc) { auto obj = doc->getObject(qPrintable(link[3])); if(obj && obj->getNameInDocument()) { inList = obj->getInListEx(true); inList.insert(obj); } } ui->setupUi(this); ui->typeTree->hide(); if(!xlink) ui->comboBox->hide(); else { std::string linkDoc = qPrintable(link[0]); for(auto doc : App::GetApplication().getDocuments()) { QString name(QString::fromUtf8(doc->Label.getValue())); ui->comboBox->addItem(name,QVariant(QString::fromLatin1(doc->getName()))); if(linkDoc == doc->getName()) ui->comboBox->setCurrentIndex(ui->comboBox->count()-1); } } Base::Type baseType; App::Document *linkedDoc = doc; if (link.size()>FC_XLINK_VALUE_INDEX) linkedDoc = App::GetApplication().getDocument(qPrintable(link[FC_XLINK_VALUE_INDEX])); if(linkedDoc) { QString objName = link[1]; // linked object name auto obj = linkedDoc->getObject((const char*)objName.toLatin1()); if (obj && inList.find(obj)==inList.end()) { Base::Type objType = obj->getTypeId(); // get only geometric types if (objType.isDerivedFrom(App::GeoFeature::getClassTypeId())) baseType = App::GeoFeature::getClassTypeId(); else baseType = App::DocumentObject::getClassTypeId(); // get the direct base class of App::DocumentObject which 'obj' is derived from while (!objType.isBad()) { std::string name = objType.getName(); Base::Type parType = objType.getParent(); if (parType == baseType) { baseType = objType; break; } objType = parType; } } } if(!baseType.isBad()) { types.insert(baseType.getName()); ui->checkObjectType->setChecked(true); }else findObjects(); connect(ui->treeWidget, SIGNAL(itemExpanded(QTreeWidgetItem*)), this, SLOT(onItemExpanded(QTreeWidgetItem*))); } /** * Destroys the object and frees any allocated resources */ DlgPropertyLink::~DlgPropertyLink() { // no need to delete child widgets, Qt does it all for us delete ui; } void DlgPropertyLink::setSelectionMode(QAbstractItemView::SelectionMode mode) { ui->treeWidget->setSelectionMode(mode); findObjects(); } void DlgPropertyLink::accept() { if (ui->treeWidget->selectionMode() == QAbstractItemView::SingleSelection) { QList items = ui->treeWidget->selectedItems(); if (items.isEmpty()) { QMessageBox::warning(this, tr("No selection"), tr("Please select an object from the list")); return; } } QDialog::accept(); } static QStringList getLinkFromItem(const QStringList &link, QTreeWidgetItem *selItem) { QStringList list = link; if(link.size()>FC_XLINK_VALUE_INDEX) { QString subname; auto parent = selItem; for(auto item=parent;;item=parent) { parent = item->parent(); if(!parent) { list[1] = item->data(0,Qt::UserRole).toString(); break; } subname = QString::fromLatin1("%1.%2"). arg(item->data(0,Qt::UserRole).toString()).arg(subname); } list[FC_XLINK_VALUE_INDEX] = subname; if(subname.size()) list[2] = QString::fromLatin1("%1 (%2.%3)"). arg(selItem->text(0)).arg(list[1]).arg(subname); else list[2] = selItem->text(0); QString docName(selItem->data(0, Qt::UserRole+1).toString()); if(list.size()>FC_XLINK_VALUE_INDEX+1) list[FC_XLINK_VALUE_INDEX+1] = docName; else list << docName; }else{ list[1] = selItem->data(0,Qt::UserRole).toString(); list[2] = selItem->text(0); if (list[1].isEmpty()) list[2] = QString::fromUtf8(""); } return list; } QStringList DlgPropertyLink::propertyLink() const { auto items = ui->treeWidget->selectedItems(); if (items.isEmpty()) { return link; } return getLinkFromItem(link,items[0]); } QVariantList DlgPropertyLink::propertyLinkList() const { QVariantList varList; QList items = ui->treeWidget->selectedItems(); for (QList::iterator it = items.begin(); it != items.end(); ++it) varList << getLinkFromItem(link,*it); return varList; } void DlgPropertyLink::findObjects() { bool filterType = ui->checkObjectType->isChecked(); ui->treeWidget->clear(); QString docName = link[0]; // document name of the owner object of this editing property bool isSingleSelection = (ui->treeWidget->selectionMode() == QAbstractItemView::SingleSelection); App::Document* doc = App::GetApplication().getDocument((const char*)docName.toLatin1()); if (doc) { // build list of objects names already in property so we can mark them as selected later on std::set selectedNames; // Add a "None" entry on top if (isSingleSelection) { auto* item = new QTreeWidgetItem(ui->treeWidget); item->setText(0,tr("None (Remove link)")); QByteArray ba(""); item->setData(0,Qt::UserRole, ba); }else { QString ownerName = link[3]; QString proName = link[4]; auto owner = doc->getObject(ownerName.toLatin1()); if(owner) { App::Property* prop = owner->getPropertyByName((const char*)proName.toLatin1()); // gather names of objects currently in property if (prop && prop->getTypeId().isDerivedFrom(App::PropertyLinkList::getClassTypeId())) { const App::PropertyLinkList* propll = static_cast(prop); std::vector links = propll->getValues(); for (std::vector::iterator it = links.begin(); it != links.end(); ++it) { selectedNames.insert((*it)->getNameInDocument()); } } } } Base::PyGILStateLocker lock; std::map typeInfos; QTreeWidgetItem *selected = 0; for(auto obj : doc->getObjects()) { auto it = types.end(); auto prop = Base::freecad_dynamic_cast( obj->getPropertyByName("Proxy")); bool pass = false; if(prop && !prop->getValue().isNone()) { std::string typeName = prop->getValue().type().as_string(); if(refreshTypes) { QIcon &icon = typeInfos[typeName]; if(icon.isNull()) { auto vp = Application::Instance->getViewProvider(obj); if(vp) icon = vp->getIcon(); } } if(filterType) pass = types.count(typeName); } if(refreshTypes) { auto type = obj->getTypeId(); QIcon &icon = typeInfos[type.getName()]; if(!prop && icon.isNull()) { auto vp = Application::Instance->getViewProvider(obj); if(vp) icon = vp->getIcon(); } for(type = type.getParent();!type.isBad();type=type.getParent()) typeInfos.emplace(type.getName(),QIcon()); } if(filterType && !pass && types.size()) { for(auto type = obj->getTypeId();!type.isBad();type=type.getParent()) { it = types.find(type.getName()); if(it!=types.end()) break; } if(it == types.end()) continue; } auto item = createItem(obj,0); if(item && selectedNames.count(obj->getNameInDocument())) { if(!selected) selected = item; item->setSelected(true); } } if(selected) ui->treeWidget->scrollToItem(selected); if(refreshTypes) { refreshTypes = false; ui->typeTree->blockSignals(true); ui->typeTree->clear(); QTreeWidgetItem *selected = 0; QIcon icon = BitmapFactory().pixmap("px"); for(auto &v : typeInfos) { auto item = new QTreeWidgetItem(ui->typeTree); item->setText(0,QString::fromLatin1(v.first.c_str())); item->setIcon(0,v.second.isNull()?icon:v.second); if(types.count(v.first)) { item->setSelected(true); if(!selected) selected = item; } } if(selected) ui->typeTree->scrollToItem(selected); ui->typeTree->blockSignals(false); if(types.size() && !selected) { types.clear(); findObjects(); } } } } QTreeWidgetItem *DlgPropertyLink::createItem(App::DocumentObject *obj, QTreeWidgetItem *parent) { if(!obj || !obj->getNameInDocument()) return 0; if(inList.find(obj)!=inList.end()) return 0; auto vp = Gui::Application::Instance->getViewProvider(obj); if(!vp) return 0; QString searchText = ui->searchBox->text(); if (!searchText.isEmpty()) { QString label = QString::fromUtf8((obj)->Label.getValue()); if (!label.contains(searchText,Qt::CaseInsensitive)) return 0; } QTreeWidgetItem* item; if(parent) item = new QTreeWidgetItem(parent); else item = new QTreeWidgetItem(ui->treeWidget); item->setIcon(0, vp->getIcon()); item->setText(0, QString::fromUtf8((obj)->Label.getValue())); item->setData(0, Qt::UserRole, QByteArray(obj->getNameInDocument())); item->setData(0, Qt::UserRole+1, QByteArray(obj->getDocument()->getName())); if(link.size()>=5) { item->setChildIndicatorPolicy(obj->hasChildElement()||vp->getChildRoot()? QTreeWidgetItem::ShowIndicator:QTreeWidgetItem::DontShowIndicator); } return item; } void DlgPropertyLink::onItemExpanded(QTreeWidgetItem * item) { if(link.size()<5 || item->childCount()) return; std::string name(qPrintable(item->data(0, Qt::UserRole).toString())); std::string docName(qPrintable(item->data(0, Qt::UserRole+1).toString())); auto doc = App::GetApplication().getDocument(docName.c_str()); if(doc) { auto obj = doc->getObject(name.c_str()); if(!obj) return; auto vp = Application::Instance->getViewProvider(obj); if(!vp) return; for(auto obj : vp->claimChildren()) createItem(obj,item); } } void DlgPropertyLink::on_checkObjectType_toggled(bool on) { ui->typeTree->setVisible(on); findObjects(); } void DlgPropertyLink::on_typeTree_itemSelectionChanged() { types.clear(); for(auto item : ui->typeTree->selectedItems()) types.insert(item->text(0).toLatin1().constData()); findObjects(); } void DlgPropertyLink::on_searchBox_textChanged(const QString& /*search*/) { findObjects(); } void DlgPropertyLink::on_comboBox_currentIndexChanged(int index) { link[0] = ui->comboBox->itemData(index).toString(); refreshTypes = true; findObjects(); } #include "moc_DlgPropertyLink.cpp"