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

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

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
























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, 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
                                        Ua

                                        Qt - Test 001. Signals and slots

                                        • Result:84points,
                                        • Rating points4
                                        Ua

                                        Qt - Test 001. Signals and slots

                                        • Result:42points,
                                        • Rating points-8
                                        ОК

                                        Qt - Test 001. Signals and slots

                                        • Result:47points,
                                        • Rating points-6
                                        Last comments
                                        ИМ
                                        Игорь МаксимовNov. 22, 2024, 9:51 p.m.
                                        Django - Tutorial 017. Customize the login page to Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                                        Evgenii Legotckoi
                                        Evgenii LegotckoiOct. 31, 2024, 11:37 p.m.
                                        Django - Lesson 064. How to write a Python Markdown extension Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                                        A
                                        ALO1ZEOct. 19, 2024, 5:19 p.m.
                                        Fb3 file reader on Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                                        ИМ
                                        Игорь МаксимовOct. 5, 2024, 4:51 p.m.
                                        Django - Lesson 064. How to write a Python Markdown extension Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                                        d
                                        dblas5July 5, 2024, 8:02 p.m.
                                        QML - Lesson 016. SQLite database and the working with it in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                                        Now discuss on the forum
                                        f
                                        firstlunoxodFeb. 15, 2025, 1:46 p.m.
                                        Рисование на QGraphicsScene при зажатой кнопке мыши Подскажите, пожалуйста! Как данный класс можно дополнить, чтобы созданные объекты можно было перемещать мышкой по сцене?
                                        Дмитрий
                                        ДмитрийFeb. 3, 2025, 4:24 p.m.
                                        Создание deb-пакета. Как создать ярлык на рабочем столе после установки собственного deb-пакета? Всем привет. Сделал свой deb-пакет с программой. Всё устанавливается и работает. Ставлю по пути /usr/bin/my_application. Как для пользователя при установке пакета сразу создать ярлык на раб…
                                        NW
                                        Nayo WaiJan. 30, 2025, 7:22 p.m.
                                        не запускается компьютер!!! Не запускается компьютер (точнее работает блок , но сам монитор вообще жесть)В общем я ничего с интернета не скачивала в последнее время. На компе никаких левых пр…
                                        n
                                        nklyJan. 3, 2025, 12:52 p.m.
                                        Нужно запретить перемещение только некоторых итемов, остальные перемещать можно. Вопрос решен. Узнать QModelIndex элемента на который мы перетаскиваем другой элемент, можно с помощью функции indexAt(event->position().toPoint()) представления QTreeViev вызываемой в переопр…
                                        M
                                        MarselAug. 17, 2023, 12:26 a.m.
                                        OAuth2.0 через VK, получение email Спасибо большое за помощь и простите за то что отнял время своей невнимательностью.

                                        Follow us in social networks