kwin/kcmkwin/kwintabbox/layoutpreview.cpp
2020-08-07 19:57:56 +00:00

255 lines
6.6 KiB
C++

/*
KWin - the KDE window manager
This file is part of the KDE project.
SPDX-FileCopyrightText: 2009, 2011 Martin Gräßlin <mgraesslin@kde.org>
SPDX-License-Identifier: GPL-2.0-or-later
*/
// own
#include "layoutpreview.h"
#include "thumbnailitem.h"
#include <QApplication>
#include <QDebug>
#include <QDesktopWidget>
#include <QQmlEngine>
#include <QQmlContext>
#include <QScreen>
#include <QStandardPaths>
#include <KConfigGroup>
#include <KDesktopFile>
#include <KLocalizedString>
#include <KMimeTypeTrader>
namespace KWin
{
namespace TabBox
{
LayoutPreview::LayoutPreview(const QString &path, QObject *parent)
: QObject(parent)
, m_item(nullptr)
{
QQmlEngine *engine = new QQmlEngine(this);
QQmlComponent *component = new QQmlComponent(engine, this);
qmlRegisterType<WindowThumbnailItem>("org.kde.kwin", 2, 0, "ThumbnailItem");
qmlRegisterType<SwitcherItem>("org.kde.kwin", 2, 0, "Switcher");
qmlRegisterType<QAbstractItemModel>();
component->loadUrl(QUrl::fromLocalFile(path));
if (component->isError()) {
qDebug() << component->errorString();
}
QObject *item = component->create();
auto findSwitcher = [item]() -> SwitcherItem* {
if (!item) {
return nullptr;
}
if (SwitcherItem *i = qobject_cast<SwitcherItem*>(item)) {
return i;
} else if (QQuickWindow *w = qobject_cast<QQuickWindow*>(item)) {
return w->contentItem()->findChild<SwitcherItem*>();
}
return item->findChild<SwitcherItem*>();
};
if (SwitcherItem *switcher = findSwitcher()) {
m_item = switcher;
switcher->setVisible(true);
}
auto findWindow = [item]() -> QQuickWindow* {
if (!item) {
return nullptr;
}
if (QQuickWindow *w = qobject_cast<QQuickWindow*>(item)) {
return w;
}
return item->findChild<QQuickWindow*>();
};
if (QQuickWindow *w = findWindow()) {
w->setKeyboardGrabEnabled(true);
w->setMouseGrabEnabled(true);
w->installEventFilter(this);
}
}
LayoutPreview::~LayoutPreview()
{
}
bool LayoutPreview::eventFilter(QObject *object, QEvent *event)
{
if (event->type() == QEvent::KeyPress) {
QKeyEvent *keyEvent = static_cast<QKeyEvent*>(event);
if (keyEvent->key() == Qt::Key_Escape ||
keyEvent->key() == Qt::Key_Return ||
keyEvent->key() == Qt::Key_Enter ||
keyEvent->key() == Qt::Key_Space) {
object->deleteLater();
deleteLater();
}
if (m_item && keyEvent->key() == Qt::Key_Tab) {
m_item->incrementIndex();
}
if (m_item && keyEvent->key() == Qt::Key_Backtab) {
m_item->decrementIndex();
}
} else if (event->type() == QEvent::MouseButtonPress) {
if (QWindow *w = qobject_cast<QWindow*>(object)) {
if (!w->geometry().contains(static_cast<QMouseEvent*>(event)->globalPos())) {
object->deleteLater();
deleteLater();
}
}
}
return QObject::eventFilter(object, event);
}
ExampleClientModel::ExampleClientModel (QObject* parent)
: QAbstractListModel (parent)
{
init();
}
ExampleClientModel::~ExampleClientModel()
{
}
void ExampleClientModel::init()
{
if (const auto s = KMimeTypeTrader::self()->preferredService(QStringLiteral("inode/directory"))) {
m_services << s;
m_fileManager = s;
}
if (const auto s = KMimeTypeTrader::self()->preferredService(QStringLiteral("text/html"))) {
m_services << s;
m_browser = s;
}
if (const auto s = KMimeTypeTrader::self()->preferredService(QStringLiteral("message/rfc822"))) {
m_services << s;
m_email = s;
}
if (const auto s = KService::serviceByDesktopName(QStringLiteral("kdesystemsettings"))) {
m_services << s;
m_systemSettings = s;
}
}
QVariant ExampleClientModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid()) {
return QVariant();
}
switch (role) {
case Qt::DisplayRole:
case CaptionRole:
return m_services.at(index.row())->name();
case MinimizedRole:
return false;
case DesktopNameRole:
return i18nc("An example Desktop Name", "Desktop 1");
case IconRole:
return m_services.at(index.row())->icon();
case WindowIdRole:
const auto s = m_services.at(index.row());
if (s == m_browser) {
return WindowThumbnailItem::Konqueror;
} else if (s == m_email) {
return WindowThumbnailItem::KMail;
} else if (s == m_systemSettings) {
return WindowThumbnailItem::Systemsettings;
} else if (s == m_fileManager) {
return WindowThumbnailItem::Dolphin;
}
return 0;
}
return QVariant();
}
QString ExampleClientModel::longestCaption() const
{
QString caption;
for (const auto item : m_services) {
const QString name = item->name();
if (name.size() > caption.size()) {
caption = name;
}
}
return caption;
}
int ExampleClientModel::rowCount(const QModelIndex &parent) const
{
Q_UNUSED(parent)
return m_services.size();
}
QHash<int, QByteArray> ExampleClientModel::roleNames() const
{
return {
{ CaptionRole, QByteArrayLiteral("caption") },
{ MinimizedRole, QByteArrayLiteral("minimized") },
{ DesktopNameRole, QByteArrayLiteral("desktopName") },
{ IconRole, QByteArrayLiteral("icon") },
{ WindowIdRole, QByteArrayLiteral("windowId") },
};
}
SwitcherItem::SwitcherItem(QObject *parent)
: QObject(parent)
, m_model(new ExampleClientModel(this))
, m_item(nullptr)
, m_currentIndex(0)
, m_visible(false)
{
}
SwitcherItem::~SwitcherItem()
{
}
void SwitcherItem::setVisible(bool visible)
{
if (m_visible == visible) {
return;
}
m_visible = visible;
emit visibleChanged();
}
void SwitcherItem::setItem(QObject *item)
{
m_item = item;
emit itemChanged();
}
void SwitcherItem::setCurrentIndex(int index)
{
if (m_currentIndex == index) {
return;
}
m_currentIndex = index;
emit currentIndexChanged(m_currentIndex);
}
QRect SwitcherItem::screenGeometry() const
{
const QScreen *primaryScreen = qApp->primaryScreen();
return primaryScreen->geometry();
}
void SwitcherItem::incrementIndex()
{
setCurrentIndex((m_currentIndex + 1) % m_model->rowCount());
}
void SwitcherItem::decrementIndex()
{
int index = m_currentIndex -1;
if (index < 0) {
index = m_model->rowCount() -1;
}
setCurrentIndex(index);
}
} // namespace KWin
} // namespace TabBox