Arrow
Arrow24. Juni 2018 10: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
Stabiles Hosting des sozialen Netzwerks EVILEG. Wir empfehlen VDS-Hosting für Django-Projekte.

Magst du es? In sozialen Netzwerken teilen!

19
Arrow
  • 24. Juni 2018 10:27
  • Die Antwort wurde als Lösung markiert.

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

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

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


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

    Но это вообще не работает.
      Evgenii Legotckoi
      • 25. Juni 2018 05:38

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

        Arrow
        • 25. Juni 2018 05:50
        • (bearbeitet)
        Оно то работает только:


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

        id                            type

        1                             data
        2                             key
        3                             stat

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


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

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


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

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



            Arrow
            • 25. Juni 2018 08: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. Juni 2018 08:04

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

                Arrow
                • 25. Juni 2018 09:12
                • (bearbeitet)
                И если можно еще один вопрос.

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


                Если писать:

                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. Juli 2018 06:14

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

                    Evgenii Legotckoi
                    • 3. Juli 2018 06:17

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

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

                      Ruslan Polupan
                      • 3. Juli 2018 08:27



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






                        Arrow
                        • 3. Juli 2018 08: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. Juli 2018 08:34

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

                            Ruslan Polupan
                            • 3. Juli 2018 08: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. Juli 2018 08:54

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

                                Arrow
                                • 3. Juli 2018 08:58
                                • (bearbeitet)

                                Такой вопрос. Никто случайно не писал свой класс для 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. Juli 2018 09:01

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

                                    Ruslan Polupan
                                    • 3. Juli 2018 09:03

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

                                      Arrow
                                      • 3. Juli 2018 09:07

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

                                        Kommentare

                                        Nur autorisierte Benutzer können Kommentare posten.
                                        Bitte Anmelden oder Registrieren
                                        Letzte Kommentare
                                        A
                                        ALO1ZE19. Oktober 2024 08:19
                                        Fb3-Dateileser auf Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                                        ИМ
                                        Игорь Максимов5. Oktober 2024 07:51
                                        Django – Lektion 064. So schreiben Sie eine Python-Markdown-Erweiterung Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                                        d
                                        dblas55. Juli 2024 11:02
                                        QML - Lektion 016. SQLite-Datenbank und das Arbeiten damit in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                                        k
                                        kmssr8. Februar 2024 18:43
                                        Qt Linux - Lektion 001. Autorun Qt-Anwendung unter Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                                        Qt WinAPI - Lektion 007. Arbeiten mit ICMP-Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                                        Jetzt im Forum diskutieren
                                        J
                                        JacobFib17. Oktober 2024 03:27
                                        добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
                                        JW
                                        Jhon Wick1. Oktober 2024 15:52
                                        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…
                                        КГ
                                        Кирилл Гусарев27. September 2024 09:09
                                        Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
                                        F
                                        Fynjy22. Juli 2024 04:15
                                        при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …

                                        Folgen Sie uns in sozialen Netzwerken