ДК
Джон КофиJuly 11, 2020, 6:51 a.m.

Drug Drop problems

Drug & Drop, QMimeData, Qt, QSqlTableModel

Привет. Есть делегат QSqlTableModel, на основе которого сделал Drug Drop.
Проблема в том, что перетаскивание работает корректно только тогда, когда я отображаю первую колонку - ID, если её скрыть, то поля съезжают и вместо перенесения строки отрабатывает вставка пустой, а копируемые данные меняются полями, иногда строку переносит нормально, но поля съезжают всё-равно.
Вот я и разбираюсь теперь в отладчике. Возникают вопросы, ответы на которые не нашел:
1)QMimeData копирует только отображенную информацию или также и из скрытых колонок? Просто в отладчике, что вижу: если отображенные поля есть все,
то имею все данные отображенных полей, данные одного скрытого, а еще данные от 2 скрытых нет.
Конкретно про колонку с ID, если я отображаю колонку с ID, то вижу, что данные с неё заносятся в QMimeData, если скрываю её, то этот ID в QMimeData не заносится.
2)Где можно почитать про ВСЕ форматы данных для QMimeData?
Я использую "application/x-qabstractitemmodeldatalist", а какие еще есть? Просто в доке такого формата вообще не нашел, а именно с ним работает.

Сам делегат:

SqlTableModelDrAndDr::SqlTableModelDrAndDr(QObject *parent, QSqlDatabase db) :
    QSqlTableModel(parent, db)
{
}

Qt::DropActions SqlTableModelDrAndDr::supportedDropActions() const
{
    return Qt::CopyAction | Qt::MoveAction;
}

Qt::ItemFlags SqlTableModelDrAndDr::flags(const QModelIndex &index) const
{
    Qt::ItemFlags defaultFlags = QSqlTableModel::flags(index);

    if (index.isValid())
        return Qt::ItemIsDragEnabled | Qt::ItemIsDropEnabled | defaultFlags;
    else
        return Qt::ItemIsDropEnabled | defaultFlags;
}

QMimeData *SqlTableModelDrAndDr::mimeData(const QModelIndexList &indexes) const
{
    QMimeData *mimeData = new QMimeData();
    QByteArray encodedData;

    QDataStream stream(&encodedData, QIODevice::WriteOnly);

    stream << indexes[0].row();

    foreach (const QModelIndex &index, indexes) {
        if (index.isValid()) {
            QString text = data(index, Qt::DisplayRole).toString();
            stream << text;
        }
    }

    mimeData->setData("application/x-qabstractitemmodeldatalist", encodedData);
    return mimeData;
}

bool SqlTableModelDrAndDr::canDropMimeData(const QMimeData *data,
    Qt::DropAction action, int row, int column, const QModelIndex &parent) const
{
    Q_UNUSED(action)
    Q_UNUSED(row)
    Q_UNUSED(column)
    Q_UNUSED(parent)

    if (!data->hasFormat("application/x-qabstractitemmodeldatalist"))
        return false;

    return true;
}

bool SqlTableModelDrAndDr::dropMimeData(const QMimeData *data,
      Qt::DropAction action, int row, int column, const QModelIndex &parent)
{
    if (!canDropMimeData(data, action, row, column, parent))
        return false;

    if (action == Qt::IgnoreAction)
        return true;

    int beginRow;

    if (row != -1)
        beginRow = row;
    else if (parent.isValid())
        beginRow = parent.row();
    else
        beginRow = rowCount(QModelIndex());

    QByteArray encodedData = data->data("application/x-qabstractitemmodeldatalist");
    QDataStream stream(&encodedData, QIODevice::ReadOnly);
    QStringList newItems;

    int r;
    stream >> r;

    while (!stream.atEnd()) {
        QString text;
        stream >> text;
        newItems << text;
    }

    removeRow(r, QModelIndex());
    submitAll();
    if (beginRow > r)
        --beginRow;

    _finishRow = beginRow;
    insertRow(beginRow, QModelIndex());
    int i = 0;
    foreach (const QString &text, newItems) {
        QModelIndex idx = index(beginRow, i++, QModelIndex());
        setData(idx, text);
    }

    return true;
}

Настройки вью и модели:

_reportsSQL->setSort(ReportDataBase::WEIGHT, Qt::SortOrder::AscendingOrder);
ui->_reportsTableView->setEditTriggers(QTableView::NoEditTriggers);
ui->_reportsTableView->setModel(_reportsSQL);
/*
 * Установказа заголовков
 * */
for (int i = 0, j = 0; i < _reportsSQL->columnCount(); ++i, ++j)
{
    _reportsSQL->setHeaderData(i, Qt::Horizontal, REPORTDATABASE->header(j).toUpper());
}

if(!_roles.contains(Models::UserRole::ROLES::ADMIN))
    ui->_reportsTableView->setColumnHidden(ReportDataBase::PATH, true);
