kde-workspace/qguiplatformplugin_kde/qguiplatformplugin_kde.cpp

438 lines
13 KiB
C++
Raw Normal View History

2014-11-13 19:30:51 +02:00
/* This file is part of the KDE project
Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). <qt-info@nokia.com>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library 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
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include <KStandardDirs>
#include <KGlobal>
#include <KComponentData>
#include <KGlobalSettings>
#include <KStyle>
#include <KConfigGroup>
#include <KIcon>
#include <KFileDialog>
#include <KColorDialog>
#include <KDebug>
2014-11-13 19:30:51 +02:00
#include <QtCore/QHash>
#include <QtCore/QTextStream>
2014-11-13 19:30:51 +02:00
#include <QtCore/QTimer>
#include <QtGui/QFileDialog>
#include <QtGui/QColorDialog>
#include <QtGui/QApplication>
#include <QtGui/QToolButton>
#include <QtGui/QToolBar>
#include <QtGui/QMainWindow>
#include <QtGui/QGuiPlatformPlugin>
2014-11-13 19:30:51 +02:00
/*
* Map a Katie filter string into a KDE one.
2014-11-13 19:30:51 +02:00
* (from kfiledialog.cpp)
*/
static QString qt2KdeFilter(const QString &f)
{
QString filter;
QTextStream str(&filter, QIODevice::WriteOnly);
const QStringList list(f.split(";;").replaceInStrings("/", "\\/"));
bool first = true;
2014-11-13 19:30:51 +02:00
foreach (const QString &it, list) {
int ob = it.lastIndexOf('('), cb = it.lastIndexOf(')');
2014-11-13 19:30:51 +02:00
if (cb != -1 && ob < cb) {
if (first) {
first = false;
} else {
2014-11-13 19:30:51 +02:00
str << '\n';
}
str << it.mid(ob+1, (cb-ob)-1) << '|' << it.mid(0, ob);
2014-11-13 19:30:51 +02:00
}
}
return filter;
}
/*
* Map a KDE filter string into a Qt one.
* (from kfiledialog.cpp)
*/
static void kde2QtFilter(const QString &orig, const QString &kde, QString *sel)
{
if (sel) {
2014-11-13 19:30:51 +02:00
const QStringList list(orig.split(";;"));
int pos;
foreach (const QString &it, list) {
pos = it.indexOf(kde);
if (pos != -1 && pos > 0 &&
(it[pos-1] == '(' || it[pos-1] == ' ') &&
it.length() >= (kde.length() + pos) &&
(it[pos+kde.length()] == ')' || it[pos+kde.length()] == ' ')) {
*sel = it;
2014-11-13 19:30:51 +02:00
return;
}
}
2014-11-13 19:30:51 +02:00
}
}
class KFileDialogBridge : public KFileDialog
{
public:
KFileDialogBridge (const KUrl &startDir, const QString &filter, QFileDialog *original_)
: KFileDialog (startDir, filter, original_), original(original_)
{
connect(this, SIGNAL(fileSelected(QString)), original, SIGNAL(currentChanged(QString)));
}
virtual void accept()
{
KFileDialog::accept();
QMetaObject::invokeMethod(original, "accept"); //workaround protected
}
virtual void reject()
{
KFileDialog::reject();
QMetaObject::invokeMethod(original, "reject"); //workaround protected
}
QFileDialog *original;
};
class KColorDialogBridge : public KColorDialog
{
public:
KColorDialogBridge(QColorDialog* original_ = 0L) : KColorDialog(original_, true) , original(original_)
{
connect(this, SIGNAL(colorSelected(QColor)), original, SIGNAL(currentColorChanged(QColor)));
}
QColorDialog *original;
virtual void accept()
{
KColorDialog::accept();
original->setCurrentColor(color());
QMetaObject::invokeMethod(original, "accept"); //workaround protected
}
virtual void reject()
{
KColorDialog::reject();
QMetaObject::invokeMethod(original, "reject"); //workaround protected
}
};
Q_DECLARE_METATYPE(KFileDialogBridge *)
Q_DECLARE_METATYPE(KColorDialogBridge *)
class KQGuiPlatformPlugin : public QGuiPlatformPlugin
{
Q_OBJECT
public:
KQGuiPlatformPlugin()
{
QMetaObject::invokeMethod(this, "init", Qt::QueuedConnection);
}
QString styleName() final
2014-11-13 19:30:51 +02:00
{
const KConfigGroup pConfig(KGlobal::config(), "General");
return pConfig.readEntry("widgetStyle", KStyle::defaultStyle());
2014-11-13 19:30:51 +02:00
}
QPalette palette() final
2014-11-13 19:30:51 +02:00
{
return KGlobalSettings::createApplicationPalette();
}
QString systemIconThemeName() final
2014-11-13 19:30:51 +02:00
{
return KIconTheme::current();
}
QStringList iconThemeSearchPaths() final
2014-11-13 19:30:51 +02:00
{
return KGlobal::dirs()->resourceDirs("icon");
}
QIcon systemIcon(const QString &name) final
{
return KIcon(name);
}
QIcon fileSystemIcon(const QFileInfo &file) final
2014-11-13 19:30:51 +02:00
{
KMimeType::Ptr mime = KMimeType::findByPath(file.filePath(), 0, true);
if (!mime)
return QIcon();
return KIcon(mime->iconName());
2014-11-13 19:30:51 +02:00
}
int platformHint(QGuiPlatformPlugin::PlatformHint hint) final
2014-11-13 19:30:51 +02:00
{
switch(hint) {
case PH_ToolButtonStyle: {
KConfigGroup group(KGlobal::config(), "Toolbar style");
QString style = group.readEntry("ToolButtonStyle", "TextUnderIcon").toLower();
if (style == "textbesideicon" || style == "icontextright") {
return Qt::ToolButtonTextBesideIcon;
} else if (style == "textundericon" || style == "icontextbottom") {
return Qt::ToolButtonTextUnderIcon;
} else if (style == "textonly") {
return Qt::ToolButtonTextOnly;
}
2014-11-13 19:30:51 +02:00
return Qt::ToolButtonIconOnly;
}
case PH_ToolBarIconSize: {
return KIconLoader::global()->currentSize(KIconLoader::MainToolbar);
}
case PH_ItemView_ActivateItemOnSingleClick: {
return KGlobalSettings::singleClick();
}
default: {
break;
}
2014-11-13 19:30:51 +02:00
}
return QGuiPlatformPlugin::platformHint(hint);
}
public: // File Dialog integration
#define K_FD(QFD) KFileDialogBridge *kdefd = qvariant_cast<KFileDialogBridge *>(QFD->property("_k_bridge"))
void fileDialogDelete(QFileDialog *qfd) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
delete kdefd;
}
bool fileDialogSetVisible(QFileDialog *qfd, bool visible) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
if (!kdefd && visible) {
kdefd = new KFileDialogBridge(KUrl::fromPath(qfd->directory().canonicalPath()),
qt2KdeFilter(qfd->nameFilters().join(";;")), qfd);
qfd->setProperty("_k_bridge", QVariant::fromValue(kdefd));
}
if (visible) {
switch (qfd->fileMode()) {
case QFileDialog::AnyFile: {
2014-11-13 19:30:51 +02:00
kdefd->setMode(KFile::LocalOnly | KFile::File);
break;
}
case QFileDialog::ExistingFile: {
2014-11-13 19:30:51 +02:00
kdefd->setMode(KFile::LocalOnly | KFile::File | KFile::ExistingOnly);
break;
}
case QFileDialog::ExistingFiles: {
2014-11-13 19:30:51 +02:00
kdefd->setMode(KFile::LocalOnly | KFile::Files | KFile::ExistingOnly);
break;
}
2014-11-13 19:30:51 +02:00
case QFileDialog::Directory:
case QFileDialog::DirectoryOnly: {
2014-11-13 19:30:51 +02:00
kdefd->setMode(KFile::LocalOnly | KFile::Directory);
break;
}
2014-11-13 19:30:51 +02:00
}
kdefd->setOperationMode((qfd->acceptMode() == QFileDialog::AcceptSave) ? KFileDialog::Saving : KFileDialog::Opening);
kdefd->setCaption(qfd->windowTitle());
kdefd->setConfirmOverwrite(qfd->confirmOverwrite());
kdefd->setSelection(qfd->selectedFiles().value(0));
}
kdefd->setVisible(visible);
return true;
}
QDialog::DialogCode fileDialogResultCode(QFileDialog *qfd) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
Q_ASSERT(kdefd);
return QDialog::DialogCode(kdefd->result());
}
void fileDialogSetDirectory(QFileDialog *qfd, const QString &directory) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
kdefd->setUrl(KUrl::fromPath(directory));
}
QString fileDialogDirectory(const QFileDialog *qfd) const final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
Q_ASSERT(kdefd);
return kdefd->baseUrl().pathOrUrl();
}
void fileDialogSelectFile(QFileDialog *qfd, const QString &filename) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
Q_ASSERT(kdefd);
kdefd->setSelection(filename);
}
2014-11-13 19:30:51 +02:00
virtual QStringList fileDialogSelectedFiles(const QFileDialog *qfd) const
{
K_FD(qfd);
Q_ASSERT(kdefd);
return kdefd->selectedFiles();
}
#if 0
virtual void fileDialogSetFilter(QFileDialog *qfd)
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
}
#endif
void fileDialogSetNameFilters(QFileDialog *qfd, const QStringList &filters) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
Q_ASSERT(kdefd);
kdefd->setFilter(qt2KdeFilter(filters.join(";;")));
}
#if 0
void fileDialogSelectNameFilter(QFileDialog *qfd, const QString &filter) final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
}
#endif
QString fileDialogSelectedNameFilter(const QFileDialog *qfd) const final
2014-11-13 19:30:51 +02:00
{
K_FD(qfd);
Q_ASSERT(kdefd);
QString ret;
kde2QtFilter(qfd->nameFilters().join(";;"), kdefd->currentFilter(), &ret);
return ret;
}
2014-11-13 19:30:51 +02:00
public: // ColorDialog
#define K_CD(QCD) KColorDialogBridge *kdecd = qvariant_cast<KColorDialogBridge *>(QCD->property("_k_bridge"))
void colorDialogDelete(QColorDialog *qcd) final
2014-11-13 19:30:51 +02:00
{
K_CD(qcd);
delete kdecd;
}
bool colorDialogSetVisible(QColorDialog *qcd, bool visible) final
2014-11-13 19:30:51 +02:00
{
K_CD(qcd);
if (!kdecd) {
kdecd = new KColorDialogBridge(qcd);
kdecd->setColor(qcd->currentColor());
if (qcd->options() & QColorDialog::NoButtons) {
kdecd->setButtons(KDialog::None);
}
kdecd->setModal(qcd->isModal());
qcd->setProperty("_k_bridge", QVariant::fromValue(kdecd));
}
if (visible) {
kdecd->setCaption(qcd->windowTitle());
kdecd->setAlphaChannelEnabled(qcd->options() & QColorDialog::ShowAlphaChannel);
}
kdecd->setVisible(visible);
return true;
}
void colorDialogSetCurrentColor(QColorDialog *qcd, const QColor &color) final
2014-11-13 19:30:51 +02:00
{
K_CD(qcd);
if (kdecd) {
kdecd->setColor(color);
}
}
private slots:
void init()
{
updateEffects();
2014-11-13 19:30:51 +02:00
connect(KIconLoader::global(), SIGNAL(iconLoaderSettingsChanged()), this, SLOT(updateToolbarIcons()));
connect(KGlobalSettings::self(), SIGNAL(toolbarAppearanceChanged(int)), this, SLOT(updateToolbarStyle()));
connect(KGlobalSettings::self(), SIGNAL(kdisplayStyleChanged()), this, SLOT(updateStyle()));
connect(KGlobalSettings::self(), SIGNAL(kdisplayPaletteChanged()), this, SLOT(updatePalette()));
2014-11-13 19:30:51 +02:00
}
void updateToolbarStyle()
{
QWidgetList widgets = QApplication::allWidgets();
for (int i = 0; i < widgets.size(); ++i) {
QWidget *widget = widgets.at(i);
if (qobject_cast<QToolButton*>(widget)) {
QEvent event(QEvent::StyleChange);
QApplication::sendEvent(widget, &event);
}
}
}
void updateToolbarIcons()
{
QWidgetList widgets = QApplication::allWidgets();
for (int i = 0; i < widgets.size(); ++i) {
QWidget *widget = widgets.at(i);
if (qobject_cast<QToolBar*>(widget) || qobject_cast<QMainWindow*>(widget)) {
QEvent event(QEvent::StyleChange);
QApplication::sendEvent(widget, &event);
}
}
}
void updateEffects()
{
KGlobalSettings::GraphicEffects graphicEffects = KGlobalSettings::graphicEffectsLevel();
bool effectsEnabled = (graphicEffects != KGlobalSettings::NoEffects);
bool complexEffects = (graphicEffects & KGlobalSettings::ComplexAnimationEffects);
if (effectsEnabled) {
QApplication::setEffectEnabled(Qt::UI_General, true);
// the fade effect requires compositor and as such is enabled only when complex animation is on
if (complexEffects) {
QApplication::setEffectEnabled(Qt::UI_FadeMenu, true);
QApplication::setEffectEnabled(Qt::UI_FadeTooltip, true);
}
} else {
QApplication::setEffectEnabled(Qt::UI_General, false);
QApplication::setEffectEnabled(Qt::UI_FadeMenu, false);
QApplication::setEffectEnabled(Qt::UI_FadeTooltip, false);
}
}
void updateStyle()
2014-11-13 19:30:51 +02:00
{
if (qApp) {
if (qApp->style()->objectName() != styleName()) {
qApp->setStyle(styleName());
}
}
updateEffects();
2014-11-13 19:30:51 +02:00
}
void updatePalette()
{
QApplication::setPalette(palette());
}
2014-11-13 19:30:51 +02:00
};
Q_EXPORT_PLUGIN(KQGuiPlatformPlugin)
2014-11-13 19:30:51 +02:00
#include "qguiplatformplugin_kde.moc"