ПБ
Павел Богдевич11. Januar 2019 07:09
Qt, Qml, QAbstractTableModel
Всем привет, часто нахожу ответы на вашем форуме. Огромное спасибо. Пролистал кучу других форумов, просмотрел документацию. Ничего не получается.
Не могу заставить чекбокс в хэдере таблицы работать с чекбоксами в колонке таблицы. Мне нужно либо выделить все, либо снять все чекбоксы. Пробовал двумя способами. Не работает. Помогите, пожалуйста.
#ifndef FOODTABLEMODEL_H #define FOODTABLEMODEL_H #include <QAbstractTableModel> #include <QModelIndex> #include <QHash> #include <QVariant> #include <QByteArray> #include <QList> struct Food { bool check; QString description; int price; QString code; }; class FoodTableModel : public QAbstractTableModel { Q_OBJECT public: enum RoleNames { CHECK = 0, DESCRIPTION = 1, PRICE = 2, CODE = 3 }; FoodTableModel(QObject *parent = nullptr); int rowCount(const QModelIndex &parent) const; int columnCount(const QModelIndex &parent) const; QVariant data(const QModelIndex &index, int role) const; QHash<int, QByteArray> roleNames() const; bool setData(const QModelIndex &index, const QVariant &value, int role); Q_INVOKABLE void checkTable(bool value); private: QList<Food> foods; }; #endif // FOODTABLEMODEL_H
#include "foodtablemodel.h" #include <QDebug> FoodTableModel::FoodTableModel(QObject *parent) : QAbstractTableModel { parent } { foods.append({ false, "Pizza", 12, "ASR1" }); foods.append({ false, "Milk", 1, "BE-23" }); foods.append({ false, "Coffe", 4, "ARTY8-9" }); /*foods.append({ "Lizza", "12", "ASR1" }); foods.append({ "SMilk", "34", "BE-23" }); foods.append({ "YCoffe", "5", "QRTY8-9" });*/ } int FoodTableModel::rowCount(const QModelIndex &parent) const { return foods.size(); } int FoodTableModel::columnCount(const QModelIndex &parent) const { return roleNames().size(); } QVariant FoodTableModel::data(const QModelIndex &index, int role) const { QVariant variant; const int row = index.row(); const int col = role; switch (col) { case CHECK: variant = foods.at(row).check; break; case DESCRIPTION: variant = foods.at(row).description; break; case PRICE: variant = foods.at(row).price; break; case CODE: variant = foods.at(row).code; break; } return variant; } QHash<int, QByteArray> FoodTableModel::roleNames() const { QHash<int, QByteArray> roles; roles.insert(CHECK, "check"); roles.insert(DESCRIPTION, "description"); roles.insert(PRICE, "price"); roles.insert(CODE, "code"); return roles; } bool FoodTableModel::setData(const QModelIndex &index, const QVariant &value, int role) { if(index.isValid() && role == CHECK) { foods[index.row()].check = value.toBool(); emit dataChanged(index, index, { role }); return true; } else return false; } void FoodTableModel::checkTable(bool value) { for (int i = 0; i < foods.size(); i++) { qDebug() << i << value; foods[i].check = value; } }
#include <QGuiApplication> #include <QQmlApplicationEngine> #include <QQmlContext> #include "foodtablemodel.h" #include "person.h" #include "sortfilterproxymodel.h" int main(int argc, char *argv[]) { QGuiApplication app(argc, argv); QQmlApplicationEngine engine; FoodTableModel food; qmlRegisterType<SortFilterProxyModel>("SortFilterProxyModel", 0, 1, "SortFilterProxyModel"); engine.rootContext()->setContextProperty("food", &food); engine.load(QUrl(QStringLiteral("qrc:/main.qml"))); if (engine.rootObjects().isEmpty()) return -1; return app.exec(); }
import QtQuick 2.12 import QtQuick.Controls 1.4 as C1 import QtQuick.Controls.Styles 1.4 import QtQuick.Controls 2.4 import SortFilterProxyModel 0.1 import "JavaScript/TableViewCheck.js" as TVC C1.TableView { id: tableView clip: true sortIndicatorVisible: true currentRow: rowCount ? 0 : -1 model: SortFilterProxyModel { source: food.rowCount() > 0 ? food : null sortOrder: tableView.sortIndicatorOrder sortCaseSensitivity: Qt.CaseInsensitive sortRole: food.rowCount() && tableView.getColumn(tableView.sortIndicatorColumn).role !== "check" > 0 ? tableView.getColumn(tableView.sortIndicatorColumn).role : "" } C1.TableViewColumn { width: 30 role: "check" resizable: false delegate: C1.CheckBox { id: checkBox anchors.left: parent.left anchors.leftMargin: parent.width / 3 checked: model.check onVisibleChanged: if (visible) checked = model.check onClicked: model.check = checked } } C1.TableViewColumn { role: "description" title: "Description" } C1.TableViewColumn { role: "price" title: "Price" } C1.TableViewColumn { role: "code" title: "Code" } style: TableViewStyle { headerDelegate: Rectangle { height: textItem.implicitHeight width: textItem.implicitWidth Text { id: textItem anchors.fill: parent anchors.leftMargin: 6 verticalAlignment: Text.AlignVCenter horizontalAlignment: styleData.textAlignment text: styleData.value } C1.CheckBox { anchors.left: parent.left anchors.leftMargin: parent.width / 3 property bool isPressed: styleData.pressed visible: styleData.column === 0 onIsPressedChanged: { if(isPressed) { checked = !checked for(var i = 0; i < rowCount; i++) { food.setData(foods.index(i, 0), checked, "check") } //food.checkTable(checked) } } } } } }
Рекомендуємо хостинг TIMEWEB
Stabiles Hosting des sozialen Netzwerks EVILEG. Wir empfehlen VDS-Hosting für Django-Projekte.Magst du es? In sozialen Netzwerken teilen!
Kommentare
sf
- sdfsdfkp fgskpgokspdog
- 14. Oktober 2024 15:09
C++ - Тест 004. Указатели, Массивы и Циклы
- Ergebnis:90punkte,
- Bewertungspunkte8
МВ
- Максим Васильев
- 2. Oktober 2024 04:14
Qt - Тест 001. Сигналы и слоты
- Ergebnis:68punkte,
- Bewertungspunkte-1
ЛС
- Лев Семенов
- 30. September 2024 11:04
C++ - Тест 001. Первая программа и типы данных
- Ergebnis:53punkte,
- Bewertungspunkte-4
Letzte Kommentare
ИМ
Django – Lektion 064. So schreiben Sie eine Python-Markdown-Erweiterung Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
Игорь Максимов5. Oktober 2024 07:51
QML - Lektion 016. SQLite-Datenbank und das Arbeiten damit in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
Qt Linux - Lektion 001. Autorun Qt-Anwendung unter Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Qt WinAPI - Lektion 007. Arbeiten mit ICMP-Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
Анатолий Кононенко5. Februar 2024 01:50
EVA25. Dezember 2023 10:30
Jetzt im Forum diskutieren
добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
ИМ
Реализация навигации по разделам Спасибо Евгений!
Игорь Максимов3. Oktober 2024 04:05
Indian Food Restaurant In Columbus OH| Layla’s Kitchen Indian Restaurant If you're looking for a truly authentic https://www.laylaskitchenrestaurantohio.com/ , Layla’s Kitchen Indian Restaurant is your go-to destination. Located at 6152 Cleveland Ave, Colu…
КГ
Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
Кирилл Гусарев27. September 2024 09:09
при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …
Добрый день. Интересная задача. Мне тоже нужно будет что-то подобное сделать. Так что с удовольствием поковыряюсь. Но большая просьба - оформите исходники, пожалуйста. В текущем виде ими невозможно воспользоваться.
Александр, я все исправил, но случайно ответил ваш ответ как решение. Теперь вы просто обязаны мне помочь.
:)))) это было очень ошибочное решение, настолько поверить в меня. Но я постараюсь.
Не увидел в исходниках описание этой модели - SortFilterProxyModel.
Из QML кода обращение идёт к ней.
Эта модель отвечает за сортировку колонок, поэтому я не скидывал файлы реализации.
Несколько упросил ваш пример и сделал его рабочим - проверяйте:
main.c
checkmodel.h
checkmodel.cpp
main.qml
Также прикладываю исходники для удобства.
QmlAllColumnCheckBox.zip
На идеальный код мой вариант не претендует. Если будут замечания у вас или других более опытных коллег - буду рад услышать.
Но в целом поставленную вами задачу он решает.
Будут вопросы - задавайте.
Маленькие заметки:
- QML != JS. Старайтесь избегать использования кода javascript.
- struct заставляет выдумывать массу не нужных костылей (enum, например, или ручное прописывание ролей) - замените его на класс, туда же можно кучу логики вынести.
- в своем примере вы добавили в качестве модели прокси-модель - это хорошо, но я не зря просил её реализацию, т.к. в дальнейшем вы пытались обращаться к функциям модели. В таком случае вам нужно было либо сделать подобные функции в прокси, либо обращаться к модели прокси-модели.
Александр, огромное спасибо. Я надеюсь, что у вас опыта побольше, чем у меня, и те нюансы, которые вы внесли, они обоснованы. Если у меня возникнут проблемы с объединением прокси модели и модели, которую вы реализовали, можно я буду у вас спрашивать здесь (или лучше задать новый вопрос)?
Обращайтесь, чем смогу - помогу.
Александр, как я и ожидал, проблема появилась, причем очень странная. Когда кликая по хедеру табдицы, чтобы отсортировать колонку таблицы, то выделяются все чекбоксы. Прикладываю исходники модели сортровки (Это код не мой, я его нашел на просторах интернета). Стоит объеденить модели?
Это измененное место инициализации модели в файле main.qml
Дописал вопрос, и понял. Выделение должно происходить, когда мы кликаем по чекбоксу в хедере, а оно происходит при нажатии в любом месте хедера.
Вот решение.
Qt не поддерживает чекбоксы в хедерах таблицы, чтобы это сделать
Попробуйте создать класс отнаследованный от QHeaderView и в нем переопределить метод data ( if(role==CheckStateRole и т.д.)
и добавить сигнал (checkedChanged)
HeaderClass{
id:header
}
// in tableView
delegate: Checkbox {
id: headerDelegate
checked: header.checked
onCheckedChanged: headerDelegate.checked = checked // пишем состояние куда надо
}
или засунуть в роли модели роль 'HeaderCheckedState'
и брать оттуда