Arrow
Arrow24 июня 2018 г. 9:10

QComboBox и База данных

Qt, DataBase, QComboBox

Добрый день.

Необходима помощь в работе с QComboBox в связке с базой данных.

Описание:

Есть две таблицы:

1) "extension" с полями: id, library, function, ext.
2) "ext_type" с полями: id, type.

Таблицы связаны по полям ext и type. В первую таблицу в поле ext записывается id из второй таблицы.

В главном окне расположена таблица в которой отображаются данные из таблицы extension, в поле ext  подставляется вместо id соответствующий type из таблицы "ext_type".

Код:


mainModel = new QSqlRelationalTableModel(this);
mainModel->setTable("extension");

mainModel->setSort(0, Qt::AscendingOrder);
ui->mainTableView->setModel(mainModel);

// Выбор extension с таблицы ext_type по id
mainModel->setRelation(3, QSqlRelation("ext_type", "id", "type"));
ui->mainTableView->setItemDelegate(new QSqlRelationalDelegate(ui->mainTableView));
mainModel->select();

ui->mainTableView->setColumnHidden(0, true);

for (int i = 0; i < ui->mainTableView->horizontalHeader()->count(); i++) {
 ui->mainTableView->horizontalHeader()->setSectionResizeMode(i, QHeaderView::Stretch);
}
Все подставляется хорошо.

Второе окно открывается при редактировании записи в таблице или создании новой. На нем расположены QLineEdit для полей library, function и QComboBox для поля extension.

Нужно чтобы при создании новой записи из QComboBox пользователь мог выбрать значения из поля type таблицы ext_type.


Сделал так (под выбор работает а при редактировании существующей записи в QComboBox появляется не та строка):

ОШИБКА ЗДЕСЬ:

model = new QSqlTableModel(this);

mapper = new QDataWidgetMapper();
mapper->setModel(model);

mapper->addMapping(ui->Line1, 1);
mapper->addMapping(ui->Line2, 2);

QSqlQueryModel *select = new QSqlQueryModel(this);
select->setQuery("select type from ext_type");
ui->comboBox->setModel(select);

mapper->addMapping(ui->comboBox, model->fieldIndex("extension"), "currentIndex");

mapper->setSubmitPolicy(QDataWidgetMapper::ManualSubmit);

mapper->toLast();
Сохранение так (сохраняет, но из comboBox в таблицу попадает не та запись (см. выше)):

mapper->submit();
model->submitAll();

model->select();
mapper->toLast();
Помогите решить проблему.

И почему-то
mainModel->setRelation(3, QSqlRelation("ext_type", "id", "type"));
с таблицами выше работает.

А в этом случае нет:

mainModel = new QSqlRelationalTableModel(this);
mainModel->setTable("security_question");

mainModel->setSort(0, Qt::AscendingOrder);
ui->mainTableView->setModel(mainModel);

// Здесь
mainModel->setRelation(1, QSqlRelation("question", "id", "type"));
ui->mainTableView->setItemDelegate(new QSqlRelationalDelegate(ui->mainTableView));
mainModel->select();

ui->mainTableView->setColumnHidden(0, true);

for (int i = 0; i < ui->mainTableView->horizontalHeader()->count(); i++) {
ui->mainTableView->horizontalHeader()->setSectionResizeMode(i, QHeaderView::Stretch);
}
В таблицу на форме  вообще не выводятся данные (в сама таблица в базе данных заполнена).

Структура таблиц:
1) Таблица security_question: id, SecurityQuestion, Answer.
2) Таблица question: id, type.

Что может быль не так?

И подскажите как возможно организовать в базе данных запись в поле таблицы типа result1 или result2 в зависимости от того, какой QRadioButton пользователь выбрал на форме. А также, чтобы при редактировании данных на форме включался нужный QRadioButton в зависимости от записи?

В этом случае пока ничего не придумал.

