You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
448 lines
15 KiB
C++
448 lines
15 KiB
C++
/*
|
|
* Copyright 2021 Michail Vourlakos <mvourlakos@gmail.com>
|
|
*
|
|
* This file is part of Latte-Dock
|
|
*
|
|
* Latte-Dock 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 2 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* Latte-Dock 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 "viewscontroller.h"
|
|
|
|
// local
|
|
#include "ui_viewsdialog.h"
|
|
#include "viewsdialog.h"
|
|
#include "viewshandler.h"
|
|
#include "viewsmodel.h"
|
|
#include "viewstableview.h"
|
|
#include "delegates/namedelegate.h"
|
|
#include "delegates/singleoptiondelegate.h"
|
|
#include "delegates/singletextdelegate.h"
|
|
#include "../generic/generictools.h"
|
|
#include "../settingsdialog/templateskeeper.h"
|
|
#include "../../layout/centrallayout.h"
|
|
#include "../../layouts/manager.h"
|
|
#include "../../layouts/synchronizer.h"
|
|
|
|
// Qt
|
|
#include <QHeaderView>
|
|
#include <QItemSelection>
|
|
|
|
// KDE
|
|
#include <KMessageWidget>
|
|
|
|
|
|
namespace Latte {
|
|
namespace Settings {
|
|
namespace Controller {
|
|
|
|
|
|
Views::Views(Settings::Handler::ViewsHandler *parent)
|
|
: QObject(parent),
|
|
m_handler(parent),
|
|
m_model(new Model::Views(this, m_handler->corona())),
|
|
m_proxyModel(new QSortFilterProxyModel(this)),
|
|
m_view(m_handler->ui()->viewsTable),
|
|
m_storage(KConfigGroup(KSharedConfig::openConfig(),"LatteSettingsDialog").group("ViewsDialog"))
|
|
{
|
|
loadConfig();
|
|
m_proxyModel->setSourceModel(m_model);
|
|
|
|
connect(m_model, &QAbstractItemModel::dataChanged, this, &Views::dataChanged);
|
|
connect(m_model, &Model::Views::rowsInserted, this, &Views::dataChanged);
|
|
connect(m_model, &Model::Views::rowsRemoved, this, &Views::dataChanged);
|
|
|
|
connect(m_handler, &Handler::ViewsHandler::currentLayoutChanged, this, &Views::onCurrentLayoutChanged);
|
|
|
|
init();
|
|
}
|
|
|
|
Views::~Views()
|
|
{
|
|
saveConfig();
|
|
}
|
|
|
|
QAbstractItemModel *Views::proxyModel() const
|
|
{
|
|
return m_proxyModel;
|
|
}
|
|
|
|
QAbstractItemModel *Views::baseModel() const
|
|
{
|
|
return m_model;
|
|
}
|
|
|
|
QTableView *Views::view() const
|
|
{
|
|
return m_view;
|
|
}
|
|
|
|
void Views::init()
|
|
{
|
|
m_view->setModel(m_proxyModel);
|
|
//m_view->setHorizontalHeader(m_headerView);
|
|
m_view->verticalHeader()->setVisible(false);
|
|
m_view->setSortingEnabled(true);
|
|
|
|
m_proxyModel->setSortRole(Model::Views::SORTINGROLE);
|
|
m_proxyModel->setSortCaseSensitivity(Qt::CaseInsensitive);
|
|
|
|
m_view->sortByColumn(m_viewSortColumn, m_viewSortOrder);
|
|
|
|
m_view->setItemDelegateForColumn(Model::Views::IDCOLUMN, new Settings::View::Delegate::SingleText(this));
|
|
m_view->setItemDelegateForColumn(Model::Views::NAMECOLUMN, new Settings::View::Delegate::NameDelegate(this));
|
|
m_view->setItemDelegateForColumn(Model::Views::SCREENCOLUMN, new Settings::View::Delegate::SingleOption(this));
|
|
m_view->setItemDelegateForColumn(Model::Views::EDGECOLUMN, new Settings::View::Delegate::SingleOption(this));
|
|
m_view->setItemDelegateForColumn(Model::Views::ALIGNMENTCOLUMN, new Settings::View::Delegate::SingleOption(this));
|
|
m_view->setItemDelegateForColumn(Model::Views::SUBCONTAINMENTSCOLUMN, new Settings::View::Delegate::SingleText(this));
|
|
|
|
applyColumnWidths();
|
|
|
|
m_cutAction = new QAction(QIcon::fromTheme("edit-cut"), i18n("Cut"), m_view);
|
|
m_cutAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_X));
|
|
|
|
m_copyAction = new QAction(QIcon::fromTheme("edit-copy"), i18n("Copy"), m_view);
|
|
m_copyAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_C));
|
|
connect(m_copyAction, &QAction::triggered, this, &Views::copySelectedViews);
|
|
|
|
m_pasteAction = new QAction(QIcon::fromTheme("edit-paste"), i18n("Paste"), m_view);
|
|
m_pasteAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_V));
|
|
|
|
m_duplicateAction = new QAction(QIcon::fromTheme("edit-copy"), i18n("Duplicate Here"), m_view);
|
|
m_duplicateAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_D));
|
|
connect(m_duplicateAction, &QAction::triggered, this, &Views::duplicateSelectedViews);
|
|
|
|
m_view->addAction(m_cutAction);
|
|
m_view->addAction(m_copyAction);
|
|
m_view->addAction(m_pasteAction);
|
|
m_view->addAction(m_duplicateAction);
|
|
|
|
connect(m_view, &View::ViewsTableView::selectionsChanged, this, &Views::onSelectionsChanged);
|
|
connect(m_view, &QObject::destroyed, this, &Views::storeColumnWidths);
|
|
|
|
connect(m_view->horizontalHeader(), &QObject::destroyed, this, [&]() {
|
|
m_viewSortColumn = m_view->horizontalHeader()->sortIndicatorSection();
|
|
m_viewSortOrder = m_view->horizontalHeader()->sortIndicatorOrder();
|
|
});
|
|
}
|
|
|
|
void Views::reset()
|
|
{
|
|
m_model->resetData();
|
|
|
|
//! Clear any templates keeper data in order to produce reupdates if needed
|
|
m_handler->layoutsController()->templatesKeeper()->clear();
|
|
}
|
|
|
|
bool Views::hasChangedData() const
|
|
{
|
|
return m_model->hasChangedData();
|
|
}
|
|
|
|
bool Views::hasSelectedView() const
|
|
{
|
|
return m_view->selectionModel()->hasSelection();
|
|
}
|
|
|
|
int Views::rowForId(QString id) const
|
|
{
|
|
for (int i = 0; i < m_proxyModel->rowCount(); ++i) {
|
|
QString rowId = m_proxyModel->data(m_proxyModel->index(i, Model::Views::IDCOLUMN), Qt::UserRole).toString();
|
|
|
|
if (rowId == id) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
const Data::ViewsTable Views::selectedViewsCurrentData() const
|
|
{
|
|
Data::ViewsTable selectedviews;
|
|
|
|
if (!hasSelectedView()) {
|
|
return selectedviews;
|
|
}
|
|
|
|
QModelIndexList layoutidindexes = m_view->selectionModel()->selectedRows(Model::Views::IDCOLUMN);
|
|
|
|
for(int i=0; i<layoutidindexes.count(); ++i) {
|
|
QString selectedid = layoutidindexes[i].data(Qt::UserRole).toString();
|
|
selectedviews << m_model->currentData(selectedid);
|
|
}
|
|
|
|
return selectedviews;
|
|
}
|
|
|
|
const Latte::Data::View Views::appendViewFromViewTemplate(const Data::View &view)
|
|
{
|
|
Data::View newview = view;
|
|
newview.name = uniqueViewName(view.name);
|
|
m_model->appendTemporaryView(newview);
|
|
return newview;
|
|
}
|
|
|
|
void Views::copySelectedViews()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
|
|
if (!hasSelectedView()) {
|
|
return;
|
|
}
|
|
|
|
Data::ViewsTable selectedviews = selectedViewsCurrentData();
|
|
Latte::Data::Layout currentlayout = m_handler->currentData();
|
|
|
|
Data::ViewsTable clipboardviews;
|
|
|
|
for(int i=0; i<selectedviews.rowCount(); ++i) {
|
|
if (selectedviews[i].state() == Data::View::IsCreated) {
|
|
QString storedviewpath = m_handler->layoutsController()->templatesKeeper()->storedView(currentlayout.id, selectedviews[i].id);
|
|
Latte::Data::View copiedview = selectedviews[i];
|
|
copiedview.setState(Data::View::OriginFromLayout, storedviewpath, currentlayout.id, selectedviews[i].id);
|
|
copiedview.isActive = false;
|
|
clipboardviews << copiedview;
|
|
} else if (selectedviews[i].state() == Data::View::OriginFromViewTemplate
|
|
|| selectedviews[i].state() == Data::View::OriginFromLayout) {
|
|
Latte::Data::View copiedview = selectedviews[i];
|
|
copiedview.isActive = false;
|
|
clipboardviews << copiedview;
|
|
}
|
|
}
|
|
|
|
m_handler->layoutsController()->templatesKeeper()->setClipboardContents(clipboardviews);
|
|
}
|
|
|
|
void Views::duplicateSelectedViews()
|
|
{
|
|
qDebug() << Q_FUNC_INFO;
|
|
|
|
if (!hasSelectedView()) {
|
|
return;
|
|
}
|
|
|
|
Data::ViewsTable selectedviews = selectedViewsCurrentData();
|
|
Latte::Data::Layout currentlayout = m_handler->currentData();
|
|
|
|
for(int i=0; i<selectedviews.rowCount(); ++i) {
|
|
if (selectedviews[i].state() == Data::View::IsCreated) {
|
|
QString storedviewpath = m_handler->layoutsController()->templatesKeeper()->storedView(currentlayout.id, selectedviews[i].id);
|
|
Latte::Data::View duplicatedview = selectedviews[i];
|
|
duplicatedview.setState(Data::View::OriginFromLayout, storedviewpath, currentlayout.id, selectedviews[i].id);
|
|
duplicatedview.isActive = false;
|
|
appendViewFromViewTemplate(duplicatedview);
|
|
} else if (selectedviews[i].state() == Data::View::OriginFromViewTemplate
|
|
|| selectedviews[i].state() == Data::View::OriginFromLayout) {
|
|
Latte::Data::View duplicatedview = selectedviews[i];
|
|
duplicatedview.isActive = false;
|
|
appendViewFromViewTemplate(duplicatedview);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Views::removeSelectedViews()
|
|
{
|
|
if (!hasSelectedView()) {
|
|
return;
|
|
}
|
|
|
|
Data::ViewsTable selectedviews = selectedViewsCurrentData();;
|
|
|
|
int selectionheadrow = m_model->rowForId(selectedviews[0].id);
|
|
|
|
for (int i=0; i<selectedviews.rowCount(); ++i) {
|
|
m_model->removeView(selectedviews[i].id);
|
|
}
|
|
|
|
m_view->selectRow(qMax(0, selectionheadrow-1));
|
|
}
|
|
|
|
void Views::selectRow(const QString &id)
|
|
{
|
|
m_view->selectRow(rowForId(id));
|
|
}
|
|
|
|
void Views::onCurrentLayoutChanged()
|
|
{
|
|
Data::Layout layout = m_handler->currentData();
|
|
m_model->setOriginalData(layout.views);
|
|
}
|
|
|
|
void Views::onSelectionsChanged()
|
|
{
|
|
bool hasselectedview = hasSelectedView();
|
|
|
|
m_cutAction->setVisible(hasselectedview);
|
|
m_copyAction->setVisible(hasselectedview);
|
|
m_duplicateAction->setVisible(hasselectedview);
|
|
m_pasteAction->setVisible(!hasselectedview);
|
|
}
|
|
|
|
int Views::viewsForRemovalCount() const
|
|
{
|
|
if (!hasChangedData()) {
|
|
return 0;
|
|
}
|
|
|
|
Latte::Data::ViewsTable originalViews = m_model->originalViewsData();
|
|
Latte::Data::ViewsTable currentViews = m_model->currentViewsData();
|
|
Latte::Data::ViewsTable removedViews = originalViews.subtracted(currentViews);
|
|
|
|
return removedViews.rowCount();
|
|
}
|
|
|
|
void Views::save()
|
|
{
|
|
//! when this function is called we consider that removal has already been approved
|
|
|
|
Latte::Data::Layout originallayout = m_handler->originalData();
|
|
Latte::Data::Layout currentlayout = m_handler->currentData();
|
|
Latte::CentralLayout *centralActive = m_handler->isSelectedLayoutOriginal() ? m_handler->corona()->layoutsManager()->synchronizer()->centralLayout(originallayout.name) : nullptr;
|
|
Latte::CentralLayout *central = centralActive ? centralActive : new Latte::CentralLayout(this, currentlayout.id);
|
|
|
|
//! views in model
|
|
Latte::Data::ViewsTable originalViews = m_model->originalViewsData();
|
|
Latte::Data::ViewsTable currentViews = m_model->currentViewsData();
|
|
Latte::Data::ViewsTable alteredViews = m_model->alteredViews();
|
|
Latte::Data::ViewsTable newViews = m_model->newViews();
|
|
|
|
QHash<QString, Data::View> newviewsresponses;
|
|
|
|
//! add new views
|
|
for(int i=0; i<newViews.rowCount(); ++i){
|
|
if (newViews[i].state() == Data::View::OriginFromViewTemplate) {
|
|
Data::View addedview = central->newView(newViews[i]);
|
|
|
|
newviewsresponses[newViews[i].id] = addedview;
|
|
} else if (newViews[i].state() == Data::View::OriginFromLayout) {
|
|
Data::View adjustedview = newViews[i];
|
|
adjustedview.setState(Data::View::OriginFromViewTemplate, newViews[i].originFile(), QString(), QString());
|
|
Data::View addedview = central->newView(adjustedview);
|
|
|
|
newviewsresponses[newViews[i].id] = addedview;
|
|
}
|
|
}
|
|
|
|
//! update altered views
|
|
for (int i=0; i<alteredViews.rowCount(); ++i) {
|
|
if (alteredViews[i].state() == Data::View::IsCreated) {
|
|
qDebug() << "org.kde.latte updating altered view :: " << alteredViews[i];
|
|
central->updateView(alteredViews[i]);
|
|
}
|
|
}
|
|
|
|
//! remove deprecated views
|
|
Latte::Data::ViewsTable removedViews = originalViews.subtracted(currentViews);
|
|
|
|
for (int i=0; i<removedViews.rowCount(); ++i) {
|
|
central->removeView(removedViews[i]);
|
|
}
|
|
|
|
if ((removedViews.rowCount() > 0) || (newViews.rowCount() > 0)) {
|
|
m_handler->corona()->layoutsManager()->synchronizer()->syncActiveLayoutsToOriginalFiles();
|
|
}
|
|
|
|
//! update model for newly added views
|
|
for (const auto vid: newviewsresponses.keys()) {
|
|
m_model->setOriginalView(vid, newviewsresponses[vid]);
|
|
}
|
|
|
|
//! update all table with latest data
|
|
currentViews = m_model->currentViewsData();
|
|
|
|
//! update model original data
|
|
m_model->setOriginalData(currentViews);
|
|
|
|
if (central->isActive()) {
|
|
m_model->updateActiveStatesBasedOn(central);
|
|
}
|
|
|
|
//! Clear any templates keeper data in order to produce reupdates if needed
|
|
m_handler->layoutsController()->templatesKeeper()->clear();
|
|
}
|
|
|
|
QString Views::uniqueViewName(QString name)
|
|
{
|
|
if (name.isEmpty()) {
|
|
return name;
|
|
}
|
|
|
|
int pos_ = name.lastIndexOf(QRegExp(QString(" - [0-9]+")));
|
|
|
|
if (m_model->containsCurrentName(name) && pos_ > 0) {
|
|
name = name.left(pos_);
|
|
}
|
|
|
|
int i = 2;
|
|
|
|
QString namePart = name;
|
|
|
|
while (m_model->containsCurrentName(name)) {
|
|
name = namePart + " - " + QString::number(i);
|
|
i++;
|
|
}
|
|
|
|
return name;
|
|
}
|
|
|
|
void Views::applyColumnWidths()
|
|
{
|
|
m_view->horizontalHeader()->setSectionResizeMode(Model::Views::NAMECOLUMN, QHeaderView::Stretch);
|
|
|
|
if (m_viewColumnWidths.count()<(Model::Views::columnCount()-1)) {
|
|
return;
|
|
}
|
|
|
|
m_view->setColumnWidth(Model::Views::IDCOLUMN, m_viewColumnWidths[0].toInt());
|
|
m_view->setColumnWidth(Model::Views::SCREENCOLUMN, m_viewColumnWidths[1].toInt());
|
|
m_view->setColumnWidth(Model::Views::EDGECOLUMN, m_viewColumnWidths[2].toInt());
|
|
m_view->setColumnWidth(Model::Views::ALIGNMENTCOLUMN, m_viewColumnWidths[3].toInt());
|
|
m_view->setColumnWidth(Model::Views::SUBCONTAINMENTSCOLUMN, m_viewColumnWidths[4].toInt());
|
|
}
|
|
|
|
void Views::storeColumnWidths()
|
|
{
|
|
if (m_viewColumnWidths.isEmpty() || (m_viewColumnWidths.count()<Model::Views::columnCount()-1)) {
|
|
m_viewColumnWidths.clear();
|
|
for (int i=0; i<Model::Views::columnCount(); ++i) {
|
|
m_viewColumnWidths << "";
|
|
}
|
|
}
|
|
|
|
m_viewColumnWidths[0] = QString::number(m_view->columnWidth(Model::Views::IDCOLUMN));
|
|
m_viewColumnWidths[1] = QString::number(m_view->columnWidth(Model::Views::SCREENCOLUMN));
|
|
m_viewColumnWidths[2] = QString::number(m_view->columnWidth(Model::Views::EDGECOLUMN));
|
|
m_viewColumnWidths[3] = QString::number(m_view->columnWidth(Model::Views::ALIGNMENTCOLUMN));
|
|
m_viewColumnWidths[4] = QString::number(m_view->columnWidth(Model::Views::SUBCONTAINMENTSCOLUMN));
|
|
}
|
|
|
|
void Views::loadConfig()
|
|
{
|
|
m_viewColumnWidths = m_storage.readEntry("columnWidths", QStringList());
|
|
m_viewSortColumn = m_storage.readEntry("sortColumn", (int)Model::Views::SCREENCOLUMN);
|
|
m_viewSortOrder = static_cast<Qt::SortOrder>(m_storage.readEntry("sortOrder", (int)Qt::AscendingOrder));
|
|
}
|
|
|
|
void Views::saveConfig()
|
|
{
|
|
m_storage.writeEntry("columnWidths", m_viewColumnWidths);
|
|
m_storage.writeEntry("sortColumn", m_viewSortColumn);
|
|
m_storage.writeEntry("sortOrder", (int)m_viewSortOrder);
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|