Павел Дорофеев
Павел ДорофеевМамыр 29, 2022, 6: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, 10: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, 11:49 Т.Ж.
    • Жауап шешім ретінде белгіленді.

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

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

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

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

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

      Пікірлер

      Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
      Кіріңіз немесе Тіркеліңіз
      OI
      • Ora Iro
      • Жел. 24, 2024, 6:38 Т.Ж.

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

      • Нәтиже:40ұпай,
      • Бағалау ұпайлары-8
      AD

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

      • Нәтиже:50ұпай,
      • Бағалау ұпайлары-4
      m
      • molni99
      • Қаз. 26, 2024, 1:37 Т.Ж.

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

      • Нәтиже:80ұпай,
      • Бағалау ұпайлары4
      Соңғы пікірлер
      ИМ
      Игорь МаксимовҚар. 22, 2024, 11:51 Т.Ж.
      Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
      Evgenii Legotckoi
      Evgenii LegotckoiҚаз. 31, 2024, 2:37 Т.Қ.
      Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
      A
      ALO1ZEҚаз. 19, 2024, 8:19 Т.Ж.
      Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
      ИМ
      Игорь МаксимовҚаз. 5, 2024, 7:51 Т.Ж.
      Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
      d
      dblas5Шілде 5, 2024, 11:02 Т.Ж.
      QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
      Енді форумда талқылаңыз
      Evgenii Legotckoi
      Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
      добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
      t
      tonypeachey1Қар. 15, 2024, 6:04 Т.Ж.
      google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
      NSProject
      NSProjectМаусым 4, 2022, 3:49 Т.Ж.
      Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
      9
      9AnonimҚаз. 25, 2024, 9:10 Т.Ж.
      Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

      Бізді әлеуметтік желілерде бақылаңыз