ПБ
Павел Богдевич11 січня 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
Стабільний хостинг, на якому розміщується соціальна мережа EVILEG. Для проектів на Django радимо VDS хостинг.Вам це подобається? Поділіться в соціальних мережах!
AD
- Akiv Doros
- 11 листопада 2024 р. 14:58
C++ - Тест 004. Указатели, Массивы и Циклы
- Результат:50бали,
- Рейтинг балів-4
m
- molni99
- 26 жовтня 2024 р. 01:37
C++ - Тест 004. Указатели, Массивы и Циклы
- Результат:80бали,
- Рейтинг балів4
m
- molni99
- 26 жовтня 2024 р. 01:29
C++ - Тест 004. Указатели, Массивы и Циклы
- Результат:20бали,
- Рейтинг балів-10
Останні коментарі
ИМ
Django - Підручник 017. Налаштуйте сторінку входу до Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
Игорь Максимов22 листопада 2024 р. 11:51
Evgenii Legotckoi31 жовтня 2024 р. 14:37
Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
ИМ
Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
Игорь Максимов05 жовтня 2024 р. 07:51
QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
Тепер обговоріть на форумі
Evgenii Legotckoi24 червня 2024 р. 15:11
t
google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
tonypeachey115 листопада 2024 р. 06:04
NSProject04 червня 2022 р. 03:49
IscanderChe31 жовтня 2024 р. 15:43
Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…
Добрый день. Интересная задача. Мне тоже нужно будет что-то подобное сделать. Так что с удовольствием поковыряюсь. Но большая просьба - оформите исходники, пожалуйста. В текущем виде ими невозможно воспользоваться.
Александр, я все исправил, но случайно ответил ваш ответ как решение. Теперь вы просто обязаны мне помочь.
:)))) это было очень ошибочное решение, настолько поверить в меня. Но я постараюсь.
Не увидел в исходниках описание этой модели - 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'
и брать оттуда