/* This is the new kwindecoration kcontrol module Copyright (c) 2001 Karol Szwed http://gallium.n3.net/ Copyright 2009, 2010 Martin Gräßlin Supports new kwin configuration plugins, and titlebar button position modification via dnd interface. Based on original "kwintheme" (Window Borders) Copyright (C) 2001 Rik Hemsley (rikkus) 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 2 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, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ // own #include "kwindecoration.h" #include "buttonsconfigdialog.h" #include "configdialog.h" #include "decorationmodel.h" #include "auroraetheme.h" #include "preview.h" // Qt #include #include #include #include #include #include #include #include // KDE #include #include #include #include #include #include #include // KCModule plugin interface // ========================= K_PLUGIN_FACTORY(KWinDecoFactory, registerPlugin(); ) namespace KWin { static const QString s_defaultPluginName = QStringLiteral("Oxygen"); KWinDecorationForm::KWinDecorationForm(QWidget* parent) : QWidget(parent) { setupUi(this); } KWinDecorationModule::KWinDecorationModule(QWidget* parent, const QVariantList &) : KCModule(parent) , kwinConfig(KSharedConfig::openConfig("kwinrc")) , m_showTooltips(false) , m_model(nullptr) , m_proxyModel(nullptr) , m_configLoaded(false) , m_decorationButtons(new DecorationButtons(this)) , m_listView(new QQuickView()) { qmlRegisterType("org.kde.kwin.aurorae", 0, 1, "AuroraeTheme"); qmlRegisterType("org.kde.kwin.kcmdecoration", 0, 1, "PreviewItem"); m_ui = new KWinDecorationForm(this); m_ui->configureDecorationButton->setIcon(QIcon::fromTheme("configure")); m_ui->configureButtonsButton->setIcon(QIcon::fromTheme("configure")); m_ui->ghnsButton->setIcon(QIcon::fromTheme("get-hot-new-stuff")); QWidget *container = QWidget::createWindowContainer(m_listView.data(), m_ui->decorationList->viewport()); QVBoxLayout *containerLayout = new QVBoxLayout(m_ui->decorationList->viewport()); containerLayout->addWidget(container); QVBoxLayout* layout = new QVBoxLayout(this); layout->addWidget(m_ui); KAboutData *about = new KAboutData(QStringLiteral("kcmkwindecoration"), i18n("Window Decoration Control Module"), QString(), QString(), KAboutData::License_GPL, i18n("(c) 2001 Karol Szwed")); about->addAuthor(i18n("Karol Szwed"), QString(), "gallium@kde.org"); setAboutData(about); } KWinDecorationModule::~KWinDecorationModule() { } void KWinDecorationModule::showEvent(QShowEvent *ev) { KCModule::showEvent(ev); init(); } void KWinDecorationModule::init() { if (m_model) { // init already called return; } const QString mainQmlPath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, "kwin/kcm_kwindecoration/main.qml"); if (mainQmlPath.isNull()) { KMessageBox::error(this, i18n("

Installation error

" "The resource

kwin/kcm_kwindecoration/main.qml

could not be located in any application data path." "

Please contact your distribution

" "The application will now abort"), i18n("Installation Error")); abort(); } KConfigGroup style(kwinConfig, "Style"); // Set up the decoration lists and other UI settings m_model = new DecorationModel(kwinConfig, this); m_proxyModel = new QSortFilterProxyModel(this); m_proxyModel->setSourceModel(m_model); m_proxyModel->setFilterCaseSensitivity(Qt::CaseInsensitive); m_listView->setResizeMode(QQuickView::SizeRootObjectToView); m_listView->rootContext()->setContextProperty("decorationModel", m_proxyModel); m_listView->rootContext()->setContextProperty("decorationBaseModel", m_model); m_listView->rootContext()->setContextProperty("options", m_decorationButtons); m_listView->rootContext()->setContextProperty("highlightColor", m_ui->decorationList->palette().color(QPalette::Highlight)); m_listView->rootContext()->setContextProperty("auroraeSource", QStandardPaths::locate(QStandardPaths::GenericDataLocation, "kwin/aurorae/aurorae.qml")); m_listView->rootContext()->setContextProperty("decorationActiveCaptionColor", KDecoration::options()->color(ColorFont, true)); m_listView->rootContext()->setContextProperty("decorationInactiveCaptionColor", KDecoration::options()->color(ColorFont, false)); m_listView->rootContext()->setContextProperty("decorationActiveTitleBarColor", KDecoration::options()->color(ColorTitleBar, true)); m_listView->rootContext()->setContextProperty("decorationInactiveTitleBarColor", KDecoration::options()->color(ColorTitleBar, false)); m_listView->setSource(QUrl::fromLocalFile(mainQmlPath)); readConfig(style); connect(m_listView->rootObject(), SIGNAL(currentIndexChanged()), SLOT(slotSelectionChanged())); connect(m_ui->configureButtonsButton, SIGNAL(clicked(bool)), this, SLOT(slotConfigureButtons())); connect(m_ui->ghnsButton, SIGNAL(clicked(bool)), SLOT(slotGHNSClicked())); connect(m_ui->searchEdit, SIGNAL(textChanged(QString)), m_proxyModel, SLOT(setFilterFixedString(QString))); connect(m_ui->searchEdit, SIGNAL(textChanged(QString)), m_listView->rootObject(), SLOT(returnToBounds()), Qt::QueuedConnection); connect(m_ui->searchEdit, SIGNAL(textChanged(QString)), SLOT(updateScrollbarRange()), Qt::QueuedConnection); connect(m_ui->configureDecorationButton, SIGNAL(clicked(bool)), SLOT(slotConfigureDecoration())); m_ui->decorationList->disconnect(m_ui->decorationList->verticalScrollBar()); m_ui->decorationList->verticalScrollBar()->disconnect(m_ui->decorationList); connect(m_listView->rootObject(), SIGNAL(contentYChanged()), SLOT(updateScrollbarValue())); connect(m_listView->rootObject(), SIGNAL(contentHeightChanged()), SLOT(updateScrollbarRange())); connect(m_ui->decorationList->verticalScrollBar(), SIGNAL(rangeChanged(int,int)), SLOT(updateScrollbarRange())); connect(m_ui->decorationList->verticalScrollBar(), SIGNAL(valueChanged(int)), SLOT(updateViewPosition(int))); m_ui->decorationList->installEventFilter(this); m_ui->decorationList->viewport()->installEventFilter(this); QMetaObject::invokeMethod(this, "updatePreviews", Qt::QueuedConnection); updateScrollbarRange(); } // This is the selection handler setting void KWinDecorationModule::slotSelectionChanged() { emit KCModule::changed(true); } // Reads the kwin config settings, and sets all UI controls to those settings // Updating the config plugin if required void KWinDecorationModule::readConfig(const KConfigGroup & conf) { m_showTooltips = conf.readEntry("ShowToolTips", true); // Find the corresponding decoration name to that of // the current plugin library name QString libraryName = conf.readEntry("PluginLib", s_defaultPluginName); if (libraryName.isEmpty()) { // Selected decoration doesn't exist, use the default libraryName = s_defaultPluginName; } const int bsize = conf.readEntry("BorderSize", (int)BorderNormal); BorderSize borderSize = BorderNormal; if (bsize >= BorderTiny && bsize < BordersCount) borderSize = static_cast< BorderSize >(bsize); if (libraryName == "aurorae") { KConfig auroraeConfig("auroraerc"); KConfigGroup group(&auroraeConfig, "Engine"); const QString themeName = group.readEntry("ThemeName", "example-deco"); const QString type = group.readEntry("EngineType", "aurorae"); const QModelIndex index = m_proxyModel->mapFromSource(m_model->indexOfAuroraeName(themeName, type)); if (index.isValid()) { m_listView->rootObject()->setProperty("currentIndex", index.row()); } } else { const QModelIndex index = m_proxyModel->mapFromSource(m_model->indexOfLibrary(libraryName)); if (index.isValid()) { m_model->setBorderSize(index, borderSize); m_listView->rootObject()->setProperty("currentIndex", index.row()); } } // Buttons tab // ============ m_decorationButtons->setCustomPositions(conf.readEntry("CustomButtonPositions", false)); // Menu and onAllDesktops buttons are default on LHS m_decorationButtons->setLeftButtons(KDecorationOptions::readDecorationButtons(conf, "ButtonsOnLeft", KDecorationOptions::defaultTitleButtonsLeft())); // Help, Minimize, Maximize and Close are default on RHS m_decorationButtons->setRightButtons(KDecorationOptions::readDecorationButtons(conf, "ButtonsOnRight", KDecorationOptions::defaultTitleButtonsRight())); if (m_configLoaded) m_model->changeButtons(m_decorationButtons); else { m_configLoaded = true; m_model->setButtons(m_decorationButtons->customPositions(), m_decorationButtons->leftButtons(), m_decorationButtons->rightButtons()); } emit KCModule::changed(false); } // Writes the selected user configuration to the kwin config file void KWinDecorationModule::writeConfig(KConfigGroup & conf) { const QModelIndex index = m_proxyModel->mapToSource(m_proxyModel->index(m_listView->rootObject()->property("currentIndex").toInt(), 0)); const QString libName = m_model->data(index, DecorationModel::LibraryNameRole).toString(); // General settings conf.writeEntry("PluginLib", libName); conf.writeEntry("CustomButtonPositions", m_decorationButtons->customPositions()); conf.writeEntry("ShowToolTips", m_showTooltips); // Button settings KDecorationOptions::writeDecorationButtons(conf, "ButtonsOnLeft", m_decorationButtons->leftButtons()); KDecorationOptions::writeDecorationButtons(conf, "ButtonsOnRight", m_decorationButtons->rightButtons()); conf.writeEntry("BorderSize", static_cast(m_model->data(index, DecorationModel::BorderSizeRole).toInt())); if (m_model->data(index, DecorationModel::TypeRole).toInt() == DecorationModelData::AuroraeDecoration || m_model->data(index, DecorationModel::TypeRole).toInt() == DecorationModelData::QmlDecoration) { KConfig auroraeConfig("auroraerc"); KConfigGroup group(&auroraeConfig, "Engine"); group.writeEntry("ThemeName", m_model->data(index, DecorationModel::AuroraeNameRole).toString()); if (m_model->data(index, DecorationModel::TypeRole).toInt() == DecorationModelData::QmlDecoration) { group.writeEntry("EngineType", "qml"); } else { group.deleteEntry("EngineType"); } group.sync(); } // We saved, so tell kcmodule that there have been no new user changes made. emit KCModule::changed(false); } // Virutal functions required by KCModule void KWinDecorationModule::load() { const KConfigGroup config(kwinConfig, "Style"); // Reset by re-reading the config readConfig(config); } void KWinDecorationModule::save() { KConfigGroup config(kwinConfig, "Style"); writeConfig(config); config.sync(); // Send signal to all kwin instances QDBusMessage message = QDBusMessage::createSignal("/KWin", "org.kde.KWin", "reloadConfig"); QDBusConnection::sessionBus().send(message); } void KWinDecorationModule::defaults() { // Set the KDE defaults m_showTooltips = true; const QModelIndex index = m_proxyModel->mapFromSource(m_model->indexOfName(i18n("Oxygen"))); if (index.isValid()) m_listView->rootObject()->setProperty("currentIndex", index.row()); m_decorationButtons->resetToDefaults(); m_model->changeButtons(m_decorationButtons); emit changed(true); } QString KWinDecorationModule::quickHelp() const { return i18n("

Window Manager Decoration

" "

This module allows you to choose the window border decorations, " "as well as titlebar button positions and custom decoration options.

" "To choose a theme for your window decoration click on its name and apply your choice by clicking the \"Apply\" button below." " If you do not want to apply your choice you can click the \"Reset\" button to discard your changes." "

You can configure each theme. There are different options specific for each theme.

" "

On the \"Buttons\" tab check the \"Use custom titlebar button positions\" box " "and you can change the positions of the buttons to your liking.

"); } void KWinDecorationModule::slotConfigureButtons() { QPointer< KWinDecorationButtonsConfigDialog > configDialog = new KWinDecorationButtonsConfigDialog(m_decorationButtons, m_showTooltips, this); if (configDialog->exec() == QDialog::Accepted) { m_decorationButtons->setCustomPositions(configDialog->customPositions()); m_showTooltips = configDialog->showTooltips(); m_decorationButtons->setLeftButtons(configDialog->buttonsLeft()); m_decorationButtons->setRightButtons(configDialog->buttonsRight()); m_model->changeButtons(m_decorationButtons); emit changed(true); } delete configDialog; } void KWinDecorationModule::slotGHNSClicked() { QPointer downloadDialog = new KNS3::DownloadDialog("aurorae.knsrc", this); if (downloadDialog->exec() == QDialog::Accepted) { if (!downloadDialog->changedEntries().isEmpty()) { const QModelIndex index = m_proxyModel->mapToSource(m_proxyModel->index(m_listView->rootObject()->property("currentIndex").toInt(), 0)); const QString libraryName = m_model->data(index, DecorationModel::LibraryNameRole).toString(); bool aurorae = m_model->data(index, DecorationModel::TypeRole).toInt() == DecorationModelData::AuroraeDecoration; bool qml = m_model->data(index, DecorationModel::TypeRole).toInt() == DecorationModelData::QmlDecoration; const QString auroraeName = m_model->data(index, DecorationModel::AuroraeNameRole).toString(); m_model->reload(); if (aurorae) { const QModelIndex proxyIndex = m_proxyModel->mapFromSource(m_model->indexOfAuroraeName(auroraeName, "aurorae")); if (proxyIndex.isValid()) m_listView->rootObject()->setProperty("currentIndex", proxyIndex.row()); } else if (qml) { const QModelIndex proxyIndex = m_proxyModel->mapFromSource(m_model->indexOfAuroraeName(auroraeName, "qml")); if (proxyIndex.isValid()) m_listView->rootObject()->setProperty("currentIndex", proxyIndex.row()); } else { const QModelIndex proxyIndex = m_proxyModel->mapFromSource(m_model->indexOfLibrary(libraryName)); if (proxyIndex.isValid()) m_listView->rootObject()->setProperty("currentIndex", proxyIndex.row()); } } } delete downloadDialog; } void KWinDecorationModule::slotConfigureDecoration() { const QModelIndex index = m_proxyModel->mapToSource(m_proxyModel->index(m_listView->rootObject()->property("currentIndex").toInt(), 0)); bool reload = false; if (index.data(DecorationModel::TypeRole).toInt() == DecorationModelData::AuroraeDecoration || index.data(DecorationModel::TypeRole).toInt() == DecorationModelData::QmlDecoration) { QPointer dlg = new QDialog(this); dlg->setWindowTitle(i18n("Decoration Options")); KWinAuroraeConfigForm *form = new KWinAuroraeConfigForm(dlg); form->enableNoSideBorderSupport(index.data(DecorationModel::TypeRole).toInt() == DecorationModelData::QmlDecoration); dlg->setLayout(new QVBoxLayout); QDialogButtonBox* buttons = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel, dlg.data()); connect(buttons, SIGNAL(accepted()), dlg, SLOT(accept())); connect(buttons, SIGNAL(rejected()), dlg, SLOT(reject())); dlg->layout()->addWidget(form); dlg->layout()->addWidget(buttons); form->borderSizesCombo->setCurrentIndex(index.data(DecorationModel::BorderSizeRole).toInt()); form->buttonSizesCombo->setCurrentIndex(index.data(DecorationModel::ButtonSizeRole).toInt()); form->closeWindowsDoubleClick->setChecked(index.data(DecorationModel::CloseOnDblClickRole).toBool()); form->doubleClickMessage->setCloseButtonVisible(true); form->doubleClickMessage->setText(i18n("Close by double clicking:\n To open the menu, keep the button pressed until it appears.")); form->doubleClickMessage->setVisible(false); connect(form->closeWindowsDoubleClick, &QCheckBox::toggled, [form](bool toggled) { if (!toggled) { return; } form->doubleClickMessage->animatedShow(); }); // in case of QmlDecoration look for a config.ui in the package structure KConfigDialogManager *configManager = nullptr; if (index.data(DecorationModel::TypeRole).toInt() == DecorationModelData::QmlDecoration) { const QString packageName = index.data(DecorationModel::AuroraeNameRole).toString(); const QString uiPath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, "kwin/decorations/" + packageName + "/contents/ui/config.ui"); const QString configPath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, "kwin/decorations/" + packageName + "/contents/config/main.xml"); if (!uiPath.isEmpty() && !configPath.isEmpty()) { // load the KConfigSkeleton QFile configFile(configPath); KSharedConfigPtr auroraeConfig = KSharedConfig::openConfig("auroraerc"); KConfigGroup configGroup = auroraeConfig->group(packageName); KConfigLoader *skeleton = new KConfigLoader(configGroup, &configFile, dlg); // load the ui file QUiLoader *loader = new QUiLoader(dlg); QFile uiFile(uiPath); uiFile.open(QFile::ReadOnly); QWidget *customConfigForm = loader->load(&uiFile, form); uiFile.close(); form->layout()->addWidget(customConfigForm); // connect the ui file with the skeleton configManager = new KConfigDialogManager(customConfigForm, skeleton); configManager->updateWidgets(); } } if (dlg->exec() == QDialog::Accepted) { m_model->setData(index, form->borderSizesCombo->currentIndex(), DecorationModel::BorderSizeRole); m_model->setData(index, form->buttonSizesCombo->currentIndex(), DecorationModel::ButtonSizeRole); m_model->setData(index, form->closeWindowsDoubleClick->isChecked(), DecorationModel::CloseOnDblClickRole); if (configManager && configManager->hasChanged()) { // we have a config manager and the settings changed configManager->updateSettings(); m_model->notifyConfigChanged(index); } reload = true; } delete dlg; } else { QString name = index.data(DecorationModel::LibraryNameRole).toString(); QList< QVariant > borderSizes = index.data(DecorationModel::BorderSizesRole).toList(); const KDecorationDefines::BorderSize size = static_cast(index.data(DecorationModel::BorderSizeRole).toInt()); QPointer< KWinDecorationConfigDialog > configDialog = new KWinDecorationConfigDialog(name, borderSizes, size, this); if (configDialog->exec() == QDialog::Accepted) { m_model->setData(index, configDialog->borderSize(), DecorationModel::BorderSizeRole); reload = true; } delete configDialog; } if (reload) { save(); } } bool KWinDecorationModule::eventFilter(QObject *o, QEvent *e) { if (o == m_ui->decorationList) { if (e->type() == QEvent::Resize) updateScrollbarRange(); else if (e->type() == QEvent::KeyPress) { int d = 0; const int currentRow = m_listView->rootObject()->property("currentIndex").toInt(); const int key = static_cast(e)->key(); switch (key) { case Qt::Key_Home: d = -currentRow; break; case Qt::Key_End: d = m_proxyModel->rowCount() - (1 + currentRow); break; case Qt::Key_Up: d = -1; break; case Qt::Key_Down: d = 1; break; case Qt::Key_PageUp: case Qt::Key_PageDown: d = 150; if (QObject *decoItem = m_listView->rootObject()->findChild("decorationItem")) { QVariant v = decoItem->property("height"); if (v.isValid()) d = v.toInt(); } if (d > 0) d = qMax(m_ui->decorationList->height() / d, 1); if (key == Qt::Key_PageUp) d = -d; break; default: break; } if (d) { d = qMin(qMax(0, currentRow + d), m_proxyModel->rowCount()); m_listView->rootObject()->setProperty("currentIndex", d); return true; } } } else if (m_ui->decorationList->viewport()) { if (e->type() == QEvent::Wheel) { return static_cast(e)->orientation() == Qt::Horizontal; } } return KCModule::eventFilter(o, e); } void KWinDecorationModule::updateScrollbarRange() { m_ui->decorationList->verticalScrollBar()->blockSignals(true); const bool atMinimum = m_listView->rootObject()->property("atYBeginning").toBool(); const int h = m_listView->rootObject()->property("contentHeight").toInt(); const int y = atMinimum ? m_listView->rootObject()->property("contentY").toInt() : 0; m_ui->decorationList->verticalScrollBar()->setRange(y, y + h - m_ui->decorationList->height()); m_ui->decorationList->verticalScrollBar()->setPageStep(m_ui->decorationList->verticalScrollBar()->maximum()/m_model->rowCount()); m_ui->decorationList->verticalScrollBar()->blockSignals(false); } void KWinDecorationModule::updateScrollbarValue() { const int v = m_listView->rootObject()->property("contentY").toInt(); m_ui->decorationList->verticalScrollBar()->blockSignals(true); // skippig this will kill kinetic scrolling but the scrollwidth is too low m_ui->decorationList->verticalScrollBar()->setValue(v); m_ui->decorationList->verticalScrollBar()->blockSignals(false); } void KWinDecorationModule::updateViewPosition(int v) { m_listView->rootObject()->setProperty("contentY", v); } DecorationButtons::DecorationButtons(QObject *parent) : QObject(parent) , m_customPositions(false) , m_leftButtons() , m_rightButtons() { setLeftButtons(KDecorationOptions::defaultTitleButtonsLeft()); setRightButtons(KDecorationOptions::defaultTitleButtonsRight()); } DecorationButtons::~DecorationButtons() { } bool DecorationButtons::customPositions() const { return m_customPositions; } const QList &DecorationButtons::leftButtons() const { return m_leftButtons; } const QList &DecorationButtons::rightButtons() const { return m_rightButtons; } void DecorationButtons::setCustomPositions(bool set) { if (m_customPositions == set) { return; } m_customPositions = set; emit customPositionsChanged(); } void DecorationButtons::setLeftButtons(const QList &leftButtons) { if (m_leftButtons == leftButtons) { return; } m_leftButtons = leftButtons; emit leftButtonsChanged(); } void DecorationButtons::setRightButtons(const QList &rightButtons) { if (m_rightButtons == rightButtons) { return; } m_rightButtons = rightButtons; emit rightButtonsChanged(); } void DecorationButtons::resetToDefaults() { setCustomPositions(false); setLeftButtons(KDecorationOptions::defaultTitleButtonsLeft()); setRightButtons(KDecorationOptions::defaultTitleButtonsRight()); } } // namespace KWin #include "kwindecoration.moc"