Arrow
June 24, 2018, 8:10 p.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 в зависимости от записи?

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

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
























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

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

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

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


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

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

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

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


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

        id                            type

        1                             data
        2                             key
        3                             stat

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


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

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


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

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



            Arrow
            • June 25, 2018, 6:03 p.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, 6:04 p.m.

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

                Arrow
                • June 25, 2018, 7:12 p.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, 4:14 p.m.

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

                    Evgenii Legotckoi
                    • July 3, 2018, 4:17 p.m.

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

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

                      Ruslan Polupan
                      • July 3, 2018, 6:27 p.m.



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






                        Arrow
                        • July 3, 2018, 6:34 p.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, 6:34 p.m.

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

                            Ruslan Polupan
                            • July 3, 2018, 6:45 p.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, 6:54 p.m.

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

                                Arrow
                                • July 3, 2018, 6:58 p.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, 7:01 p.m.

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

                                    Ruslan Polupan
                                    • July 3, 2018, 7:03 p.m.

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

                                      Arrow
                                      • July 3, 2018, 7:07 p.m.

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

                                        Comments

                                        Only authorized users can post comments.
                                        Please, Log in or Sign up
                                        • Last comments
                                        • Evgenii Legotckoi
                                          April 16, 2025, 5:08 p.m.
                                          Благодарю за отзыв. И вам желаю всяческих успехов!
                                        • IscanderChe
                                          April 12, 2025, 5:12 p.m.
                                          Добрый день. Спасибо Вам за этот проект и отдельно за ответы на форуме, которые мне очень помогли в некоммерческих пет-проектах. Профессиональным программистом я так и не стал, но узнал мно…
                                        • AK
                                          April 1, 2025, 11:41 a.m.
                                          Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
                                        • Evgenii Legotckoi
                                          March 9, 2025, 9:02 p.m.
                                          К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
                                        • VP
                                          March 9, 2025, 4:14 p.m.
                                          Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…