QmlDesigner: Add duplicate control

Add a setting/property to the EditableListView in order to control if
it allows duplicates or not. If duplicates are not allowed it will
disable the items in the ComboBox which are already applied to the
assigned backend property.

Task-number: QDS-6730
Change-Id: I31776fdfb14858c431b40d264e8e709009c517b3
Reviewed-by: Thomas Hartmann <thomas.hartmann@qt.io>
Reviewed-by: Ali Kianian <ali.kianian@qt.io>
Reviewed-by: Mahmoud Badri <mahmoud.badri@qt.io>
Reviewed-by: <github-actions-qt-creator@cristianadam.eu>
This commit is contained in:
Henning Gruendl
2023-05-05 17:08:45 +02:00
committed by Henning Gründl
parent c4ff1921ee
commit 94b019dbe0
4 changed files with 56 additions and 50 deletions

View File

@@ -1,8 +1,9 @@
// Copyright (C) 2021 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
import QtQuick 2.15
import QtQuick.Layouts 1.15
import QtQuick
import QtQuick.Layouts
import HelperWidgets 2.0 as HelperWidgets
import StudioControls 1.0 as StudioControls
import StudioTheme 1.0 as StudioTheme
@@ -23,11 +24,12 @@ Item {
property real __actionIndicatorWidth: StudioTheme.Values.squareComponentWidth
property real __actionIndicatorHeight: StudioTheme.Values.height
property string typeFilter: "QtQuick3D.Material"
property string textRole: "IdAndNameRole"
property string valueRole: "IdRole"
property string textRole: "idAndName"
property string valueRole: "id"
property int activatedReason: ComboBox.ActivatedReason.Other
property bool delegateHover: false
property bool allowDuplicates: true
property string extraButtonIcon: "" // setting this will show an extra button
property string extraButtonToolTip: ""
@@ -40,11 +42,19 @@ Item {
Layout.preferredWidth: StudioTheme.Values.height * 10
Layout.preferredHeight: myColumn.height
HelperWidgets.ItemFilterModel {
id: itemFilterModel
typeFilter: root.typeFilter
modelNodeBackendProperty: modelNodeBackend
selectedItems: root.allowDuplicates ? [] : root.model
}
Component {
id: myDelegate
Row {
property alias comboBox: itemFilterComboBox
ListViewComboBox {
id: itemFilterComboBox
@@ -54,7 +64,7 @@ Item {
validator: RegExpValidator { regExp: /(^[a-z_]\w*|^[A-Z]\w*\.{1}([a-z_]\w*\.?)+)/ }
actionIndicatorVisible: false
typeFilter: root.typeFilter
model: itemFilterModel
initialModelData: modelData
textRole: root.textRole
valueRole: root.valueRole
@@ -188,7 +198,7 @@ Item {
visible: myRepeater.count === 0
validator: RegExpValidator { regExp: /(^[a-z_]\w*|^[A-Z]\w*\.{1}([a-z_]\w*\.?)+)/ }
actionIndicatorVisible: false
typeFilter: root.typeFilter
model: itemFilterModel
textRole: root.textRole
valueRole: root.valueRole
implicitWidth: StudioTheme.Values.singleControlColumnWidth

View File

@@ -1,27 +1,19 @@
// Copyright (C) 2021 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only
import QtQuick 2.15
import QtQuick
import HelperWidgets 2.0 as HelperWidgets
import StudioControls 1.0 as StudioControls
StudioControls.ComboBox {
id: root
property alias typeFilter: itemFilterModel.typeFilter
property var initialModelData
property bool __isCompleted: false
editable: true
model: itemFilterModel
textRole: "IdRole"
valueRole: "IdRole"
HelperWidgets.ItemFilterModel {
id: itemFilterModel
modelNodeBackendProperty: modelNodeBackend
}
textRole: "id"
valueRole: "id"
Component.onCompleted: {
root.__isCompleted = true

View File

@@ -15,20 +15,11 @@
using namespace QmlDesigner;
QHash<int, QByteArray> ItemFilterModel::m_roles;
ItemFilterModel::ItemFilterModel(QObject *parent)
: QAbstractListModel(parent)
, m_typeFilter("QtQuick.Item")
, m_selectionOnly(false)
{
if (m_roles.empty()) {
m_roles = QAbstractListModel::roleNames();
const QMetaEnum roleEnum = QMetaEnum::fromType<Roles>();
for (int i = 0; i < roleEnum.keyCount(); i++)
m_roles.insert(roleEnum.value(i), roleEnum.key(i));
}
}
{}
void ItemFilterModel::setModelNodeBackend(const QVariant &modelNodeBackend)
{
@@ -64,6 +55,12 @@ void ItemFilterModel::setSelectionOnly(bool value)
emit selectionOnlyChanged();
}
void ItemFilterModel::setSelectedItems(const QStringList &selectedItems)
{
m_selectedItems = selectedItems;
emit selectedItemsChanged();
}
QString ItemFilterModel::typeFilter() const
{
return m_typeFilter;
@@ -74,6 +71,11 @@ bool ItemFilterModel::selectionOnly() const
return m_selectionOnly;
}
QStringList ItemFilterModel::selectedItems() const
{
return m_selectedItems;
}
void ItemFilterModel::registerDeclarativeType()
{
qmlRegisterType<ItemFilterModel>("HelperWidgets", 2, 0, "ItemFilterModel");
@@ -86,34 +88,34 @@ int ItemFilterModel::rowCount(const QModelIndex &) const
QVariant ItemFilterModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid())
if (!index.isValid() || index.row() >= rowCount())
return {};
const ModelNode node = modelNodeForRow(index.row());
QVariant value;
switch (role) {
case IdRole:
value = node.id();
break;
return node.id();
case NameRole:
value = node.variantProperty("objectName").value();
break;
return node.variantProperty("objectName").value();
case IdAndNameRole:
value = QString("%1 [%2]").arg(node.variantProperty("objectName").value().toString(),
node.id());
break;
return QString("%1 [%2]").arg(node.variantProperty("objectName").value().toString(),
node.id());
case EnabledRole:
return !m_selectedItems.contains(node.id());
default:
value = node.id();
break;
return {};
}
return value;
}
QHash<int, QByteArray> ItemFilterModel::roleNames() const
{
return m_roles;
static QHash<int, QByteArray> roleNames{{IdRole, "id"},
{NameRole, "name"},
{IdAndNameRole, "idAndName"},
{EnabledRole, "enabled"}};
return roleNames;
}
QVariant ItemFilterModel::modelNodeBackend() const

View File

@@ -17,25 +17,26 @@ class ItemFilterModel : public QAbstractListModel
Q_OBJECT
Q_PROPERTY(QString typeFilter READ typeFilter WRITE setTypeFilter NOTIFY typeFilterChanged)
Q_PROPERTY(QVariant modelNodeBackendProperty READ modelNodeBackend WRITE setModelNodeBackend NOTIFY modelNodeBackendChanged)
Q_PROPERTY(QVariant modelNodeBackendProperty READ modelNodeBackend WRITE setModelNodeBackend
NOTIFY modelNodeBackendChanged)
Q_PROPERTY(QStringList itemModel READ itemModel NOTIFY itemModelChanged)
Q_PROPERTY(bool selectionOnly READ selectionOnly WRITE setSelectionOnly NOTIFY selectionOnlyChanged)
Q_PROPERTY(
bool selectionOnly READ selectionOnly WRITE setSelectionOnly NOTIFY selectionOnlyChanged)
Q_PROPERTY(QStringList selectedItems READ selectedItems WRITE setSelectedItems NOTIFY
selectedItemsChanged)
public:
enum Roles {
IdRole = Qt::UserRole + 1,
NameRole,
IdAndNameRole
};
Q_ENUM(Roles)
enum { IdRole = Qt::DisplayRole, NameRole = Qt::UserRole, IdAndNameRole, EnabledRole };
explicit ItemFilterModel(QObject *parent = nullptr);
void setModelNodeBackend(const QVariant &modelNodeBackend);
void setTypeFilter(const QString &typeFilter);
void setSelectionOnly(bool value);
void setSelectedItems(const QStringList &selectedItems);
QString typeFilter() const;
bool selectionOnly() const;
QStringList selectedItems() const;
void setupModel();
QStringList itemModel() const;
@@ -51,6 +52,7 @@ signals:
void modelNodeBackendChanged();
void itemModelChanged();
void selectionOnlyChanged();
void selectedItemsChanged();
private:
QVariant modelNodeBackend() const;
@@ -61,7 +63,7 @@ private:
QList<qint32> m_modelInternalIds;
QmlDesigner::ModelNode m_modelNode;
bool m_selectionOnly;
static QHash<int, QByteArray> m_roles;
QStringList m_selectedItems;
};
QML_DECLARE_TYPE(ItemFilterModel)