Заранее спасибо за ответ и извините за долгое и возможно немного не понятное описание вопроса.
























Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

19
Arrow
  • 24 июня 2018 г. 9:27
  • Ответ был помечен как решение.

Насчет этого:

И почему-то
mainModel->setRelation(3, QSqlRelation("ext_type", "id", "type"));
с таблицами выше работает.

Извините вопрос снят, ошибся. В базе данных на поле SecurityQuestion вместо integer поставил тип text.
    Arrow
    • 25 июня 2018 г. 3:47
    На форумах нашел такой подход:


    mapper->addMapping(ui->comboBox, model->fieldIndex("extension")); 

    Но это вообще не работает.
      Evgenii Legotckoi
      • 25 июня 2018 г. 4:38

      Вообще должно работать, но там немного нюансы есть, к сожалению уже и не помню какие.

        Arrow
        • 25 июня 2018 г. 4:50
        • (ред.)
        Оно то работает только:


        В базе данных в столбце (например):

        id                            type

        1                             data
        2                             key
        3                             stat

        В таблице главного окна:


        Столбцы ......... .............. ....... ext

        ......... .............. ...... data
        ......... .............. ....... stat
        ......... .............. ....... key


        А в ComboBox при редактировании записи 2 вмето "stat" выходит "data".
          Arrow
          • 25 июня 2018 г. 4:56
          То есть запись по индексу выпадает на единицу меньше чем должна. Подозреваю ошибка здесь:

          mapper->addMapping(ui->comboBox, model->fieldIndex("extension"), "currentIndex");
          
          id записи в базе не совпадает с ее номером в comboBox. Но при попытке указать вместо currentIndex что-то другое, то вообще перестает работать.



            Arrow
            • 25 июня 2018 г. 7:03
            Нашел решение:

            model = new QSqlRelationalTableModel(this);
            model->setTable(TABLE);
            model->setEditStrategy(QSqlTableModel::OnManualSubmit);
            model->setSort(0, Qt::AscendingOrder);

            int typeIndex = model->fieldIndex("ext_type");
            model->setRelation(typeIndex, QSqlRelation("extension_type", "id", "type"));
            model->select();

            QSqlTableModel *relModel = model->relationModel(typeIndex);
            ui->extTypeBox->setModel(relModel);
            ui->extTypeBox->setModelColumn(relModel->fieldIndex("type"));

            mapper = new QDataWidgetMapper(this);
            mapper->setModel(model);
            mapper->setItemDelegate(new QSqlRelationalDelegate(this));

            mapper->addMapping(ui->extTypeBox, typeIndex);

            mapper->setSubmitPolicy(QDataWidgetMapper::ManualSubmit);
            model->insertRow(model->rowCount(QModelIndex()));
            mapper->toLast();
              Arrow
              • 25 июня 2018 г. 7:04

              Подскажите как организовать в базе данных запись в поле таблицы типа result1 или result2 в зависимости от того, какой QRadioButton пользователь выбрал на форме. А также, чтобы при редактировании данных на форме включался нужный QRadioButton в зависимости от записи?

                Arrow
                • 25 июня 2018 г. 8:12
                • (ред.)
                И если можно еще один вопрос.

                Таблицы во вложении.


                Если писать:

                mainModel = new QSqlRelationalTableModel(this);
                mainModel->setTable("menu");

                mainModel->setSort(0, Qt::AscendingOrder);
                mainModel->select();
                ui->mainTableView->setModel(mainModel);

                mainModel->setRelation(1, QSqlRelation("question", "id", "type"));
                Все работает и данные на форме в таблице отображаются корректно.

                А если вот так, то ничего не отображается вообще.

                mainModel = new QSqlRelationalTableModel(this);
                mainModel->setTable("menu");

                mainModel->setSort(0, Qt::AscendingOrder);
                mainModel->select();
                ui->mainTableView->setModel(mainModel);

                mainModel->setRelation(1, QSqlRelation("groups", "id", "Group name"));
                Если убрать последнюю строку, то все отображается в таблице (без подстановки).

                Поля id - not null, unique, auto inc, primary key.

                Что может быть не так, где я туплю?

                  Ruslan Polupan
                  • 3 июля 2018 г. 5:14

                  Имя поля таблицы  "Group name" меня смущает пробел. в нем. Просто никогда не использовал  и не видел пробелы в именах полей БД.

                    Evgenii Legotckoi
                    • 3 июля 2018 г. 5:17

                    Добрый день!
                    Спасибо за ответ, но там было найдено решение, там одинаковые именя столбцов были в одном месте.

                    К сожалению 30-го июня был сбой и часть данных в базе данных сайта была утеряна.. То сообщение топикстартера мне не удалось восстановить...

                      Ruslan Polupan
                      • 3 июля 2018 г. 7:27



                      Подскажите как организовать в базе данных запись в поле таблицы типа result1 или result2 в зависимости от того, какой QRadioButton пользователь выбрал на форме. А также, чтобы при редактировании данных на форме включался нужный QRadioButton в зависимости от записи?
                      Тут скорее все нужно уже больше полагаться на SQL. Т.е в зависимости от состояния QRadioButton делать запросы используя QSqlQuery.
                      Аналогично в зависимости от результатов запроса выставлять значения виджета.






                        Arrow
                        • 3 июля 2018 г. 7:34
                        Считывание реализовал так:

                        QSqlQuery query;
                            query.prepare( "SELECT type FROM " + Table + " WHERE id = ?" );
                            query.addBindValue(id);
                            query.exec();
                        
                            while (query.next()) {
                                if (query.value(0).toString() == "LOGO_HEADER") {
                                    ui->headLogoRButton->setChecked(true);
                                }
                                else {
                                    ui->mainLogoRButton->setChecked(true);
                                }

                        Записывать:

                        mapper->submit();
                        model->submitAll();
                        QString value; if (ui->headLogoRButton->isChecked()) value = "LOGO_HEADER"; else value = "LOGO_MAIN"; query.prepare( "UPDATE " + Table + " SET type = ? WHERE id = ?" ); query.addBindValue(value); query.addBindValue(query.lastInsertId().toInt()); query.exec();











                          Arrow
                          • 3 июля 2018 г. 7:34

                          Большое спасибо всем за помощь!

                            Ruslan Polupan
                            • 3 июля 2018 г. 7:45

                            Из опыта:  уменьшить время отладки на больших запросах я формирую строку с запросом которую можно скормить клиенту БД и проверяю запрос там. а потом уже использую в коде.

                            strSql=QString("SELECT a.TERMINAL_ID, t.name, a.SALEORDER_ID, a.NUM_CHECK, a.dat, a.sec, p.name, f.SHORTNAME, "
                                               "a.GIVE, a.ORDERED, a.SUMMA, a.DISCOUNTSUMMA,a.GIVE1 "
                                               "FROM SALEORDERS a "
                                               "LEFT JOIN fuels f ON a.FUEL_ID=f.FUEL_ID "
                                               "LEFT JOIN TERMINALS t ON a.TERMINAL_ID=t.TERMINAL_ID "
                                               "LEFT JOIN PAYTYPES p ON p.PAYTYPE_ID=a.PAYTYPE_ID "
                                               "where a.TERMINAL_ID=%1 and a.SHIFT_ID=%2 and a.NUM_CHECK=%3")
                                        .arg(ui->lineEditTerminalID->text().toInt())
                                        .arg(ui->lineEditShiftID->text().toInt())
                                        .arg(ui->lineEditNumCheck->text().toInt());
                                modelSale->setQuery(strSql,dbc);

                            Для больших запросов очень полезно.
                              Arrow
                              • 3 июля 2018 г. 7:54

                              Спасибо. У меня в таблице всего 4 столбца, но на будущее возьму на вооружение :)

                                Arrow
                                • 3 июля 2018 г. 7:58
                                • (ред.)

                                Такой вопрос. Никто случайно не писал свой класс для QComboBox для работы сполями бызы данных, а то через мапер выходит много однотипного кода. Хочется сократить.


                                Сейчас делаю так:


                                int typeIndex = model->fieldIndex("Security question");
                                model->setRelation(typeIndex, QSqlRelation("question", "id", "type"));
                                model->select();
                                QSqlTableModel *relModel = model->relationModel(typeIndex);
                                ui->sQuestionBox->setModel(relModel);
                                ui->sQuestionBox->setModelColumn(relModel->fieldIndex("type")); mapper->addMapping(ui->sQuestionBox, typeIndex);

                                  Evgenii Legotckoi
                                  • 3 июля 2018 г. 8:01

                                  Если код не имеет зависимости на какие-то внешние настройки, то можно просто наследоваться и настроить всё однотипное в конструкторе, если есть какие-то внешние параметры, то добавить публичный метод для настройки.

                                    Ruslan Polupan
                                    • 3 июля 2018 г. 8:03

                                    Не очень понял задачу?

                                      Arrow
                                      • 3 июля 2018 г. 8:07

                                      Понял, спасибо. Единственные зависимости - имена таблиц и полей.

                                        Комментарии

                                        Только авторизованные пользователи могут публиковать комментарии.
                                        Пожалуйста, авторизуйтесь или зарегистрируйтесь
                                        B

                                        C++ - Тест 002. Константы

                                        • Результат:16баллов,
                                        • Очки рейтинга-10
                                        B

                                        C++ - Тест 001. Первая программа и типы данных

                                        • Результат:46баллов,
                                        • Очки рейтинга-6
                                        FL

                                        C++ - Тест 006. Перечисления

                                        • Результат:80баллов,
                                        • Очки рейтинга4
                                        Последние комментарии
                                        k
                                        kmssr9 февраля 2024 г. 2:43
                                        Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                                        АК
                                        Анатолий Кононенко5 февраля 2024 г. 9:50
                                        Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                                        EVA
                                        EVA25 декабря 2023 г. 18:30
                                        Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
                                        J
                                        JonnyJo25 декабря 2023 г. 16:38
                                        Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
                                        G
                                        Gvozdik19 декабря 2023 г. 5:01
                                        Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
                                        Сейчас обсуждают на форуме
                                        P
                                        Pisych27 февраля 2023 г. 12:04
                                        Как получить в массив значения из связанной модели? Спасибо, разобрался:))
                                        AC
                                        Alexandru Codreanu19 января 2024 г. 19:57
                                        QML Обнулить значения SpinBox Доброго времени суток, не могу разобраться с обнулением значение SpinBox находящего в делегате. import QtQuickimport QtQuick.ControlsWindow { width: 640 height: 480 visible: tr…
                                        BlinCT
                                        BlinCT27 декабря 2023 г. 16:57
                                        Растягивать Image на парент по высоте Ну и само собою дял включения scrollbar надо чтобы был Flickable. Так что выходит как то так Flickable{ id: root anchors.fill: parent clip: true property url linkFile p…
                                        Дмитрий
                                        Дмитрий10 января 2024 г. 12:18
                                        Qt Creator загружает всю оперативную память Проблема решена. Удалось разобраться с помощью утилиты strace. Запустил ее: strace ./qtcreator Начал выводиться весь лог работы креатора. В один момент он начал считывать фай…
                                        Evgenii Legotckoi
                                        Evgenii Legotckoi12 декабря 2023 г. 14:48
                                        Побуквенное сравнение двух строк Добрый день. Там случайно не высылается этот сигнал textChanged ещё и при форматировани текста? Если решиать в лоб, то можно просто отключать сигнал/слотовое соединение внутри слота и …

                                        Следите за нами в социальных сетях