ДК
9 апреля 2020 г. 3:31

QSqlTableModel + QTableView + Drag&Drop

QSqlDatabase, Drag and Drop, QSqlTableModel, Qt, QTableView

Привет. Делаю реализацию перемещения строк на QTableView с моделью QSqlTableModel. Буду в этой теме спрашивать нужное.
Пока такой вопрос при создании модели: как мне узнать rowCount?

int SqlTableModelDrAndDr::rowCount(const QModelIndex &parent) const
{
    return ??
}

пока оставил

return QSqlQueryModel::rowCount();
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.
2
ДК

всё никак не получается, хочу на выходных победить это дело. Сейчас поведение такое- могу взять строку, когда её дропаю создается в пустая строка ниже последней, ничего никуда не перемещается, всё остаётся на своих местах. Подскажите, какие методы обычно переопределяются для dr&dr? Я уже обчитался литературы, документации, хотя QAbstr... модельные классы и так не плохо знаю. Сейчас решил что в моем случае хватает переопределения методов flags(...), supportedDropActions(); и dropMimeData(...) .

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;
}

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

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 rows = 0;

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

    rows = 1;
    qDebug() << "row insert:" << insertRows(beginRow, rows, QModelIndex());
    int i = 0;
    foreach (const QString &text, newItems) {
    QModelIndex idx = index(beginRow, i++, QModelIndex());
        setData(idx, text);
        beginRow++;
    }
    return true;
}

на вьюхе так:

ui->_reportsTableView->setDragDropMode(QAbstractItemView::DragDropMode::DragDrop);
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);

что надо добавить, какие методы еще переопределить или что не так написано? код тестовый в dropMimeData, есть там лишнее, знаю.

ДК

вот так вставляется пустая строка, код мне подсказали. Только при перемещении строки она удаляется и пересоздаётся в БД. Наверное поэтому, вставляется уже пустая строка. Пока думаю поймать момент перед удалением информации и скопировать её с индексом вставки, а потом уже вызвать insertRows с этими данными.

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;
}

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

bool SqlTableModelDrAndDr::dropMimeData(const QMimeData *data, Qt::DropAction action, int row,
                                        int column, const QModelIndex &parent)
{
    qDebug() << "dropMimeData" << action << row << column << 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 rows = 0;

    int r;
    stream >> r;

    while (!stream.atEnd()) {
        QString text;
        stream >> text;
        newItems << text;
        ++rows;
    }
    qDebug() << beginRow << newItems;
    qDebug() << "row remove:" << r << removeRows(r, 1, QModelIndex());
    submitAll();
    if (beginRow > r)
        --beginRow;
    rows = 1;
    qDebug() << "row insert:" << insertRows(beginRow, rows, QModelIndex());
    int i = 0;
    foreach (const QString &text, newItems) {
        QModelIndex idx = index(beginRow, i++, QModelIndex());
        setData(idx, text);
            //beginRow++;
    }
    return true;
}
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(SqlTableModelDrAndDr::OnManualSubmit);

*день спустя
код выше рабочий, у меня просто в бд какие-то проблемы и есть одна колонка, скрыв которую, дроп работает через одно место, если её показывать, то всё отрабатывает, как надо.

Комментарии

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

Внесите вклад в развитие сообщества EVILEG.

Узнайте, как стать автором сайта.

Изучить
Donate

Добрый день, Дорогие Пользователи !!!

Я Евгений Легоцкой, разработчик EVILEG. И это мой хобби-проект, который помогает учиться программированию другим программистам и разработчикам

Если сайт помог вам, и вы хотите также поддержать развитие сайта, то вы можете сделать пожертвование следующими способами

PayPalYandex.Money
Timeweb

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

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

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

Посмотреть Хостинг Timeweb
АС
26 мая 2020 г. 12:29
Артём Сун-Дун-Чан

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

  • Результат:50баллов,
  • Очки рейтинга-4
МН
25 мая 2020 г. 12:33
Митя Нагибин

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

  • Результат:50баллов,
  • Очки рейтинга-4
f
25 мая 2020 г. 6:05
falcon

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

  • Результат:66баллов,
  • Очки рейтинга-1
Последние комментарии
29 мая 2020 г. 7:30
Евгений Легоцкой

Qt/C++ - Урок 039. Как закрасить строку в QSqlTableModel по значению в столбце

У меня работает. Исправлял в проекте, который приложен к статье. А что происходит в вашем коде, с учётом места вызова этого кода, я знать не могу ;) Дебажьте и добавляйте условия, кото…
МА
29 мая 2020 г. 7:27
Михаил А

Qt/C++ - Урок 039. Как закрасить строку в QSqlTableModel по значению в столбце

QModelIndexList rowIndexes = ui->tableView->selectionModel()->selectedRows(); model->removeRows(rowIndexes.first().row(), rowIndexes.size()); model-&…
29 мая 2020 г. 7:14
Евгений Легоцкой

Django - Урок 036. Как добавить аутентификацию через социальные сети. ВКонтакте

Неправильно прописали URL, на который возвращается ответ от OAuth ВКонтакте. Настраивайте ваше приложение в консоли разработчика ВКонтакте
АЛ
29 мая 2020 г. 6:24
Александр Леонидов

Django - Урок 036. Как добавить аутентификацию через социальные сети. ВКонтакте

Здравствуйте! После клика на ссылку авторизации выдает json словарь с ошибкой: {"error":"invalid_request","error_description":"redirect_uri is incorrect, check application redirect uri in …
28 мая 2020 г. 16:14
Евгений Легоцкой

Qt/C++ - Урок 039. Как закрасить строку в QSqlTableModel по значению в столбце

Ну в моём примере, который в статье сработало так model->setData(model->index(1, 1), 7); Поскольку model->index(1, 0) - это индекс колонки id, которая скрыта, поэтому…
Сейчас обсуждают на форуме
29 мая 2020 г. 8:52
Vladimir Sergeevich

Масштабирование двумя пальцами на мобильных платформах

Я планировал описать этот момент на блоге, но никак руки не доходят (уже год). Летом дойдут. Тем не менее, у меня в репозитории лежит рабочий код игрушки "пазлы", где есть все это. …
29 мая 2020 г. 7:51
Евгений Легоцкой

Графическое ускорение

Зависит от платформы и поддерживаемых технологий. В QML в первую очередь используется OpenGL и отрисовка производится средствами GPU. Но может переключаться на использование CPU и прог…
ИП
29 мая 2020 г. 2:55
Игорь Порошин

QTablwView + QSqlQueryModel скрыть пустой столбец

Да, понятно. В данном случае лучше использовать серверную процедуру (если такие поддерживаются), в которой будет проверяться наличие всех пустых строк у нужного столбца и вызываться соответ…
RG
28 мая 2020 г. 19:21
Rovshan Gurbanov

Сборка под старые версии Android

У меня SDK почти все версии есть, NDK есть версии 10, 17, 21. Но собирается приложение только с NDK v21 под Android версии 7.0 и выше Версия Qt у меня 5.14.2
28 мая 2020 г. 8:58
Евгений Легоцкой

Освобождение памяти QMainWindow::setCentralWidget

Да, соглашусь. Просто удаление происходит позже, а не сразу.
О нас
Услуги
© EVILEG 2015-2020
Рекомендует хостинг TIMEWEB