| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418 |
- /*
- This file is part of Telegram Desktop,
- the official desktop application for the Telegram messaging service.
- For license and copyright information please follow this link:
- https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
- */
- #include "platform/linux/tray_linux.h"
- #include "base/invoke_queued.h"
- #include "base/qt_signal_producer.h"
- #include "base/platform/linux/base_linux_dbus_utilities.h"
- #include "core/application.h"
- #include "core/sandbox.h"
- #include "platform/platform_specific.h"
- #include "ui/ui_utility.h"
- #include "ui/widgets/popup_menu.h"
- #include "window/window_controller.h"
- #include "styles/style_window.h"
- #include <QtCore/QCoreApplication>
- #include <QtWidgets/QMenu>
- #include <QtWidgets/QSystemTrayIcon>
- #include <gio/gio.hpp>
- namespace Platform {
- namespace {
- using namespace gi::repository;
- [[nodiscard]] QString PanelIconName(int counter, bool muted) {
- return ApplicationIconName() + ((counter > 0)
- ? (muted
- ? u"-mute"_q
- : u"-attention"_q)
- : QString()) + u"-symbolic"_q;
- }
- } // namespace
- class IconGraphic final {
- public:
- explicit IconGraphic();
- ~IconGraphic();
- void updateState();
- [[nodiscard]] bool isRefreshNeeded() const;
- [[nodiscard]] QIcon trayIcon();
- private:
- struct State {
- QIcon systemIcon;
- QString iconThemeName;
- bool monochrome = false;
- int32 counter = 0;
- bool muted = false;
- };
- [[nodiscard]] QIcon systemIcon() const;
- [[nodiscard]] bool isCounterNeeded(const State &state) const;
- [[nodiscard]] int counterSlice(int counter) const;
- [[nodiscard]] QSize dprSize(const QImage &image) const;
- const int _iconSizes[7];
- base::flat_map<int, QImage> _imageBack;
- QIcon _trayIcon;
- State _current;
- State _new;
- };
- IconGraphic::IconGraphic()
- : _iconSizes{ 16, 22, 32, 48, 64, 128, 256 } {
- updateState();
- }
- IconGraphic::~IconGraphic() = default;
- QIcon IconGraphic::systemIcon() const {
- if (_new.iconThemeName == _current.iconThemeName
- && _new.monochrome == _current.monochrome
- && (_new.counter > 0) == (_current.counter > 0)
- && _new.muted == _current.muted) {
- return _current.systemIcon;
- }
- const auto candidates = {
- _new.monochrome ? PanelIconName(_new.counter, _new.muted) : QString(),
- ApplicationIconName(),
- };
- for (const auto &candidate : candidates) {
- if (candidate.isEmpty()) {
- continue;
- }
- const auto icon = QIcon::fromTheme(candidate);
- if (icon.name() == candidate) {
- return icon;
- }
- }
- return QIcon();
- }
- bool IconGraphic::isCounterNeeded(const State &state) const {
- return state.systemIcon.name() != PanelIconName(
- state.counter,
- state.muted);
- }
- int IconGraphic::counterSlice(int counter) const {
- return (counter >= 100)
- ? (100 + (counter % 10))
- : counter;
- }
- QSize IconGraphic::dprSize(const QImage &image) const {
- return image.size() / image.devicePixelRatio();
- }
- void IconGraphic::updateState() {
- _new.iconThemeName = QIcon::themeName();
- _new.monochrome = Core::App().settings().trayIconMonochrome();
- _new.counter = Core::App().unreadBadge();
- _new.muted = Core::App().unreadBadgeMuted();
- _new.systemIcon = systemIcon();
- }
- bool IconGraphic::isRefreshNeeded() const {
- return _trayIcon.isNull()
- || _new.iconThemeName != _current.iconThemeName
- || _new.systemIcon.name() != _current.systemIcon.name()
- || (isCounterNeeded(_new)
- ? _new.muted != _current.muted
- || counterSlice(_new.counter) != counterSlice(
- _current.counter)
- : false);
- }
- QIcon IconGraphic::trayIcon() {
- if (!isRefreshNeeded()) {
- return _trayIcon;
- }
- const auto guard = gsl::finally([&] {
- _current = _new;
- });
- if (!isCounterNeeded(_new)) {
- _trayIcon = _new.systemIcon;
- return _trayIcon;
- }
- QIcon result;
- for (const auto iconSize : _iconSizes) {
- auto ¤tImageBack = _imageBack[iconSize];
- const auto desiredSize = QSize(iconSize, iconSize);
- if (currentImageBack.isNull()
- || _new.iconThemeName != _current.iconThemeName
- || _new.systemIcon.name() != _current.systemIcon.name()) {
- if (!_new.systemIcon.isNull()) {
- // We can't use QIcon::actualSize here
- // since it works incorrectly with svg icon themes
- currentImageBack = _new.systemIcon
- .pixmap(desiredSize)
- .toImage();
- const auto firstAttemptSize = dprSize(currentImageBack);
- // if current icon theme is not a svg one, Qt can return
- // a pixmap that less in size even if there are a bigger one
- if (firstAttemptSize.width() < desiredSize.width()) {
- const auto availableSizes
- = _new.systemIcon.availableSizes();
- const auto biggestSize = ranges::max_element(
- availableSizes,
- std::less<>(),
- &QSize::width);
- if (biggestSize->width() > firstAttemptSize.width()) {
- currentImageBack = _new.systemIcon
- .pixmap(*biggestSize)
- .toImage();
- }
- }
- } else {
- currentImageBack = Window::Logo();
- }
- if (dprSize(currentImageBack) != desiredSize) {
- currentImageBack = currentImageBack.scaled(
- desiredSize * currentImageBack.devicePixelRatio(),
- Qt::IgnoreAspectRatio,
- Qt::SmoothTransformation);
- }
- }
- result.addPixmap(Ui::PixmapFromImage(_new.counter > 0
- ? Window::WithSmallCounter(std::move(currentImageBack), {
- .size = iconSize,
- .count = _new.counter,
- .bg = _new.muted ? st::trayCounterBgMute : st::trayCounterBg,
- .fg = st::trayCounterFg,
- }) : std::move(currentImageBack)));
- }
- _trayIcon = result;
- return _trayIcon;
- }
- class TrayEventFilter final : public QObject {
- public:
- TrayEventFilter(not_null<QObject*> parent);
- [[nodiscard]] rpl::producer<> contextMenuFilters() const;
- protected:
- bool eventFilter(QObject *watched, QEvent *event) override;
- private:
- const QString _iconObjectName;
- rpl::event_stream<> _contextMenuFilters;
- };
- TrayEventFilter::TrayEventFilter(not_null<QObject*> parent)
- : QObject(parent)
- , _iconObjectName("QSystemTrayIconSys") {
- parent->installEventFilter(this);
- }
- bool TrayEventFilter::eventFilter(QObject *obj, QEvent *event) {
- if (event->type() == QEvent::MouseButtonPress
- && obj->objectName() == _iconObjectName) {
- const auto m = static_cast<QMouseEvent*>(event);
- if (m->button() == Qt::RightButton) {
- Core::Sandbox::Instance().customEnterFromEventLoop([&] {
- _contextMenuFilters.fire({});
- });
- return true;
- }
- }
- return false;
- }
- rpl::producer<> TrayEventFilter::contextMenuFilters() const {
- return _contextMenuFilters.events();
- }
- Tray::Tray() {
- auto connection = Gio::bus_get_sync(Gio::BusType::SESSION_, nullptr);
- if (connection) {
- _sniWatcher = std::make_unique<base::Platform::DBus::ServiceWatcher>(
- connection.gobj_(),
- "org.kde.StatusNotifierWatcher",
- [=](
- const std::string &service,
- const std::string &oldOwner,
- const std::string &newOwner) {
- Core::Sandbox::Instance().customEnterFromEventLoop([&] {
- if (hasIcon()) {
- destroyIcon();
- createIcon();
- }
- });
- });
- }
- }
- void Tray::createIcon() {
- if (!_icon) {
- LOG(("System tray available: %1").arg(Logs::b(TrayIconSupported())));
- if (!_iconGraphic) {
- _iconGraphic = std::make_unique<IconGraphic>();
- }
- const auto showCustom = [=] {
- _aboutToShowRequests.fire({});
- InvokeQueued(_menuCustom.get(), [=] {
- _menuCustom->popup(QCursor::pos());
- });
- };
- _icon = base::make_unique_q<QSystemTrayIcon>(nullptr);
- _icon->setIcon(_iconGraphic->trayIcon());
- _icon->setToolTip(AppName.utf16());
- using Reason = QSystemTrayIcon::ActivationReason;
- base::qt_signal_producer(
- _icon.get(),
- &QSystemTrayIcon::activated
- ) | rpl::start_with_next([=](Reason reason) {
- if (reason == QSystemTrayIcon::Context) {
- showCustom();
- } else {
- _iconClicks.fire({});
- }
- }, _lifetime);
- _icon->setContextMenu(_menu.get());
- if (!_eventFilter) {
- _eventFilter = base::make_unique_q<TrayEventFilter>(
- QCoreApplication::instance());
- _eventFilter->contextMenuFilters(
- ) | rpl::start_with_next([=] {
- showCustom();
- }, _lifetime);
- }
- }
- updateIcon();
- _icon->show();
- }
- void Tray::destroyIcon() {
- _icon = nullptr;
- }
- void Tray::updateIcon() {
- if (!_icon || !_iconGraphic) {
- return;
- }
- _iconGraphic->updateState();
- if (_iconGraphic->isRefreshNeeded()) {
- _icon->setIcon(_iconGraphic->trayIcon());
- }
- }
- void Tray::createMenu() {
- if (!_menu) {
- _menu = base::make_unique_q<QMenu>(nullptr);
- }
- if (!_menuCustom) {
- _menuCustom = base::make_unique_q<Ui::PopupMenu>(nullptr);
- _menuCustom->deleteOnHide(false);
- }
- }
- void Tray::destroyMenu() {
- _menuCustom = nullptr;
- if (_menu) {
- _menu->clear();
- }
- _actionsLifetime.destroy();
- }
- void Tray::addAction(rpl::producer<QString> text, Fn<void()> &&callback) {
- if (_menuCustom) {
- const auto action = _menuCustom->addAction(QString(), callback);
- rpl::duplicate(
- text
- ) | rpl::start_with_next([=](const QString &text) {
- action->setText(text);
- }, _actionsLifetime);
- }
- if (_menu) {
- const auto action = _menu->addAction(QString(), std::move(callback));
- std::move(
- text
- ) | rpl::start_with_next([=](const QString &text) {
- action->setText(text);
- }, _actionsLifetime);
- }
- }
- void Tray::showTrayMessage() const {
- }
- bool Tray::hasTrayMessageSupport() const {
- return false;
- }
- rpl::producer<> Tray::aboutToShowRequests() const {
- return rpl::merge(
- _aboutToShowRequests.events(),
- _menu
- ? base::qt_signal_producer(_menu.get(), &QMenu::aboutToShow)
- : rpl::never<>() | rpl::type_erased());
- }
- rpl::producer<> Tray::showFromTrayRequests() const {
- return rpl::never<>();
- }
- rpl::producer<> Tray::hideToTrayRequests() const {
- return rpl::never<>();
- }
- rpl::producer<> Tray::iconClicks() const {
- return _iconClicks.events();
- }
- bool Tray::hasIcon() const {
- return _icon;
- }
- rpl::lifetime &Tray::lifetime() {
- return _lifetime;
- }
- Tray::~Tray() = default;
- bool HasMonochromeSetting() {
- return QIcon::hasThemeIcon(
- PanelIconName(
- Core::App().unreadBadge(),
- Core::App().unreadBadgeMuted()));
- }
- } // namespace Platform
|