Павел Дорофеев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 хостинг.
1

Есть еще принципиально другой вариант решить раз и навсегда вопрос с полей 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

Комментарии

Только авторизованные пользователи могут публиковать комментарии.
Пожалуйста, авторизуйтесь или зарегистрируйтесь
Card image cap
Pulsum Via

Проект для путешественников от EVILEG.

Перейти
Timeweb

Позвольте мне порекомендовать вам отличный хостинг, на котором расположен EVILEG.

В течение многих лет Timeweb доказывает свою стабильность.

Для проектов на Django рекомендую VDS хостинг

Посмотреть Хостинг
Поделиться в социальных сетях
Donate

Проект EVILEG перешёл на некоммерческую основу и будет развиваться исключительно на энтузиазме создателя сайта, энтузиазме пользователей, пожертвованиях и реферальной системе хостинга

Спасибо за вашу поддержку

Доступные способы поддержки проекта

PayPal

PatreonYooMoneyПодробнее
s

C++ - Тест 002. Константы

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

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

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

C++ - Тест 002. Константы

  • Результат:58баллов,
  • Очки рейтинга-2
Последние комментарии

Qt/C++ - Урок 061. Добавление изображений в приложение методом Drag And Drop из файлового менеджера

Доброго времени суток. А если нужно и изображение и текст? Что-то потерялся немного.... // Вместо отрисовки иконки и текста будем отрисовывать только одно изображение // с н…
АС

Qt/C++ - Урок 004. QSqlTableModel или Как представить таблицу из БД в Qt?

error insert into TableExample " Количество параметров не совпадает" Я путь свой прописывала и даже бд удаляла, чтобы заново сделать, не работает. (всё остальное как у вас... Вроде ка…
i
ЛД

GameDev на Qt - Урок 1. Отслеживание перемещения мыши в QGraphicsScene

Вполне возможно, что ты не закинул graphicsView в дизайнере в виджет
ЛД

GameDev на Qt - Урок 1. Отслеживание перемещения мыши в QGraphicsScene

Кому интересно, поворот в slotTarget можно в одну строку организовать this->setRotation(90 + rotation() + qRadiansToDegrees(qAtan2(mapFromScene(point).y(), mapFromScene(point).x())));
Сейчас обсуждают на форуме
K

Sorting the added QML elements in the ListModel

buy accutane online forum https://isotretinoinxp.top
  • Nomad
  • 30 июля 2022 г. 5:42

Как работать с HTMX?

Приветствую колеги. На днях наткнулся на вот это : https://htmx.org/ На офф сайте написанно вот такая фраза: htmx gives you access to AJAX, CSS Transitions, We…
h
  • harisr
  • 25 июля 2022 г. 2:56

QT - Native App Integration

Привет, у нас уже есть собственное приложение для Android. Можем ли мы интегрировать пользовательское представление QT в приложение со всем приложением QT внутри представления. Если да, ука…

Правильный запуск сервера на vps - Django

О я как то себе дома локальный сервер создавал. Вам же нужно просто сделать ручками конфигурацию системы. Настроить Nginx ну либо Apache (тут кому что нравится). Соответственно БД и всё остально…
o

Распознание объектов

Я к тому, что, возможно, софт уже есть.
О нас
Услуги
© EVILEG 2015-2022
Рекомендует хостинг TIMEWEB