Павел Дорофеев
Павел Дорофеев29 мая 2022 г. 16:19

QSqlRelatipnalTabelModel Qt 4.8.1 как получить id внешней связи?

QSqlRelationalTableModel

На самом деле хочу поделится опытом как получать автоматически поле id внешней связи.

И хотелось бы услышать кто и как решает эту проблему.

Речь о том, что устанавливая связь с внешней таблицей QSqlRelation мы в итоговой таблице имеем текстовую замену значения id.

Но само id получить без отдельного запроса к базе данных теперь не представляется возможным.

Могу предолжить такой вариант: подправить исходники Qt (у нас 4.8.1) таким образом:

QString QSqlRelationalTableModel::selectStatement() const
{
Q_D(const QSqlRelationalTableModel);
QString query;

if (tableName().isEmpty())
    return query;
if (d->relations.isEmpty())
    return QSqlTableModel::selectStatement();

QString tList;
QString fList;
QString where;

QSqlRecord rec = d->baseRec;
QStringList tables;
const QRelation nullRelation;

// Count how many times each field name occurs in the record
QHash<QString, int> fieldNames;
QStringList fieldList;

for (int i = 0; i < rec.count(); ++i)
{
    QSqlRelation relation = d->relations.value(i, nullRelation).rel;
    QString name;
    if (relation.isValid())
    {
        // Count the display column name, not the original foreign key
        name = relation.displayColumn();
        if (d->db.driver()->isIdentifierEscaped(name, QSqlDriver::FieldName))
            name = d->db.driver()->stripDelimiters(name, QSqlDriver::FieldName);

        QSqlRecord rec = database().record(relation.tableName());
        for (int i = 0; i < rec.count(); ++i) {
            if (name.compare(rec.fieldName(i), Qt::CaseInsensitive) == 0) {
                name = rec.fieldName(i);
                break;
            }
        }
    }
    else
        name = rec.fieldName(i);
    fieldNames.insert(name, fieldNames.value(name, 0) + 1);
    fieldList.append(name);
}

//!! -------------- my --------------------------
QString my;

for (int i = 0; i < rec.count(); ++i)
{
    QSqlRelation relation = d->relations.value(i, nullRelation).rel;

    if (relation.isValid())
    {
        QString relTableAlias = QString::fromLatin1("relTblAl_%1").arg(i);

        if (!fList.isEmpty())
            fList.append(QLatin1String(", "));

        fList.append(d->relationField(relTableAlias , relation.displayColumn()));

        // If there are duplicate field names they must be aliased
        if (fieldNames.value(fieldList[i]) > 1)
        {
            QString relTableName = relation.tableName().section(QChar::fromLatin1('.'), -1, -1);

            if (d->db.driver()->isIdentifierEscaped(relTableName, QSqlDriver::TableName))
                relTableName = d->db.driver()->stripDelimiters(relTableName, QSqlDriver::TableName);

            QString displayColumn = relation.displayColumn();

            if (d->db.driver()->isIdentifierEscaped(displayColumn, QSqlDriver::FieldName))
                displayColumn = d->db.driver()->stripDelimiters(displayColumn, QSqlDriver::FieldName);
            fList.append(QString::fromLatin1(" AS %1_%2_%3").arg(relTableName).arg(displayColumn).arg(fieldNames.value(fieldList[i])));
            fieldNames.insert(fieldList[i], fieldNames.value(fieldList[i])-1);
        }

        //!! -------------- my --------------------------
        my.append(QLatin1String(", "));
        my.append(relTableAlias);
        my.append(QLatin1String("."));
        my.append(relation.indexColumn());
        my.append(QLatin1String(" as "));
        my.append(relation.tableName());
        my.append(QLatin1String("_"));
        my.append(relation.indexColumn());
        my.append(QLatin1String("_"));
        // --------------------------------------------

        if (d->joinMode == QSqlRelationalTableModel::InnerJoin)
        {
            // this needs fixing!! the below if is borken.
            // Use LeftJoin mode if you want correct behavior
            tables.append(relation.tableName().append(QLatin1Char(' ')).append(relTableAlias));

            if(!where.isEmpty())
                where.append(QLatin1String(" AND "));

            where.append(d->relationField(tableName(), d->db.driver()->escapeIdentifier(rec.fieldName(i), QSqlDriver::FieldName)));
            where.append(QLatin1String(" = "));
            where.append(d->relationField(relTableAlias, relation.indexColumn()));
        } else {
            tables.append(QLatin1String(" LEFT JOIN"));
            tables.append(relation.tableName().append(QLatin1Char(' ')).append(relTableAlias));
            tables.append(QLatin1String("ON"));

            QString clause;
            clause.append(d->relationField(tableName(), d->db.driver()->escapeIdentifier(rec.fieldName(i), QSqlDriver::FieldName)));
            clause.append(QLatin1String(" = "));
            clause.append(d->relationField(relTableAlias, relation.indexColumn()));

            tables.append(clause);
        }
    }
    else
    {
        if (!fList.isEmpty())
            fList.append(QLatin1String(", "));
        fList.append(d->relationField(tableName(), d->db.driver()->escapeIdentifier(rec.fieldName(i), QSqlDriver::FieldName)));
    }
}

if (d->joinMode == QSqlRelationalTableModel::InnerJoin && !tables.isEmpty())
{
    tList.append(tables.join(QLatin1String(", ")));
    if(!tList.isEmpty())
        tList.prepend(QLatin1String(", "));
} else
    tList.append(tables.join(QLatin1String(" ")));

if (fList.isEmpty())
    return query;

tList.prepend(tableName());
query.append(QLatin1String("SELECT "));

//!! --------------- my -------------------------
//!!query.append(fList).append(QLatin1String(" FROM ")).append(tList);
query.append(fList).append(my).append(QLatin1String(" FROM ")).append(tList);

if (d->joinMode == QSqlRelationalTableModel::InnerJoin) {
    qAppendWhereClause(query, where, filter());
} else if (!filter().isEmpty()) {
    query.append(QLatin1String(" WHERE ("));
    query.append(filter());
    query.append(QLatin1String(")"));
}

QString orderBy = orderByClause();
if (!orderBy.isEmpty())
    query.append(QLatin1Char(' ')).append(orderBy);

return query;

}

