Arrow
ArrowJune 24, 2018, 10:10 a.m.

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 в зависимости от записи?

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

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
























We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.

Do you like it? Share on social networks!

19
Arrow
  • June 24, 2018, 10:27 a.m.
  • The answer was marked as a solution.

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

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

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


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

    Но это вообще не работает.
      Evgenii Legotckoi
      • June 25, 2018, 5:38 a.m.

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

        Arrow
        • June 25, 2018, 5:50 a.m.
        • (edited)
        Оно то работает только:


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

        id                            type

        1                             data
        2                             key
        3                             stat

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


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

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


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

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



            Arrow
            • June 25, 2018, 8:03 a.m.
            Нашел решение:

            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
              • June 25, 2018, 8:04 a.m.

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

                Arrow
                • June 25, 2018, 9:12 a.m.
                • (edited)
                И если можно еще один вопрос.

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


                Если писать:

                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
                  • July 3, 2018, 6:14 a.m.

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

                    Evgenii Legotckoi
                    • July 3, 2018, 6:17 a.m.

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

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

                      Ruslan Polupan
                      • July 3, 2018, 8:27 a.m.



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






                        Arrow
                        • July 3, 2018, 8:34 a.m.
                        Считывание реализовал так:

                        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
                          • July 3, 2018, 8:34 a.m.

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

                            Ruslan Polupan
                            • July 3, 2018, 8:45 a.m.

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

                            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
                              • July 3, 2018, 8:54 a.m.

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

                                Arrow
                                • July 3, 2018, 8:58 a.m.
                                • (edited)

                                Такой вопрос. Никто случайно не писал свой класс для 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
                                  • July 3, 2018, 9:01 a.m.

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

                                    Ruslan Polupan
                                    • July 3, 2018, 9:03 a.m.

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

                                      Arrow
                                      • July 3, 2018, 9:07 a.m.

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

                                        Comments

                                        Only authorized users can post comments.
                                        Please, Log in or Sign up
                                        г
                                        • ги
                                        • April 23, 2024, 9:51 p.m.

                                        C++ - Test 005. Structures and Classes

                                        • Result:41points,
                                        • Rating points-8
                                        l
                                        • laei
                                        • April 23, 2024, 3:19 p.m.

                                        C ++ - Test 004. Pointers, Arrays and Loops

                                        • Result:10points,
                                        • Rating points-10
                                        l
                                        • laei
                                        • April 23, 2024, 3:17 p.m.

                                        C++ - Тест 003. Условия и циклы

                                        • Result:50points,
                                        • Rating points-4
                                        Last comments
                                        k
                                        kmssrFeb. 9, 2024, 12:43 a.m.
                                        Qt Linux - Lesson 001. Autorun Qt application under Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                                        Qt WinAPI - Lesson 007. Working with ICMP Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                                        EVA
                                        EVADec. 25, 2023, 4:30 p.m.
                                        Boost - static linking in CMake project under Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
                                        J
                                        JonnyJoDec. 25, 2023, 2:38 p.m.
                                        Boost - static linking in CMake project under Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
                                        G
                                        GvozdikDec. 19, 2023, 3:01 a.m.
                                        Qt/C++ - Lesson 056. Connecting the Boost library in Qt for MinGW and MSVC compilers Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
                                        Now discuss on the forum
                                        G
                                        GarApril 22, 2024, 11:46 a.m.
                                        Clipboard Как скопировать окно целиком в clipb?
                                        DA
                                        Dr Gangil AcademicsApril 20, 2024, 1:45 p.m.
                                        Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
                                        a
                                        a_vlasovApril 14, 2024, 12:41 p.m.
                                        Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
                                        Павел Дорофеев
                                        Павел ДорофеевApril 14, 2024, 8:35 a.m.
                                        QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
                                        f
                                        fastrexApril 4, 2024, 10:47 a.m.
                                        Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

                                        Follow us in social networks