// SPDX-License-Identifier: LGPL-2.1-or-later /*************************************************************************** * Copyright (c) 2004 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 #include #include #include #include #include #include #include "PropertyItemDelegate.h" #include "MDIView.h" #include "PropertyEditor.h" #include "PropertyItem.h" #include "Tree.h" FC_LOG_LEVEL_INIT("PropertyView", true, true) using namespace Gui::PropertyEditor; PropertyItemDelegate::PropertyItemDelegate(QObject* parent) : QItemDelegate(parent) , expressionEditor(nullptr) , pressed(false) , changed(false) {} PropertyItemDelegate::~PropertyItemDelegate() = default; QSize PropertyItemDelegate::sizeHint(const QStyleOptionViewItem& option, const QModelIndex& index) const { QSize size = QItemDelegate::sizeHint(option, index); size += QSize(0, 5); return size; } void PropertyItemDelegate::paint( QPainter* painter, const QStyleOptionViewItem& opt, const QModelIndex& index ) const { QStyleOptionViewItem option = opt; auto property = static_cast(index.internalPointer()); if (property && property->isSeparator()) { QColor color = option.palette.color(QPalette::BrightText); QObject* par = parent(); if (par) { QVariant value = par->property("groupTextColor"); if (value.canConvert()) { color = value.value(); } } option.palette.setColor(QPalette::Text, color); option.font.setBold(true); // Since the group item now parents all the property items and can be // collapsed, it makes sense to have some selection visual clue for it. // // option.state &= ~QStyle::State_Selected; } else if (index.column() == 1) { option.state &= ~QStyle::State_Selected; if (property && property->isReadOnly()) { option.state &= ~QStyle::State_Enabled; } } option.state &= ~QStyle::State_HasFocus; if (property && property->isSeparator()) { QBrush brush = option.palette.dark(); QObject* par = parent(); if (par) { QVariant value = par->property("groupBackground"); if (value.canConvert()) { brush = value.value(); } } painter->fillRect(option.rect, brush); } QPen savedPen = painter->pen(); if (index.column() == 1 && property && dynamic_cast(property)) { bool checked = index.data(Qt::EditRole).toBool(); bool readonly = property->isReadOnly(); QStyle* style = option.widget ? option.widget->style() : QApplication::style(); QPalette palette = option.widget ? option.widget->palette() : QApplication::palette(); QStyleOptionButton checkboxOption; checkboxOption.state |= readonly ? QStyle::State_ReadOnly : QStyle::State_Enabled; checkboxOption.state |= checked ? QStyle::State_On : QStyle::State_Off; // draw the item (background etc.) style->drawPrimitive(QStyle::PE_PanelItemViewItem, &option, painter, option.widget); // Draw the checkbox checkboxOption.rect = style->subElementRect(QStyle::SE_CheckBoxIndicator, &checkboxOption, option.widget); int leftSpacing = style->pixelMetric(QStyle::PM_FocusFrameHMargin, nullptr, option.widget); QRect checkboxRect = QStyle::alignedRect( option.direction, Qt::AlignVCenter, checkboxOption.rect.size(), option.rect.adjusted(leftSpacing, 0, -leftSpacing, 0) ); checkboxOption.rect = checkboxRect; style->drawPrimitive(QStyle::PE_IndicatorCheckBox, &checkboxOption, painter, option.widget); // Draw the label of the checkbox QString labelText = checked ? tr("Yes") : tr("No"); int spacing = style->pixelMetric(QStyle::PM_CheckBoxLabelSpacing, nullptr, option.widget); QRect textRect( checkboxOption.rect.right() + spacing, checkboxOption.rect.top(), option.rect.right() - (checkboxOption.rect.right() + spacing), checkboxOption.rect.height() ); painter->setPen(palette.color(QPalette::Text)); painter->drawText(textRect, Qt::AlignVCenter | Qt::AlignLeft, labelText); } else { QItemDelegate::paint(painter, option, index); } QColor color = static_cast(QApplication::style()->styleHint( QStyle::SH_Table_GridLineColor, &opt, qobject_cast(parent()) )); painter->setPen(QPen(color)); if (index.column() == 1 || !(property && property->isSeparator())) { int right = (option.direction == Qt::LeftToRight) ? option.rect.right() : option.rect.left(); painter->drawLine(right, option.rect.y(), right, option.rect.bottom()); } painter->drawLine(option.rect.x(), option.rect.bottom(), option.rect.right(), option.rect.bottom()); painter->setPen(savedPen); } bool PropertyItemDelegate::editorEvent( QEvent* event, QAbstractItemModel* model, const QStyleOptionViewItem& option, const QModelIndex& index ) { auto property = static_cast(index.internalPointer()); if ((property && !property->isSeparator()) && (!event || event->type() == QEvent::MouseButtonDblClick)) { // ignore double click, as it could cause editor lock with checkboxes // due to the editor being close immediately after toggling the checkbox // which is currently done on first click return true; } this->pressed = event->type() == QEvent::MouseButtonPress; return QItemDelegate::editorEvent(event, model, option, index); } bool PropertyItemDelegate::eventFilter(QObject* o, QEvent* ev) { if (ev->type() == QEvent::FocusIn) { auto* comboBox = qobject_cast(o); if (comboBox) { auto parentEditor = qobject_cast(this->parent()); if (parentEditor && parentEditor->activeEditor == comboBox) { comboBox->showPopup(); } } auto* checkBox = qobject_cast(o); if (checkBox) { auto parentEditor = qobject_cast(this->parent()); if (parentEditor && parentEditor->activeEditor == checkBox) { checkBox->toggle(); // Delay valueChanged to ensure proper recomputation QTimer::singleShot(0, this, [this]() { valueChanged(); }); } } } else if (ev->type() == QEvent::FocusOut) { if (auto button = qobject_cast(o)) { // Ignore the event if the ColorButton's modal dialog is active. if (button->property("modal_dialog_active").toBool()) { return true; } } auto parentEditor = qobject_cast(this->parent()); if (auto* comboBox = qobject_cast(o)) { if (parentEditor && parentEditor->activeEditor == comboBox) { parentEditor->activeEditor = nullptr; } } auto widget = qobject_cast(o); if (widget && parentEditor && parentEditor->activeEditor && widget != parentEditor->activeEditor) { // All the attempts to ignore the focus-out event has been approved to not work // reliably because there are still cases that cannot be handled. // So, the best for now is to always ignore this event. // See https://forum.freecad.org/viewtopic.php?p=579530#p579530 why this is not // possible. return false; } } return QItemDelegate::eventFilter(o, ev); } QWidget* PropertyItemDelegate::createEditor( QWidget* parent, const QStyleOptionViewItem& /*option*/, const QModelIndex& index ) const { if (!index.isValid()) { return nullptr; } auto childItem = static_cast(index.internalPointer()); if (!childItem || childItem->isSeparator() || childItem->isReadOnly()) { return nullptr; } auto parentEditor = qobject_cast(this->parent()); if (parentEditor) { parentEditor->closeEditor(); } auto createEditor = [this, childItem, parent]() { // Can't use a terniary here because the lambdas have different types. if (qobject_cast(childItem)) { // Boolean properties use a checkbox that is basically artificial // (it is not rendered). Therefore, the callback is handled in // eventFilter() return childItem->createEditor(parent, []() noexcept {}); } return childItem->createEditor(parent, [this]() { const_cast(this)->valueChanged(); // NOLINT }); }; FC_LOG("create editor " << index.row() << "," << index.column()); QWidget* editor = nullptr; expressionEditor = nullptr; userEditor = nullptr; if (parentEditor && parentEditor->isBinding()) { expressionEditor = editor = childItem->createExpressionEditor(parent, [this]() { const_cast(this)->valueChanged(); // NOLINT }); propertyEditor = editor; } else { const auto& props = childItem->getPropertyData(); if (!props.empty() && props[0]->testStatus(App::Property::UserEdit)) { editor = userEditor = childItem->createPropertyEditorWidget(parent); propertyEditor = editor; } else { propertyEditor = editor = createEditor(); } } if (editor) { // Make sure the editor background is painted so the cell content doesn't show through editor->setAutoFillBackground(true); } if (editor && childItem->isReadOnly()) { editor->setDisabled(true); } else if (editor /*&& this->pressed*/) { // We changed the way editor is activated in PropertyEditor (in response // of signal activated and clicked), so now we should grab focus // regardless of "pressed" or not (e.g. when activated by keyboard // enter) editor->setFocus(); } this->pressed = false; if (editor) { const auto widgets = editor->findChildren(); for (auto w : widgets) { if (qobject_cast(w) || qobject_cast(w)) { w->installEventFilter(const_cast(this)); } } parentEditor->activeEditor = editor; parentEditor->editingIndex = index; } return editor; } void PropertyItemDelegate::valueChanged() { if (propertyEditor) { Base::FlagToggler<> flag(changed); Q_EMIT commitData(propertyEditor); if (qobject_cast(propertyEditor) || qobject_cast(propertyEditor)) { Q_EMIT closeEditor(propertyEditor); return; } } } void PropertyItemDelegate::setEditorData(QWidget* editor, const QModelIndex& index) const { if (!index.isValid()) { return; } QVariant data = index.data(Qt::EditRole); auto childItem = static_cast(index.internalPointer()); editor->blockSignals(true); if (expressionEditor == editor) { childItem->setExpressionEditorData(editor, data); } else if (userEditor == editor) { userEditor->setValue(PropertyItemAttorney::toString(childItem, data)); } else { childItem->setEditorData(editor, data); } editor->blockSignals(false); return; } void PropertyItemDelegate::setModelData( QWidget* editor, QAbstractItemModel* model, const QModelIndex& index ) const { if (!index.isValid() || !changed || userEditor) { return; } auto childItem = static_cast(index.internalPointer()); QVariant data; if (expressionEditor == editor) { data = childItem->expressionEditorData(editor); } else { data = childItem->editorData(editor); } model->setData(index, data, Qt::EditRole); } #include "moc_PropertyItemDelegate.cpp"