1
mirror of https://invent.kde.org/network/falkon.git synced 2024-09-23 02:32:10 +02:00
falkonOfficial/src/lib/tabwidget/tabmodel.cpp

222 lines
5.9 KiB
C++
Raw Normal View History

2018-01-30 14:53:18 +01:00
/* ============================================================
* Falkon - Qt web browser
* Copyright (C) 2018 David Rosca <nowrep@gmail.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
* ============================================================ */
#include "tabmodel.h"
#include "webtab.h"
#include "tabwidget.h"
#include "tabbedwebview.h"
#include "browserwindow.h"
2018-01-30 18:55:18 +01:00
#include <QMimeData>
2018-01-30 14:53:18 +01:00
TabModel::TabModel(BrowserWindow *window, QObject *parent)
: QAbstractListModel(parent)
, m_window(window)
{
init();
}
QModelIndex TabModel::tabIndex(WebTab *tab) const
2018-01-30 14:53:18 +01:00
{
const int idx = m_tabs.indexOf(tab);
if (idx < 0) {
return QModelIndex();
}
return index(idx);
}
WebTab *TabModel::tab(const QModelIndex &index) const
{
return m_tabs.value(index.row());
2018-01-30 14:53:18 +01:00
}
int TabModel::rowCount(const QModelIndex &parent) const
{
if (parent.isValid()) {
return 0;
}
return m_tabs.count();
2018-01-30 14:53:18 +01:00
}
2018-01-30 18:55:18 +01:00
Qt::ItemFlags TabModel::flags(const QModelIndex &index) const
{
if (!index.isValid()) {
return Qt::ItemIsDropEnabled;
}
return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsDragEnabled;
}
2018-01-30 14:53:18 +01:00
QVariant TabModel::data(const QModelIndex &index, int role) const
{
if (index.row() < 0 || index.row() > m_tabs.count()) {
2018-01-30 14:53:18 +01:00
return QVariant();
}
WebTab *t = tab(index);
if (!t) {
2018-01-30 14:53:18 +01:00
return QVariant();
}
switch (role) {
case WebTabRole:
return QVariant::fromValue(t);
2018-01-30 14:53:18 +01:00
case TitleRole:
case Qt::DisplayRole:
return t->title();
2018-01-30 14:53:18 +01:00
case IconRole:
case Qt::DecorationRole:
return t->icon();
2018-01-30 14:53:18 +01:00
case PinnedRole:
return t->isPinned();
2018-01-30 14:53:18 +01:00
case RestoredRole:
return t->isRestored();
case CurrentTabRole:
return t->isCurrentTab();
case LoadingRole:
return t->isLoading();
2018-01-30 14:53:18 +01:00
default:
return QVariant();
}
}
2018-01-30 18:55:18 +01:00
Qt::DropActions TabModel::supportedDropActions() const
{
return Qt::MoveAction;
}
#define MIMETYPE QStringLiteral("application/falkon.tabmodel.tab")
QStringList TabModel::mimeTypes() const
{
return {MIMETYPE};
}
QMimeData *TabModel::mimeData(const QModelIndexList &indexes) const
{
QByteArray data;
QDataStream stream(&data, QIODevice::WriteOnly);
for (const QModelIndex &index : indexes) {
if (index.isValid() && index.column() == 0) {
stream << index.row();
}
}
QMimeData *mimeData = new QMimeData();
mimeData->setData(MIMETYPE, data);
return mimeData;
}
bool TabModel::dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent)
{
if (action == Qt::IgnoreAction) {
return true;
}
if (!m_window || !data->hasFormat(MIMETYPE) || parent.isValid() || column != 0) {
return false;
}
QByteArray encodedData = data->data(MIMETYPE);
QDataStream stream(&encodedData, QIODevice::ReadOnly);
QVector<WebTab*> tabs;
while (!stream.atEnd()) {
int idx;
stream >> idx;
WebTab *t = tab(index(idx));
if (t) {
tabs.append(t);
2018-01-30 18:55:18 +01:00
}
}
if (tabs.isEmpty()) {
return false;
}
for (int i = 0; i < tabs.count(); ++i) {
const int from = tabs.at(i)->tabIndex();
const int to = row >= from ? row - 1 : row++;
// FIXME: This switches order when moving > 2 non-contiguous indices
m_window->tabWidget()->moveTab(from, to);
}
return true;
}
2018-01-30 14:53:18 +01:00
void TabModel::init()
{
for (int i = 0; i < m_window->tabCount(); ++i) {
tabInserted(i);
}
connect(m_window->tabWidget(), &TabWidget::tabInserted, this, &TabModel::tabInserted);
connect(m_window->tabWidget(), &TabWidget::tabRemoved, this, &TabModel::tabRemoved);
connect(m_window->tabWidget(), &TabWidget::tabMoved, this, &TabModel::tabMoved);
connect(m_window, &QObject::destroyed, this, [this]() {
beginResetModel();
m_window = nullptr;
m_tabs.clear();
endResetModel();
});
2018-01-30 14:53:18 +01:00
}
void TabModel::tabInserted(int index)
{
WebTab *tab = m_window->weView(index)->webTab();
beginInsertRows(QModelIndex(), index, index);
m_tabs.insert(index, tab);
endInsertRows();
auto emitDataChanged = [this](WebTab *tab, int role) {
const QModelIndex idx = tabIndex(tab);
2018-01-30 14:53:18 +01:00
emit dataChanged(idx, idx, {role});
};
connect(tab, &WebTab::titleChanged, this, std::bind(emitDataChanged, tab, Qt::DisplayRole));
connect(tab, &WebTab::titleChanged, this, std::bind(emitDataChanged, tab, TitleRole));
connect(tab, &WebTab::iconChanged, this, std::bind(emitDataChanged, tab, Qt::DecorationRole));
connect(tab, &WebTab::iconChanged, this, std::bind(emitDataChanged, tab, IconRole));
connect(tab, &WebTab::pinnedChanged, this, std::bind(emitDataChanged, tab, PinnedRole));
connect(tab, &WebTab::restoredChanged, this, std::bind(emitDataChanged, tab, RestoredRole));
connect(tab, &WebTab::currentTabChanged, this, std::bind(emitDataChanged, tab, CurrentTabRole));
connect(tab, &WebTab::loadingChanged, this, std::bind(emitDataChanged, tab, LoadingRole));
2018-01-30 14:53:18 +01:00
}
void TabModel::tabRemoved(int index)
{
beginRemoveRows(QModelIndex(), index, index);
m_tabs.remove(index);
endRemoveRows();
}
void TabModel::tabMoved(int from, int to)
{
beginMoveRows(QModelIndex(), from, from, QModelIndex(), to > from ? to + 1 : to);
m_tabs.insert(to, m_tabs.takeAt(from));
endMoveRows();
}