X-Git-Url: https://git.quassel-irc.org/?p=quassel.git;a=blobdiff_plain;f=src%2Fqtui%2Fqtui.cpp;h=0082bd06d1a869d762bce80d36f8f3bef2dba92f;hp=30727676bf9ca71e21d3aeb689a19e9be35b3625;hb=3abd4b7d5ab303f8d990c104748e5d5aef4db355;hpb=5c78a50fa720e5f82fcaa03c0176feab71d74c8e diff --git a/src/qtui/qtui.cpp b/src/qtui/qtui.cpp index 30727676..0082bd06 100644 --- a/src/qtui/qtui.cpp +++ b/src/qtui/qtui.cpp @@ -1,5 +1,5 @@ /*************************************************************************** - * Copyright (C) 2005-08 by the Quassel Project * + * Copyright (C) 2005-2018 by the Quassel Project * * devel@quassel-irc.org * * * * This program is free software; you can redistribute it and/or modify * @@ -15,124 +15,383 @@ * 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., * - * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * ***************************************************************************/ #include "qtui.h" -#include "actioncollection.h" +#include +#include +#include +#include +#include + +#include "abstractnotificationbackend.h" +#include "buffermodel.h" #include "chatlinemodel.h" +#include "contextmenuactionprovider.h" #include "mainwin.h" -#include "abstractnotificationbackend.h" #include "qtuimessageprocessor.h" +#include "qtuisettings.h" #include "qtuistyle.h" +#include "systemtray.h" +#include "toolbaractionprovider.h" #include "types.h" -#include "uisettings.h" #include "util.h" -ActionCollection *QtUi::_actionCollection = 0; -MainWin *QtUi::_mainWin = 0; -QSet QtUi::_notificationBackends; +QtUi *QtUi::_instance = nullptr; +MainWin *QtUi::_mainWin = nullptr; +QList QtUi::_notificationBackends; QList QtUi::_notifications; -QtUiStyle *QtUi::_style = 0; -QtUi::QtUi() : AbstractUi() { - if(_style != 0) { - qWarning() << "QtUi has been instantiated again!"; - return; - } - _actionCollection = new ActionCollection(this); +QtUi::QtUi() + : GraphicalUi() + , _systemIconTheme{QIcon::themeName()} +{ + if (_instance != nullptr) { + qWarning() << "QtUi has been instantiated again!"; + return; + } + _instance = this; + + if (Quassel::isOptionSet("icontheme")) { + _systemIconTheme = Quassel::optionValue("icontheme"); + QIcon::setThemeName(_systemIconTheme); + } + + QtUiSettings uiSettings; + Quassel::loadTranslation(uiSettings.value("Locale", QLocale::system()).value()); + + setupIconTheme(); + + QApplication::setWindowIcon(QIcon::fromTheme("quassel")); + + setContextMenuActionProvider(new ContextMenuActionProvider(this)); + setToolBarActionProvider(new ToolBarActionProvider(this)); + + setUiStyle(new QtUiStyle(this)); + _mainWin = new MainWin(); + + setMainWidget(_mainWin); + + connect(_mainWin, SIGNAL(connectToCore(const QVariantMap &)), this, SIGNAL(connectToCore(const QVariantMap &))); + connect(_mainWin, SIGNAL(disconnectFromCore()), this, SIGNAL(disconnectFromCore())); + connect(Client::instance(), SIGNAL(bufferMarkedAsRead(BufferId)), SLOT(closeNotifications(BufferId))); +} + + +QtUi::~QtUi() +{ + unregisterAllNotificationBackends(); + delete _mainWin; + _mainWin = nullptr; + _instance = nullptr; +} + + +void QtUi::init() +{ + _mainWin->init(); + QtUiSettings uiSettings; + uiSettings.initAndNotify("UseSystemTrayIcon", this, SLOT(useSystemTrayChanged(QVariant)), true); + + GraphicalUi::init(); // needs to be called after the mainWin is initialized +} + - UiSettings uiSettings; - loadTranslation(uiSettings.value("Locale", QLocale::system()).value()); +MessageModel *QtUi::createMessageModel(QObject *parent) +{ + return new ChatLineModel(parent); +} - _mainWin = new MainWin(); - _style = new QtUiStyle; - connect(_mainWin, SIGNAL(connectToCore(const QVariantMap &)), this, SIGNAL(connectToCore(const QVariantMap &))); - connect(_mainWin, SIGNAL(disconnectFromCore()), this, SIGNAL(disconnectFromCore())); +AbstractMessageProcessor *QtUi::createMessageProcessor(QObject *parent) +{ + return new QtUiMessageProcessor(parent); } -QtUi::~QtUi() { - unregisterAllNotificationBackends(); - delete _style; - delete _mainWin; + +void QtUi::connectedToCore() +{ + _mainWin->connectedToCore(); } -void QtUi::init() { - _mainWin->init(); + +void QtUi::disconnectedFromCore() +{ + _mainWin->disconnectedFromCore(); + GraphicalUi::disconnectedFromCore(); } -MessageModel *QtUi::createMessageModel(QObject *parent) { - return new ChatLineModel(parent); + +void QtUi::useSystemTrayChanged(const QVariant &v) +{ + _useSystemTray = v.toBool(); + SystemTray *tray = mainWindow()->systemTray(); + if (_useSystemTray) { + if (tray->isSystemTrayAvailable()) + tray->setVisible(true); + } + else { + if (tray->isSystemTrayAvailable() && mainWindow()->isVisible()) + tray->setVisible(false); + } } -AbstractMessageProcessor *QtUi::createMessageProcessor(QObject *parent) { - return new QtUiMessageProcessor(parent); + +bool QtUi::haveSystemTray() +{ + return mainWindow()->systemTray()->isSystemTrayAvailable() && instance()->_useSystemTray; } -void QtUi::connectedToCore() { - _mainWin->connectedToCore(); + +bool QtUi::isHidingMainWidgetAllowed() const +{ + return haveSystemTray(); } -void QtUi::disconnectedFromCore() { - _mainWin->disconnectedFromCore(); + +void QtUi::minimizeRestore(bool show) +{ + SystemTray *tray = mainWindow()->systemTray(); + if (show) { + if (tray && !_useSystemTray) + tray->setVisible(false); + } + else { + if (tray && _useSystemTray) + tray->setVisible(true); + } + GraphicalUi::minimizeRestore(show); } -void QtUi::registerNotificationBackend(AbstractNotificationBackend *backend) { - if(!_notificationBackends.contains(backend)) { - _notificationBackends.insert(backend); - } + +void QtUi::registerNotificationBackend(AbstractNotificationBackend *backend) +{ + if (!_notificationBackends.contains(backend)) { + _notificationBackends.append(backend); + instance()->connect(backend, SIGNAL(activated(uint)), SLOT(notificationActivated(uint))); + } } -void QtUi::unregisterNotificationBackend(AbstractNotificationBackend *backend) { - _notificationBackends.remove(backend); + +void QtUi::unregisterNotificationBackend(AbstractNotificationBackend *backend) +{ + _notificationBackends.removeAll(backend); } -void QtUi::unregisterAllNotificationBackends() { - _notificationBackends.clear(); + +void QtUi::unregisterAllNotificationBackends() +{ + _notificationBackends.clear(); } -const QSet &QtUi::notificationBackends() { - return _notificationBackends; + +const QList &QtUi::notificationBackends() +{ + return _notificationBackends; } -uint QtUi::invokeNotification(BufferId bufId, const QString &sender, const QString &text) { - static int notificationId = 0; - //notificationId++; - AbstractNotificationBackend::Notification notification(++notificationId, bufId, sender, text); - _notifications.append(notification); - foreach(AbstractNotificationBackend *backend, _notificationBackends) + +uint QtUi::invokeNotification(BufferId bufId, AbstractNotificationBackend::NotificationType type, const QString &sender, const QString &text) +{ + static int notificationId = 0; + + AbstractNotificationBackend::Notification notification(++notificationId, bufId, type, sender, text); + _notifications.append(notification); + foreach(AbstractNotificationBackend *backend, _notificationBackends) backend->notify(notification); - return notificationId; + return notificationId; } -void QtUi::closeNotification(uint notificationId) { - QList::iterator i = _notifications.begin(); - while(i != _notifications.end()) { - if((*i).notificationId == notificationId) { - foreach(AbstractNotificationBackend *backend, _notificationBackends) - backend->close(notificationId); - _notifications.erase(i); - break; + +void QtUi::closeNotification(uint notificationId) +{ + QList::iterator i = _notifications.begin(); + while (i != _notifications.end()) { + if (i->notificationId == notificationId) { + foreach(AbstractNotificationBackend *backend, _notificationBackends) + backend->close(notificationId); + i = _notifications.erase(i); + } + else ++i; + } +} + + +void QtUi::closeNotifications(BufferId bufferId) +{ + QList::iterator i = _notifications.begin(); + while (i != _notifications.end()) { + if (!bufferId.isValid() || i->bufferId == bufferId) { + foreach(AbstractNotificationBackend *backend, _notificationBackends) + backend->close(i->notificationId); + i = _notifications.erase(i); + } + else ++i; + } +} + + +const QList &QtUi::activeNotifications() +{ + return _notifications; +} + + +void QtUi::notificationActivated(uint notificationId) +{ + if (notificationId != 0) { + QList::iterator i = _notifications.begin(); + while (i != _notifications.end()) { + if (i->notificationId == notificationId) { + BufferId bufId = i->bufferId; + if (bufId.isValid()) + Client::bufferModel()->switchToBuffer(bufId); + break; + } + ++i; + } + } + closeNotification(notificationId); + + activateMainWidget(); +} + + +void QtUi::bufferMarkedAsRead(BufferId bufferId) +{ + if (bufferId.isValid()) { + closeNotifications(bufferId); } - ++i; - } } -void QtUi::closeNotifications(BufferId bufferId) { - QList::iterator i = _notifications.begin(); - while(i != _notifications.end()) { - if((*i).bufferId == bufferId) { - foreach(AbstractNotificationBackend *backend, _notificationBackends) - backend->close((*i).notificationId); - _notifications.erase(i); + +std::vector> QtUi::availableIconThemes() const +{ + //: Supported icon theme names + static const std::vector> supported { + { "breeze", tr("Breeze") }, + { "breeze-dark", tr("Breeze Dark") }, + { "oxygen", tr("Oxygen") } + }; + + std::vector> result; + for (auto &&themePair : supported) { + for (auto &&dir : QIcon::themeSearchPaths()) { + if (QFileInfo{dir + "/" + themePair.first + "/index.theme"}.exists()) { + result.push_back(themePair); + break; + } + } } - ++i; - } + + return result; } -const QList &QtUi::activeNotifications() { - return _notifications; + +void QtUi::setupIconTheme() +{ + // Add paths to our own icon sets to the theme search paths + QStringList themePaths = QIcon::themeSearchPaths(); + themePaths.removeAll(":/icons"); // this should come last + for (auto &&dataDir : Quassel::dataDirPaths()) { + QString iconDir{dataDir + "icons"}; + if (QFileInfo{iconDir}.isDir()) { + themePaths << iconDir; + } + } + themePaths << ":/icons"; + QIcon::setThemeSearchPaths(themePaths); + + refreshIconTheme(); } + +void QtUi::refreshIconTheme() +{ + // List of available fallback themes + QStringList availableThemes; + for (auto &&themePair : availableIconThemes()) { + availableThemes << themePair.first; + } + + if (availableThemes.isEmpty()) { + // We could probably introduce a more sophisticated fallback handling, such as putting the "most important" icons into hicolor, + // but this just gets complex for no good reason. We really rely on a supported theme to be installed, if not system-wide, then + // as part of the Quassel installation (which is enabled by default anyway). + qWarning() << tr("No supported icon theme installed, you'll lack icons! Supported are the KDE/Plasma themes Breeze, Breeze Dark and Oxygen."); + return; + } + + UiStyleSettings s; + QString fallbackTheme{s.value("Icons/FallbackTheme").toString()}; + + if (fallbackTheme.isEmpty() || !availableThemes.contains(fallbackTheme)) { + if (availableThemes.contains(_systemIconTheme)) { + fallbackTheme = _systemIconTheme; + } + else { + fallbackTheme = availableThemes.first(); + } + } + + if (_systemIconTheme.isEmpty() || _systemIconTheme == fallbackTheme || s.value("Icons/OverrideSystemTheme", false).toBool()) { + // We have a valid fallback theme and want to override the system theme (if it's even defined), so we're basically done + QIcon::setThemeName(fallbackTheme); + return; + } + +#if QT_VERSION >= 0x050000 + // At this point, we have a system theme that we don't want to override, but that may not contain all + // required icons. + // We create a dummy theme that inherits first from the system theme, then from the supported fallback. + // This rather ugly hack allows us to inject the fallback into the inheritance chain, so non-standard + // icons missing in the system theme will be filled in by the fallback. + // Since we can't get notified when the system theme changes, this means that a restart may be required + // to apply a theme change... but you can't have everything, I guess. + if (!_dummyThemeDir) { + _dummyThemeDir.reset(new QTemporaryDir{}); + if (!_dummyThemeDir->isValid() || !QDir{_dummyThemeDir->path()}.mkpath("icons/quassel-icon-proxy/apps/32")) { + qWarning() << "Could not create temporary directory for proxying the system icon theme, using fallback"; + QIcon::setThemeName(fallbackTheme); + return; + } + // Add this to XDG_DATA_DIRS, otherwise KIconLoader complains + auto xdgDataDirs = qgetenv("XDG_DATA_DIRS"); + if (!xdgDataDirs.isEmpty()) + xdgDataDirs += ":"; + xdgDataDirs += _dummyThemeDir->path(); + qputenv("XDG_DATA_DIRS", xdgDataDirs); + + QIcon::setThemeSearchPaths(QIcon::themeSearchPaths() << _dummyThemeDir->path() + "/icons"); + } + + QFile indexFile{_dummyThemeDir->path() + "/icons/quassel-icon-proxy/index.theme"}; + if (!indexFile.open(QFile::WriteOnly|QFile::Truncate)) { + qWarning() << "Could not create index file for proxying the system icon theme, using fallback"; + QIcon::setThemeName(fallbackTheme); + return; + } + + // Write a dummy index file that is sufficient to make QIconLoader happy + auto indexContents = QString{ + "[Icon Theme]\n" + "Name=quassel-icon-proxy\n" + "Inherits=%1,%2\n" + "Directories=apps/32\n" + "[apps/32]\nSize=32\nType=Fixed\n" + }.arg(_systemIconTheme, fallbackTheme); + if (indexFile.write(indexContents.toLatin1()) < 0) { + qWarning() << "Could not write index file for proxying the system icon theme, using fallback"; + QIcon::setThemeName(fallbackTheme); + return; + } + indexFile.close(); + QIcon::setThemeName("quassel-icon-proxy"); +#else + // Qt4 doesn't support QTemporaryDir. Since it's deprecated and slated to be removed soon anyway, we don't bother + // writing a replacement and simply don't support not overriding the system theme. + QIcon::setThemeName(fallbackTheme); +#endif +}