| |
|
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| | |
| |
|
| |
|
| | #include <QApplication>
|
| | #include <QCheckBox>
|
| | #include <QComboBox>
|
| | #include <QModelIndex>
|
| | #include <QPainter>
|
| | #include <QTimer>
|
| |
|
| | #include <Base/Tools.h>
|
| |
|
| | #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<PropertyItem*>(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<QColor>()) {
|
| | color = value.value<QColor>();
|
| | }
|
| | }
|
| | option.palette.setColor(QPalette::Text, color);
|
| | option.font.setBold(true);
|
| |
|
| |
|
| |
|
| |
|
| |
|
| | }
|
| | 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<QBrush>()) {
|
| | brush = value.value<QBrush>();
|
| | }
|
| | }
|
| | painter->fillRect(option.rect, brush);
|
| | }
|
| |
|
| | QPen savedPen = painter->pen();
|
| |
|
| | if (index.column() == 1 && property && dynamic_cast<PropertyBoolItem*>(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;
|
| |
|
| |
|
| | style->drawPrimitive(QStyle::PE_PanelItemViewItem, &option, painter, option.widget);
|
| |
|
| |
|
| | 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);
|
| |
|
| |
|
| | 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<QRgb>(QApplication::style()->styleHint(
|
| | QStyle::SH_Table_GridLineColor,
|
| | &opt,
|
| | qobject_cast<QWidget*>(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<PropertyItem*>(index.internalPointer());
|
| |
|
| | if ((property && !property->isSeparator())
|
| | && (!event || event->type() == QEvent::MouseButtonDblClick)) {
|
| |
|
| |
|
| |
|
| | 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<QComboBox*>(o);
|
| | if (comboBox) {
|
| | auto parentEditor = qobject_cast<PropertyEditor*>(this->parent());
|
| | if (parentEditor && parentEditor->activeEditor == comboBox) {
|
| | comboBox->showPopup();
|
| | }
|
| | }
|
| | auto* checkBox = qobject_cast<QCheckBox*>(o);
|
| | if (checkBox) {
|
| | auto parentEditor = qobject_cast<PropertyEditor*>(this->parent());
|
| | if (parentEditor && parentEditor->activeEditor == checkBox) {
|
| | checkBox->toggle();
|
| |
|
| | QTimer::singleShot(0, this, [this]() { valueChanged(); });
|
| | }
|
| | }
|
| | }
|
| | else if (ev->type() == QEvent::FocusOut) {
|
| | if (auto button = qobject_cast<Gui::ColorButton*>(o)) {
|
| |
|
| | if (button->property("modal_dialog_active").toBool()) {
|
| | return true;
|
| | }
|
| | }
|
| | auto parentEditor = qobject_cast<PropertyEditor*>(this->parent());
|
| | if (auto* comboBox = qobject_cast<QComboBox*>(o)) {
|
| | if (parentEditor && parentEditor->activeEditor == comboBox) {
|
| | parentEditor->activeEditor = nullptr;
|
| | }
|
| | }
|
| | auto widget = qobject_cast<QWidget*>(o);
|
| | if (widget && parentEditor && parentEditor->activeEditor
|
| | && widget != parentEditor->activeEditor) {
|
| |
|
| |
|
| |
|
| |
|
| |
|
| | return false;
|
| | }
|
| | }
|
| | return QItemDelegate::eventFilter(o, ev);
|
| | }
|
| |
|
| | QWidget* PropertyItemDelegate::createEditor(
|
| | QWidget* parent,
|
| | const QStyleOptionViewItem& ,
|
| | const QModelIndex& index
|
| | ) const
|
| | {
|
| | if (!index.isValid()) {
|
| | return nullptr;
|
| | }
|
| |
|
| | auto childItem = static_cast<PropertyItem*>(index.internalPointer());
|
| | if (!childItem || childItem->isSeparator() || childItem->isReadOnly()) {
|
| | return nullptr;
|
| | }
|
| |
|
| | auto parentEditor = qobject_cast<PropertyEditor*>(this->parent());
|
| | if (parentEditor) {
|
| | parentEditor->closeEditor();
|
| | }
|
| |
|
| | auto createEditor = [this, childItem, parent]() {
|
| |
|
| | if (qobject_cast<PropertyBoolItem*>(childItem)) {
|
| |
|
| |
|
| |
|
| | return childItem->createEditor(parent, []() noexcept {});
|
| | }
|
| | return childItem->createEditor(parent, [this]() {
|
| | const_cast<PropertyItemDelegate*>(this)->valueChanged();
|
| | });
|
| | };
|
| |
|
| | 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<PropertyItemDelegate*>(this)->valueChanged();
|
| | });
|
| | 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) {
|
| |
|
| | editor->setAutoFillBackground(true);
|
| | }
|
| | if (editor && childItem->isReadOnly()) {
|
| | editor->setDisabled(true);
|
| | }
|
| | else if (editor ) {
|
| |
|
| |
|
| |
|
| |
|
| | editor->setFocus();
|
| | }
|
| | this->pressed = false;
|
| |
|
| | if (editor) {
|
| | const auto widgets = editor->findChildren<QWidget*>();
|
| | for (auto w : widgets) {
|
| | if (qobject_cast<QAbstractButton*>(w) || qobject_cast<QLabel*>(w)) {
|
| | w->installEventFilter(const_cast<PropertyItemDelegate*>(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<QComboBox*>(propertyEditor) || qobject_cast<QCheckBox*>(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<PropertyItem*>(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<PropertyItem*>(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"
|
| |
|