В результате в выборке select появится дополнительное поле к полю suppliers например suppliets_id_.

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

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

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

2
Павел Дорофеев
  • 18 июня 2022 г. 20:15

Есть еще принципиально другой вариант решить раз и навсегда вопрос с полей id внешней связи. Это форкнуть Qt 4.8.1 QSqlTableModel, то есть создать свою ветку развития.

Например создадим класс наследник от QSqlTableModel, но прямо в составе исходников Qt (папка src/sql/model). То есть в результате собираем только ветку sql для генерации QtSqld4.dll (пока отладочный вариант xxxxxd4.dll).

Класс обзовем пока QSqlRelationalTableModelMod1. Наследуемся в стиле самого Qt, то есть создаем также приватный класс QSqlRelationalTableModelMod1Private, который наследует QSqlTableModelPrivate.

Бета версию QSqlRelationalTableModelMod1 можно скачать уже здесь :
qsqlrelationaltablemodelmod1.zip qsqlrelationaltablemodelmod1.zip

    Павел Дорофеев
    • 25 января 2023 г. 22:49
    • Ответ был помечен как решение.

    Наконец-то готовы представить полноценное развитие Qt QSqlTableModel и QTableView.

    Посмотреть можно у нас на сайте здесь

    На github здесь здесь

    Радостная новость: не надо править и наследоваться от приватных классов Qt, только штатное наследование, то есть dll Qt остаются оригинальные.

    Не надо беспокоится насчет внешних связей. Не надо создавать кнопки вставки, удаления, копирования строки. Не надо делать поиск, отбор по значению. Все это уже реализовано и кстати для всех трех стратегий редактирования.

      Комментарии

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

      C++ - Тест 004. Указатели, Массивы и Циклы

      • Результат:60баллов,
      • Очки рейтинга-1
      СЦ

      C++ - Тест 004. Указатели, Массивы и Циклы

      • Результат:50баллов,
      • Очки рейтинга-4
      AT

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

      • Результат:73баллов,
      • Очки рейтинга1
      Последние комментарии
      J
      JonnyJo30 марта 2023 г. 18:57
      Qt/C++ - Урок 021. Рисование мышью в Qt Евгений, здравствуйте! Только начал изучение Qt и возник вопрос по 21ому уроку. После написания кода, выдаёт следующие ошибки В чём может быть проблема?
      АН
      Алексей Николаев26 марта 2023 г. 16:10
      Qt/C++ - Урок 042. PopUp уведомление в стиле Gnome с помощью Qt Добрый день, взял за основу ваш PopUp notification , и немного доработал его под свои нужды. Добавил в отдельном eventloop'e всплывающую очередь уведомлений с анимацией и таймеро…
      АН
      Алексей Николаев26 марта 2023 г. 16:04
      Qt/C++ - Урок 042. PopUp уведомление в стиле Gnome с помощью Qt Включите прозрачность в композит менеджере fly-admin-theme : fly-admin-theme ->Эффекты и всё заработает.
      Evgenii Legotckoi
      Evgenii Legotckoi24 марта 2023 г. 17:09
      Django - Урок 062. Как написать блочный шаблонный тег tabbar наподобие тега blocktranslate Почитайте эту статью про "хлебные крошки"
      Сейчас обсуждают на форуме
      BlinCT
      BlinCT1 апреля 2023 г. 12:16
      Нужен совет по работе с ListView и несколькими моделями Спасибо, сейчас займусь этим.
      NSProject
      NSProject31 марта 2023 г. 9:55
      Проверка комментария принадлежит он пользователю или нет DRF (Django Rest Framework) Здравствуйте! Сегодня я столкнулся с такой проблеммой. Существует модель комметариев. Где их соответственно достаточное количество. Все они выводятся при помощи запроса ajax (axios). Так ка…
      P
      Pisych30 марта 2023 г. 9:50
      Как подсчитать количество по условию? Да! Вот так работает! Огромное Вам спасибо! ........
      Evgenii Legotckoi
      Evgenii Legotckoi29 марта 2023 г. 11:11
      Замена поля ManyToMany Картинки точно нужно хранить в медиа директории на сервере, а для обращения использовать ImageField. Который будет хранить только путь к изображению на сервере. Хранить изображения в базе данных…
      ВА
      Виталий Анисимов29 января 2023 г. 23:17
      Как добавить виртуальную клавиатура с Т9 в своей проект на QML. Добрый день. Прошу помочь, пишу небольше приложение в Qt. Добвил в свой проект виртуальную клавиатуру от Qt. Но как добавить в него возможность изменения Т9 никак не могу понять.

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