ui->_reportsTableView->setColumnHidden(ReportDataBase::ID,true);
ui->_reportsTableView->setColumnHidden(ReportDataBase::SERVER_ID,true);
ui->_reportsTableView->setColumnHidden(ReportDataBase::LAB_ID,true);
ui->_reportsTableView->setColumnHidden(ReportDataBase::WEIGHT, true);
ui->_reportsTableView->setItemDelegateForColumn(ReportDataBase::PATH, pathDelegate);
ui->_reportsTableView->setSelectionBehavior(QAbstractItemView::SelectRows);
ui->_reportsTableView->setSelectionMode(QAbstractItemView::SingleSelection);
ui->_reportsTableView->horizontalHeader()->setSectionResizeMode(QHeaderView::ResizeToContents);
ui->_reportsTableView->horizontalHeader()->setStretchLastSection(true);
ui->_reportsTableView->horizontalHeader()->moveSection(ReportDataBase::PATH, ReportDataBase::DESCRIPTION);
ui->_reportsTableView->verticalHeader()->hide();

/*
 * For drag and drop
 * */
ui->_reportsTableView->setDragDropMode(QAbstractItemView::DragDropMode::InternalMove);
ui->_reportsTableView->setDragEnabled(true);
ui->_reportsTableView->setAcceptDrops(true);
ui->_reportsTableView->setDropIndicatorShown(true);
ui->_reportsTableView->viewport()->setAcceptDrops(true);
ui->_reportsTableView->setDefaultDropAction(Qt::MoveAction);
ui->_reportsTableView->setDragDropOverwriteMode(false);
_reportsSQL->setEditStrategy(QSqlTableModel::OnManualSubmit);

В классе, управляющим QSqlDatabase, поля установлены верно, если бы это было не так, то, навреное, и на вью информация отображалась не в правильных колонках.

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!

4
Evgenii Legotckoi
  • July 13, 2020, 2:49 a.m.

Добрый день.
mime data можно именовать как заблагоросудится. Просто есть стандартные типы данных, а есть кастомные. То есть разработчики сами решают как их называть.
Копирование со всех колонок скорее всего происходит из-за того, что у вас настроено построчное выделение.
По идее копировать должно только видимые поля, хотя при желании можно и переписать поведение. Но если учесть описание проблемы с первой колонкой и приложенный код, то становится ясно, что это ожидаемое поведение

В этом коде по логике забирается первая строка из QMimeData, если колонка скрыта, то естественно, что данные могут интерпретироваться как угодно. Может оказаться нужная строка, а может быть мусор.

    int r;
    stream >> r;

Так что тут нужно продумывать логику и решать, что конкретно вам нужно.

    ДК
    • July 13, 2020, 3:58 a.m.

    Доброе утро! спасибо за ответ! а как настраивается поведение, чтобы копировались все поля всегда? тогда и проблем не будет со скрытыми колонками. Я имею ввиду в делегате что-то переделывать с data надо или просто есть какой-то флаг под эту задачу?

      Evgenii Legotckoi
      • July 13, 2020, 4:14 a.m.
      • The answer was marked as a solution.

      Я думаю, что вам этот метод нужно переписывать

      QMimeData *SqlTableModelDrAndDr::mimeData(const QModelIndexList &indexes) const
      

      То есть доделывать недостающие индексы, например, и по ним забирать данные.

        ДК
        • July 13, 2020, 6:43 a.m.

        благодорю!

          Comments

          Only authorized users can post comments.
          Please, Log in or Sign up
          B

          C++ - Test 002. Constants

          • Result:16points,
          • Rating points-10
          B

          C++ - Test 001. The first program and data types

          • Result:46points,
          • Rating points-6
          FL

          C++ - Test 006. Enumerations

          • Result:80points,
          • Rating points4
          Last comments
          k
          kmssrFeb. 8, 2024, 5:43 p.m.
          Qt Linux - Lesson 001. Autorun Qt application under Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
          Qt WinAPI - Lesson 007. Working with ICMP Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
          EVA
          EVADec. 25, 2023, 9:30 a.m.
          Boost - static linking in CMake project under Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
          J
          JonnyJoDec. 25, 2023, 7:38 a.m.
          Boost - static linking in CMake project under Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
          G
          GvozdikDec. 18, 2023, 8:01 p.m.
          Qt/C++ - Lesson 056. Connecting the Boost library in Qt for MinGW and MSVC compilers Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
          Now discuss on the forum
          AC
          Alexandru CodreanuJan. 19, 2024, 10:57 a.m.
          QML Обнулить значения SpinBox Доброго времени суток, не могу разобраться с обнулением значение SpinBox находящего в делегате. import QtQuickimport QtQuick.ControlsWindow { width: 640 height: 480 visible: tr…
          BlinCT
          BlinCTDec. 27, 2023, 7:57 a.m.
          Растягивать Image на парент по высоте Ну и само собою дял включения scrollbar надо чтобы был Flickable. Так что выходит как то так Flickable{ id: root anchors.fill: parent clip: true property url linkFile p…
          Дмитрий
          ДмитрийJan. 10, 2024, 3:18 a.m.
          Qt Creator загружает всю оперативную память Проблема решена. Удалось разобраться с помощью утилиты strace. Запустил ее: strace ./qtcreator Начал выводиться весь лог работы креатора. В один момент он начал считывать фай…
          Evgenii Legotckoi
          Evgenii LegotckoiDec. 12, 2023, 5:48 a.m.
          Побуквенное сравнение двух строк Добрый день. Там случайно не высылается этот сигнал textChanged ещё и при форматировани текста? Если решиать в лоб, то можно просто отключать сигнал/слотовое соединение внутри слота и …

          Follow us in social networks