ДК
Джон КофиJuly 11, 2020, 7: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, 3:49 a.m.

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

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

    int r;
    stream >> r;

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

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

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

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

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

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

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

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

        благодорю!

          Comments

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

          C ++ - Test 004. Pointers, Arrays and Loops

          • Result:50points,
          • Rating points-4
          m

          C ++ - Test 004. Pointers, Arrays and Loops

          • Result:80points,
          • Rating points4
          m

          C ++ - Test 004. Pointers, Arrays and Loops

          • Result:20points,
          • Rating points-10
          Last comments
          ИМ
          Игорь МаксимовNov. 22, 2024, 11:51 a.m.
          Django - Tutorial 017. Customize the login page to Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
          Evgenii Legotckoi
          Evgenii LegotckoiOct. 31, 2024, 2:37 p.m.
          Django - Lesson 064. How to write a Python Markdown extension Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
          A
          ALO1ZEOct. 19, 2024, 8:19 a.m.
          Fb3 file reader on Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
          ИМ
          Игорь МаксимовOct. 5, 2024, 7:51 a.m.
          Django - Lesson 064. How to write a Python Markdown extension Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
          d
          dblas5July 5, 2024, 11:02 a.m.
          QML - Lesson 016. SQLite database and the working with it in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
          Now discuss on the forum
          Evgenii Legotckoi
          Evgenii LegotckoiJune 24, 2024, 3:11 p.m.
          добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
          t
          tonypeachey1Nov. 15, 2024, 6:04 a.m.
          google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
          NSProject
          NSProjectJune 4, 2022, 3:49 a.m.
          Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
          9
          9AnonimOct. 25, 2024, 9:10 a.m.
          Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

          Follow us in social networks