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.
315 lines
10 KiB
C++
315 lines
10 KiB
C++
/*
|
|
* Copyright 2019 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 "shareddelegate.h"
|
|
|
|
// local
|
|
#include "persistentmenu.h"
|
|
#include "../data/layoutdata.h"
|
|
#include "../models/layoutsmodel.h"
|
|
#include "../tools/settingstools.h"
|
|
|
|
// Qt
|
|
#include <QAction>
|
|
#include <QApplication>
|
|
#include <QComboBox>
|
|
#include <QDebug>
|
|
#include <QMenu>
|
|
#include <QWidget>
|
|
#include <QModelIndex>
|
|
#include <QPainter>
|
|
#include <QPushButton>
|
|
#include <QString>
|
|
#include <QTextDocument>
|
|
|
|
|
|
namespace Latte {
|
|
namespace Settings {
|
|
namespace Layout {
|
|
namespace Delegate {
|
|
|
|
|
|
Shared::Shared(Controller::Layouts *parent)
|
|
: QStyledItemDelegate(parent),
|
|
m_controller(parent)
|
|
{
|
|
}
|
|
|
|
QWidget *Shared::createEditor(QWidget *parent, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
{
|
|
bool inMultiple = index.data(Model::Layouts::INMULTIPLELAYOUTSROLE).toBool();
|
|
|
|
QString layoutId = index.data(Model::Layouts::IDROLE).toString();
|
|
Data::LayoutsTable allLayouts = qvariant_cast<Data::LayoutsTable>(index.data(Model::Layouts::ALLLAYOUTSROLE));
|
|
QStringList assignedShares = index.data(Qt::UserRole).toStringList();
|
|
|
|
QPushButton *button = new QPushButton(parent);
|
|
PersistentMenu *menu = new PersistentMenu(button);
|
|
button->setMenu(menu);
|
|
|
|
menu->setMinimumWidth(option.rect.width());
|
|
|
|
for (int i = 0; i < allLayouts.rowCount(); ++i) {
|
|
if ((inMultiple && allLayouts[i].isShared()) || (allLayouts[i].id == layoutId)) {
|
|
continue;
|
|
}
|
|
|
|
QAction *action = new QAction(allLayouts[i].name);
|
|
action->setData(allLayouts[i].id);
|
|
action->setCheckable(true);
|
|
action->setChecked(assignedShares.contains(allLayouts[i].id));
|
|
|
|
if (allLayouts[i].isActive) {
|
|
QFont font = action->font();
|
|
font.setBold(true);
|
|
action->setFont(font);
|
|
}
|
|
|
|
menu->addAction(action);
|
|
|
|
connect(action, &QAction::toggled, this, [this, button, index]() {
|
|
updateButtonText(button, index);
|
|
});
|
|
}
|
|
|
|
updateButtonText(button, index);
|
|
|
|
m_controller->on_sharedToInEditChanged(layoutId, true);
|
|
|
|
return button;
|
|
}
|
|
|
|
void Shared::setEditorData(QWidget *editor, const QModelIndex &index) const
|
|
{
|
|
Data::LayoutsTable allLayouts = qvariant_cast<Data::LayoutsTable>(index.data(Model::Layouts::ALLLAYOUTSROLE));
|
|
updateButtonText(editor, index);
|
|
}
|
|
|
|
void Shared::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
|
|
{
|
|
QPushButton *button = static_cast<QPushButton *>(editor);
|
|
|
|
QStringList assignedLayouts;
|
|
foreach (QAction *action, button->menu()->actions()) {
|
|
if (action->isChecked()) {
|
|
assignedLayouts << action->data().toString();
|
|
}
|
|
}
|
|
|
|
model->setData(index, assignedLayouts, Qt::UserRole);
|
|
|
|
QString layoutId = index.data(Model::Layouts::IDROLE).toString();
|
|
m_controller->on_sharedToInEditChanged(layoutId, false);
|
|
}
|
|
|
|
void Shared::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
{
|
|
Q_UNUSED(index)
|
|
|
|
editor->setGeometry(option.rect);
|
|
}
|
|
|
|
void Shared::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
{
|
|
bool sharedInEdit = index.data(Model::Layouts::SHAREDTOINEDITROLE).toBool();
|
|
Data::LayoutsTable allLayouts = qvariant_cast<Data::LayoutsTable>(index.data(Model::Layouts::ALLLAYOUTSROLE));
|
|
QStringList assignedIds = index.data(Qt::UserRole).toStringList();
|
|
QStringList originalIds = index.data(Model::Layouts::ORIGINALSHARESROLE).toStringList();
|
|
|
|
Data::LayoutsTable assignedLayouts;
|
|
|
|
for (const auto &id : assignedIds) {
|
|
if (allLayouts.containsId(id)) {
|
|
assignedLayouts << allLayouts[id];
|
|
}
|
|
}
|
|
|
|
QStyleOptionViewItem myOptions = option;
|
|
//! Remove the focus dotted lines
|
|
myOptions.state = (myOptions.state & ~QStyle::State_HasFocus);
|
|
painter->save();
|
|
|
|
if (assignedLayouts.rowCount() > 0 && !sharedInEdit) {
|
|
//! indicator
|
|
if (!sharedInEdit) {
|
|
paintSharedToIndicator(painter, myOptions, index);
|
|
}
|
|
|
|
//! Text code
|
|
myOptions.text = joined(assignedLayouts, originalIds);
|
|
|
|
int thick = option.rect.height();
|
|
|
|
if (qApp->layoutDirection() == Qt::LeftToRight) {
|
|
myOptions.rect = QRect(myOptions.rect.x() + thick, myOptions.rect.y(), myOptions.rect.width() - thick, myOptions.rect.height());
|
|
} else {
|
|
myOptions.rect = QRect(myOptions.rect.x(), myOptions.rect.y(), myOptions.rect.width() - thick, myOptions.rect.height());
|
|
}
|
|
|
|
QTextDocument doc;
|
|
QString css;
|
|
QString sharesText = myOptions.text;
|
|
|
|
QPalette::ColorRole applyColor = Latte::isSelected(option) ? QPalette::HighlightedText : QPalette::Text;
|
|
QBrush nBrush = option.palette.brush(Latte::colorGroup(option), applyColor);
|
|
|
|
css = QString("body { color : %1; }").arg(nBrush.color().name());
|
|
|
|
doc.setDefaultStyleSheet(css);
|
|
doc.setHtml("<body>" + myOptions.text + "</body>");
|
|
|
|
myOptions.text = "";
|
|
myOptions.widget->style()->drawControl(QStyle::CE_ItemViewItem, &myOptions, painter);
|
|
|
|
//we need an offset to be in the same vertical center of TextEdit
|
|
int offsetY = ((myOptions.rect.height() - doc.size().height()) / 2);
|
|
|
|
if ((qApp->layoutDirection() == Qt::RightToLeft) && !sharesText.isEmpty()) {
|
|
int textWidth = doc.size().width();
|
|
|
|
painter->translate(qMax(myOptions.rect.left(), myOptions.rect.right() - textWidth), myOptions.rect.top() + offsetY + 1);
|
|
} else {
|
|
painter->translate(myOptions.rect.left(), myOptions.rect.top() + offsetY + 1);
|
|
}
|
|
|
|
QRect clip(0, 0, myOptions.rect.width(), myOptions.rect.height());
|
|
doc.drawContents(painter, clip);
|
|
} else {
|
|
QApplication::style()->drawControl(QStyle::CE_ItemViewItem, &myOptions, painter);
|
|
}
|
|
|
|
painter->restore();
|
|
}
|
|
|
|
void Shared::paintSharedToIndicator(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
|
|
{
|
|
// Disabled
|
|
bool isSelected{Latte::isSelected(option)};
|
|
QPalette::ColorRole backColorRole = isSelected ? QPalette::Highlight : QPalette::Base;
|
|
QPalette::ColorRole textColorRole = isSelected ? QPalette::HighlightedText : QPalette::Text;
|
|
|
|
int space = option.rect.height() / 2;
|
|
|
|
//! draw background below icons
|
|
//! HIDDENTEXTCOLUMN is just needed to draw empty background rectangles properly based on states
|
|
QStyleOptionViewItem backOptions = option;
|
|
if (qApp->layoutDirection() == Qt::LeftToRight) {
|
|
backOptions.rect = QRect(option.rect.x(), option.rect.y(), 2 * space, 2 * space);
|
|
} else {
|
|
backOptions.rect = QRect(option.rect.x() + option.rect.width() - (2*space), option.rect.y(), 2 * space, 2 * space);
|
|
}
|
|
|
|
QStyledItemDelegate::paint(painter, backOptions, index.model()->index(index.row(), Model::Layouts::HIDDENTEXTCOLUMN));
|
|
|
|
// text
|
|
QPen pen(Qt::DotLine);
|
|
QColor textColor = option.palette.brush(Latte::colorGroup(option), textColorRole).color();
|
|
|
|
pen.setWidth(2); pen.setColor(textColor);
|
|
int y = option.rect.y()+option.rect.height()/2;
|
|
|
|
painter->setPen(pen);
|
|
|
|
if (qApp->layoutDirection() == Qt::LeftToRight) {
|
|
int xStart = option.rect.x();
|
|
painter->drawLine(xStart, y, xStart + space, y);
|
|
|
|
int xm = option.rect.x() + space;
|
|
int thick = option.rect.height() / 2;
|
|
int ym = option.rect.y() + ((option.rect.height() - thick) / 2);
|
|
|
|
pen.setStyle(Qt::SolidLine);
|
|
painter->setPen(pen);
|
|
painter->setBrush(textColor);
|
|
|
|
//! draw ending cirlce
|
|
painter->drawEllipse(QPoint(xm, ym + thick/2), thick/4, thick/4);
|
|
} else {
|
|
int xEnd = option.rect.x() + option.rect.width();
|
|
painter->drawLine(xEnd, y, xEnd-space, y);
|
|
|
|
int xm = option.rect.x() + option.rect.width() - space;
|
|
int thick = option.rect.height() / 2;
|
|
int ym = option.rect.y() + ((option.rect.height() - thick) / 2);
|
|
|
|
pen.setStyle(Qt::SolidLine);
|
|
painter->setPen(pen);
|
|
painter->setBrush(textColor);
|
|
|
|
//! draw ending cirlce
|
|
painter->drawEllipse(QPoint(xm, ym + thick/2), thick/4, thick/4);
|
|
}
|
|
}
|
|
|
|
void Shared::updateButtonText(QWidget *editor, const QModelIndex &index) const
|
|
{
|
|
if (!editor) {
|
|
return;
|
|
}
|
|
|
|
Data::LayoutsTable allLayouts = qvariant_cast<Data::LayoutsTable>(index.data(Model::Layouts::ALLLAYOUTSROLE));
|
|
|
|
QPushButton *button = static_cast<QPushButton *>(editor);
|
|
Data::LayoutsTable assignedLayouts;
|
|
|
|
foreach (QAction *action, button->menu()->actions()) {
|
|
if (action->isChecked()) {
|
|
QString id = action->data().toString();
|
|
if (allLayouts.containsId(id)) {
|
|
assignedLayouts << allLayouts[action->data().toString()];
|
|
}
|
|
}
|
|
}
|
|
|
|
button->setText(joined(assignedLayouts, QStringList(), false));
|
|
}
|
|
|
|
QString Shared::joined(const Data::LayoutsTable &layouts, const QStringList &originalIds, bool formatText) const
|
|
{
|
|
QString finalText;
|
|
|
|
for (int i = 0; i < layouts.rowCount(); ++i) {
|
|
if (i > 0) {
|
|
finalText += ", ";
|
|
}
|
|
|
|
bool bold = layouts[i].isActive;
|
|
bool italic = !originalIds.contains(layouts[i].id);
|
|
|
|
QString name = layouts[i].name;
|
|
|
|
if (bold && formatText) {
|
|
name = "<b>" + name + "</b>";
|
|
}
|
|
|
|
if (italic && formatText) {
|
|
name = "<i>" + name + "</i>";
|
|
}
|
|
|
|
finalText += name;
|
|
}
|
|
|
|
return finalText;
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|