123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615 |
- /****************************************************************************
- **
- ** Copyright (C) 2016 The Qt Company Ltd.
- ** Contact: https://www.qt.io/licensing/
- **
- ** This file is part of the tools applications of the Qt Toolkit.
- **
- ** $QT_BEGIN_LICENSE:LGPL$
- ** Commercial License Usage
- ** Licensees holding valid commercial Qt licenses may use this file in
- ** accordance with the commercial license agreement provided with the
- ** Software or, alternatively, in accordance with the terms contained in
- ** a written agreement between you and The Qt Company. For licensing terms
- ** and conditions see https://www.qt.io/terms-conditions. For further
- ** information use the contact form at https://www.qt.io/contact-us.
- **
- ** GNU Lesser General Public License Usage
- ** Alternatively, this file may be used under the terms of the GNU Lesser
- ** General Public License version 3 as published by the Free Software
- ** Foundation and appearing in the file LICENSE.LGPL3 included in the
- ** packaging of this file. Please review the following information to
- ** ensure the GNU Lesser General Public License version 3 requirements
- ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
- **
- ** GNU General Public License Usage
- ** Alternatively, this file may be used under the terms of the GNU
- ** General Public License version 2.0 or (at your option) the GNU General
- ** Public license version 3 or any later version approved by the KDE Free
- ** Qt Foundation. The licenses are as published by the Free Software
- ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
- ** included in the packaging of this file. Please review the following
- ** information to ensure the GNU General Public License requirements will
- ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
- ** https://www.gnu.org/licenses/gpl-3.0.html.
- **
- ** $QT_END_LICENSE$
- **
- ****************************************************************************/
- #include "qtbuttonpropertybrowser.h"
- #include <QtCore/QSet>
- #include <QtWidgets/QGridLayout>
- #include <QtWidgets/QLabel>
- #include <QtCore/QTimer>
- #include <QtCore/QMap>
- #include <QtWidgets/QToolButton>
- #include <QtWidgets/QStyle>
- QT_BEGIN_NAMESPACE
- class QtButtonPropertyBrowserPrivate
- {
- QtButtonPropertyBrowser *q_ptr;
- Q_DECLARE_PUBLIC(QtButtonPropertyBrowser)
- public:
- void init(QWidget *parent);
- void propertyInserted(QtBrowserItem *index, QtBrowserItem *afterIndex);
- void propertyRemoved(QtBrowserItem *index);
- void propertyChanged(QtBrowserItem *index);
- QWidget *createEditor(QtProperty *property, QWidget *parent) const
- { return q_ptr->createEditor(property, parent); }
- void slotEditorDestroyed();
- void slotUpdate();
- void slotToggled(bool checked);
- struct WidgetItem
- {
- QWidget *widget{nullptr}; // can be null
- QLabel *label{nullptr}; // main label with property name
- QLabel *widgetLabel{nullptr}; // label substitute showing the current value if there is no widget
- QToolButton *button{nullptr}; // expandable button for items with children
- QWidget *container{nullptr}; // container which is expanded when the button is clicked
- QGridLayout *layout{nullptr}; // layout in container
- WidgetItem *parent{nullptr};
- QList<WidgetItem *> children;
- bool expanded{false};
- };
- private:
- void updateLater();
- void updateItem(WidgetItem *item);
- void insertRow(QGridLayout *layout, int row) const;
- void removeRow(QGridLayout *layout, int row) const;
- int gridRow(WidgetItem *item) const;
- int gridSpan(WidgetItem *item) const;
- void setExpanded(WidgetItem *item, bool expanded);
- QToolButton *createButton(QWidget *panret = 0) const;
- QMap<QtBrowserItem *, WidgetItem *> m_indexToItem;
- QMap<WidgetItem *, QtBrowserItem *> m_itemToIndex;
- QMap<QWidget *, WidgetItem *> m_widgetToItem;
- QMap<QObject *, WidgetItem *> m_buttonToItem;
- QGridLayout *m_mainLayout;
- QList<WidgetItem *> m_children;
- QList<WidgetItem *> m_recreateQueue;
- };
- QToolButton *QtButtonPropertyBrowserPrivate::createButton(QWidget *parent) const
- {
- QToolButton *button = new QToolButton(parent);
- button->setCheckable(true);
- button->setSizePolicy(QSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed));
- button->setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
- button->setArrowType(Qt::DownArrow);
- button->setIconSize(QSize(3, 16));
- /*
- QIcon icon;
- icon.addPixmap(q_ptr->style()->standardPixmap(QStyle::SP_ArrowDown), QIcon::Normal, QIcon::Off);
- icon.addPixmap(q_ptr->style()->standardPixmap(QStyle::SP_ArrowUp), QIcon::Normal, QIcon::On);
- button->setIcon(icon);
- */
- return button;
- }
- int QtButtonPropertyBrowserPrivate::gridRow(WidgetItem *item) const
- {
- QList<WidgetItem *> siblings;
- if (item->parent)
- siblings = item->parent->children;
- else
- siblings = m_children;
- int row = 0;
- for (WidgetItem *sibling : qAsConst(siblings)) {
- if (sibling == item)
- return row;
- row += gridSpan(sibling);
- }
- return -1;
- }
- int QtButtonPropertyBrowserPrivate::gridSpan(WidgetItem *item) const
- {
- if (item->container && item->expanded)
- return 2;
- return 1;
- }
- void QtButtonPropertyBrowserPrivate::init(QWidget *parent)
- {
- m_mainLayout = new QGridLayout();
- parent->setLayout(m_mainLayout);
- QLayoutItem *item = new QSpacerItem(0, 0,
- QSizePolicy::Fixed, QSizePolicy::Expanding);
- m_mainLayout->addItem(item, 0, 0);
- }
- void QtButtonPropertyBrowserPrivate::slotEditorDestroyed()
- {
- QWidget *editor = qobject_cast<QWidget *>(q_ptr->sender());
- if (!editor)
- return;
- if (!m_widgetToItem.contains(editor))
- return;
- m_widgetToItem[editor]->widget = 0;
- m_widgetToItem.remove(editor);
- }
- void QtButtonPropertyBrowserPrivate::slotUpdate()
- {
- for (WidgetItem *item : qAsConst(m_recreateQueue)) {
- WidgetItem *parent = item->parent;
- QWidget *w = 0;
- QGridLayout *l = 0;
- const int oldRow = gridRow(item);
- if (parent) {
- w = parent->container;
- l = parent->layout;
- } else {
- w = q_ptr;
- l = m_mainLayout;
- }
- int span = 1;
- if (!item->widget && !item->widgetLabel)
- span = 2;
- item->label = new QLabel(w);
- item->label->setSizePolicy(QSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed));
- l->addWidget(item->label, oldRow, 0, 1, span);
- updateItem(item);
- }
- m_recreateQueue.clear();
- }
- void QtButtonPropertyBrowserPrivate::setExpanded(WidgetItem *item, bool expanded)
- {
- if (item->expanded == expanded)
- return;
- if (!item->container)
- return;
- item->expanded = expanded;
- const int row = gridRow(item);
- WidgetItem *parent = item->parent;
- QGridLayout *l = 0;
- if (parent)
- l = parent->layout;
- else
- l = m_mainLayout;
- if (expanded) {
- insertRow(l, row + 1);
- l->addWidget(item->container, row + 1, 0, 1, 2);
- item->container->show();
- } else {
- l->removeWidget(item->container);
- item->container->hide();
- removeRow(l, row + 1);
- }
- item->button->setChecked(expanded);
- item->button->setArrowType(expanded ? Qt::UpArrow : Qt::DownArrow);
- }
- void QtButtonPropertyBrowserPrivate::slotToggled(bool checked)
- {
- WidgetItem *item = m_buttonToItem.value(q_ptr->sender());
- if (!item)
- return;
- setExpanded(item, checked);
- if (checked)
- emit q_ptr->expanded(m_itemToIndex.value(item));
- else
- emit q_ptr->collapsed(m_itemToIndex.value(item));
- }
- void QtButtonPropertyBrowserPrivate::updateLater()
- {
- QTimer::singleShot(0, q_ptr, SLOT(slotUpdate()));
- }
- void QtButtonPropertyBrowserPrivate::propertyInserted(QtBrowserItem *index, QtBrowserItem *afterIndex)
- {
- WidgetItem *afterItem = m_indexToItem.value(afterIndex);
- WidgetItem *parentItem = m_indexToItem.value(index->parent());
- WidgetItem *newItem = new WidgetItem();
- newItem->parent = parentItem;
- QGridLayout *layout = 0;
- QWidget *parentWidget = 0;
- int row = -1;
- if (!afterItem) {
- row = 0;
- if (parentItem)
- parentItem->children.insert(0, newItem);
- else
- m_children.insert(0, newItem);
- } else {
- row = gridRow(afterItem) + gridSpan(afterItem);
- if (parentItem)
- parentItem->children.insert(parentItem->children.indexOf(afterItem) + 1, newItem);
- else
- m_children.insert(m_children.indexOf(afterItem) + 1, newItem);
- }
- if (!parentItem) {
- layout = m_mainLayout;
- parentWidget = q_ptr;
- } else {
- if (!parentItem->container) {
- m_recreateQueue.removeAll(parentItem);
- WidgetItem *grandParent = parentItem->parent;
- QGridLayout *l = 0;
- const int oldRow = gridRow(parentItem);
- if (grandParent) {
- l = grandParent->layout;
- } else {
- l = m_mainLayout;
- }
- QFrame *container = new QFrame();
- container->setFrameShape(QFrame::Panel);
- container->setFrameShadow(QFrame::Raised);
- parentItem->container = container;
- parentItem->button = createButton();
- m_buttonToItem[parentItem->button] = parentItem;
- q_ptr->connect(parentItem->button, SIGNAL(toggled(bool)), q_ptr, SLOT(slotToggled(bool)));
- parentItem->layout = new QGridLayout();
- container->setLayout(parentItem->layout);
- if (parentItem->label) {
- l->removeWidget(parentItem->label);
- delete parentItem->label;
- parentItem->label = 0;
- }
- int span = 1;
- if (!parentItem->widget && !parentItem->widgetLabel)
- span = 2;
- l->addWidget(parentItem->button, oldRow, 0, 1, span);
- updateItem(parentItem);
- }
- layout = parentItem->layout;
- parentWidget = parentItem->container;
- }
- newItem->label = new QLabel(parentWidget);
- newItem->label->setSizePolicy(QSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed));
- newItem->widget = createEditor(index->property(), parentWidget);
- if (newItem->widget) {
- QObject::connect(newItem->widget, SIGNAL(destroyed()), q_ptr, SLOT(slotEditorDestroyed()));
- m_widgetToItem[newItem->widget] = newItem;
- } else if (index->property()->hasValue()) {
- newItem->widgetLabel = new QLabel(parentWidget);
- newItem->widgetLabel->setSizePolicy(QSizePolicy(QSizePolicy::Ignored, QSizePolicy::Fixed));
- }
- insertRow(layout, row);
- int span = 1;
- if (newItem->widget)
- layout->addWidget(newItem->widget, row, 1);
- else if (newItem->widgetLabel)
- layout->addWidget(newItem->widgetLabel, row, 1);
- else
- span = 2;
- layout->addWidget(newItem->label, row, 0, span, 1);
- m_itemToIndex[newItem] = index;
- m_indexToItem[index] = newItem;
- updateItem(newItem);
- }
- void QtButtonPropertyBrowserPrivate::propertyRemoved(QtBrowserItem *index)
- {
- WidgetItem *item = m_indexToItem.value(index);
- m_indexToItem.remove(index);
- m_itemToIndex.remove(item);
- WidgetItem *parentItem = item->parent;
- const int row = gridRow(item);
- if (parentItem)
- parentItem->children.removeAt(parentItem->children.indexOf(item));
- else
- m_children.removeAt(m_children.indexOf(item));
- const int colSpan = gridSpan(item);
- m_buttonToItem.remove(item->button);
- if (item->widget)
- delete item->widget;
- if (item->label)
- delete item->label;
- if (item->widgetLabel)
- delete item->widgetLabel;
- if (item->button)
- delete item->button;
- if (item->container)
- delete item->container;
- if (!parentItem) {
- removeRow(m_mainLayout, row);
- if (colSpan > 1)
- removeRow(m_mainLayout, row);
- } else if (parentItem->children.count() != 0) {
- removeRow(parentItem->layout, row);
- if (colSpan > 1)
- removeRow(parentItem->layout, row);
- } else {
- const WidgetItem *grandParent = parentItem->parent;
- QGridLayout *l = 0;
- if (grandParent) {
- l = grandParent->layout;
- } else {
- l = m_mainLayout;
- }
- const int parentRow = gridRow(parentItem);
- const int parentSpan = gridSpan(parentItem);
- l->removeWidget(parentItem->button);
- l->removeWidget(parentItem->container);
- delete parentItem->button;
- delete parentItem->container;
- parentItem->button = 0;
- parentItem->container = 0;
- parentItem->layout = 0;
- if (!m_recreateQueue.contains(parentItem))
- m_recreateQueue.append(parentItem);
- if (parentSpan > 1)
- removeRow(l, parentRow + 1);
- updateLater();
- }
- m_recreateQueue.removeAll(item);
- delete item;
- }
- void QtButtonPropertyBrowserPrivate::insertRow(QGridLayout *layout, int row) const
- {
- QMap<QLayoutItem *, QRect> itemToPos;
- int idx = 0;
- while (idx < layout->count()) {
- int r, c, rs, cs;
- layout->getItemPosition(idx, &r, &c, &rs, &cs);
- if (r >= row) {
- itemToPos[layout->takeAt(idx)] = QRect(r + 1, c, rs, cs);
- } else {
- idx++;
- }
- }
- for (auto it = itemToPos.constBegin(), icend = itemToPos.constEnd(); it != icend; ++it) {
- const QRect r = it.value();
- layout->addItem(it.key(), r.x(), r.y(), r.width(), r.height());
- }
- }
- void QtButtonPropertyBrowserPrivate::removeRow(QGridLayout *layout, int row) const
- {
- QMap<QLayoutItem *, QRect> itemToPos;
- int idx = 0;
- while (idx < layout->count()) {
- int r, c, rs, cs;
- layout->getItemPosition(idx, &r, &c, &rs, &cs);
- if (r > row) {
- itemToPos[layout->takeAt(idx)] = QRect(r - 1, c, rs, cs);
- } else {
- idx++;
- }
- }
- for (auto it = itemToPos.constBegin(), icend = itemToPos.constEnd(); it != icend; ++it) {
- const QRect r = it.value();
- layout->addItem(it.key(), r.x(), r.y(), r.width(), r.height());
- }
- }
- void QtButtonPropertyBrowserPrivate::propertyChanged(QtBrowserItem *index)
- {
- WidgetItem *item = m_indexToItem.value(index);
- updateItem(item);
- }
- void QtButtonPropertyBrowserPrivate::updateItem(WidgetItem *item)
- {
- QtProperty *property = m_itemToIndex[item]->property();
- if (item->button) {
- QFont font = item->button->font();
- font.setUnderline(property->isModified());
- item->button->setFont(font);
- item->button->setText(property->propertyName());
- item->button->setToolTip(property->descriptionToolTip());
- item->button->setStatusTip(property->statusTip());
- item->button->setWhatsThis(property->whatsThis());
- item->button->setEnabled(property->isEnabled());
- }
- if (item->label) {
- QFont font = item->label->font();
- font.setUnderline(property->isModified());
- item->label->setFont(font);
- item->label->setText(property->propertyName());
- item->label->setToolTip(property->descriptionToolTip());
- item->label->setStatusTip(property->statusTip());
- item->label->setWhatsThis(property->whatsThis());
- item->label->setEnabled(property->isEnabled());
- }
- if (item->widgetLabel) {
- QFont font = item->widgetLabel->font();
- font.setUnderline(false);
- item->widgetLabel->setFont(font);
- item->widgetLabel->setText(property->valueText());
- item->widgetLabel->setToolTip(property->valueText());
- item->widgetLabel->setEnabled(property->isEnabled());
- }
- if (item->widget) {
- QFont font = item->widget->font();
- font.setUnderline(false);
- item->widget->setFont(font);
- item->widget->setEnabled(property->isEnabled());
- const QString valueToolTip = property->valueToolTip();
- item->widget->setToolTip(valueToolTip.isEmpty() ? property->valueText() : valueToolTip);
- }
- }
- /*!
- \class QtButtonPropertyBrowser
- \internal
- \inmodule QtDesigner
- \since 4.4
- \brief The QtButtonPropertyBrowser class provides a drop down QToolButton
- based property browser.
- A property browser is a widget that enables the user to edit a
- given set of properties. Each property is represented by a label
- specifying the property's name, and an editing widget (e.g. a line
- edit or a combobox) holding its value. A property can have zero or
- more subproperties.
- QtButtonPropertyBrowser provides drop down button for all nested
- properties, i.e. subproperties are enclosed by a container associated with
- the drop down button. The parent property's name is displayed as button text. For example:
- \image qtbuttonpropertybrowser.png
- Use the QtAbstractPropertyBrowser API to add, insert and remove
- properties from an instance of the QtButtonPropertyBrowser
- class. The properties themselves are created and managed by
- implementations of the QtAbstractPropertyManager class.
- \sa QtTreePropertyBrowser, QtAbstractPropertyBrowser
- */
- /*!
- \fn void QtButtonPropertyBrowser::collapsed(QtBrowserItem *item)
- This signal is emitted when the \a item is collapsed.
- \sa expanded(), setExpanded()
- */
- /*!
- \fn void QtButtonPropertyBrowser::expanded(QtBrowserItem *item)
- This signal is emitted when the \a item is expanded.
- \sa collapsed(), setExpanded()
- */
- /*!
- Creates a property browser with the given \a parent.
- */
- QtButtonPropertyBrowser::QtButtonPropertyBrowser(QWidget *parent)
- : QtAbstractPropertyBrowser(parent), d_ptr(new QtButtonPropertyBrowserPrivate)
- {
- d_ptr->q_ptr = this;
- d_ptr->init(this);
- }
- /*!
- Destroys this property browser.
- Note that the properties that were inserted into this browser are
- \e not destroyed since they may still be used in other
- browsers. The properties are owned by the manager that created
- them.
- \sa QtProperty, QtAbstractPropertyManager
- */
- QtButtonPropertyBrowser::~QtButtonPropertyBrowser()
- {
- const QMap<QtButtonPropertyBrowserPrivate::WidgetItem *, QtBrowserItem *>::ConstIterator icend = d_ptr->m_itemToIndex.constEnd();
- for (QMap<QtButtonPropertyBrowserPrivate::WidgetItem *, QtBrowserItem *>::ConstIterator it = d_ptr->m_itemToIndex.constBegin(); it != icend; ++it)
- delete it.key();
- }
- /*!
- \reimp
- */
- void QtButtonPropertyBrowser::itemInserted(QtBrowserItem *item, QtBrowserItem *afterItem)
- {
- d_ptr->propertyInserted(item, afterItem);
- }
- /*!
- \reimp
- */
- void QtButtonPropertyBrowser::itemRemoved(QtBrowserItem *item)
- {
- d_ptr->propertyRemoved(item);
- }
- /*!
- \reimp
- */
- void QtButtonPropertyBrowser::itemChanged(QtBrowserItem *item)
- {
- d_ptr->propertyChanged(item);
- }
- /*!
- Sets the \a item to either collapse or expanded, depending on the value of \a expanded.
- \sa isExpanded(), expanded(), collapsed()
- */
- void QtButtonPropertyBrowser::setExpanded(QtBrowserItem *item, bool expanded)
- {
- QtButtonPropertyBrowserPrivate::WidgetItem *itm = d_ptr->m_indexToItem.value(item);
- if (itm)
- d_ptr->setExpanded(itm, expanded);
- }
- /*!
- Returns true if the \a item is expanded; otherwise returns false.
- \sa setExpanded()
- */
- bool QtButtonPropertyBrowser::isExpanded(QtBrowserItem *item) const
- {
- QtButtonPropertyBrowserPrivate::WidgetItem *itm = d_ptr->m_indexToItem.value(item);
- if (itm)
- return itm->expanded;
- return false;
- }
- QT_END_NAMESPACE
- #include "moc_qtbuttonpropertybrowser.cpp"
|