Use LXQt's StatusNotifierItem implementation instead of appindicator

This commit is contained in:
Ilya Fedin 2020-01-31 10:34:37 +04:00 committed by John Preston
parent 1b1f9d9985
commit 3b4dfa1381
22 changed files with 1089 additions and 567 deletions

View File

@ -54,7 +54,7 @@ jobs:
sudo apt-get update sudo apt-get update
sudo apt-get install software-properties-common -y && \ sudo apt-get install software-properties-common -y && \
sudo apt-get install git libexif-dev liblzma-dev libz-dev libssl-dev \ sudo apt-get install git libexif-dev liblzma-dev libz-dev libssl-dev \
libappindicator-dev libicu-dev libdee-dev libdrm-dev dh-autoreconf \ libgtk2.0-dev libice-dev libsm-dev libicu-dev libdrm-dev dh-autoreconf \
autoconf automake build-essential libass-dev libfreetype6-dev \ autoconf automake build-essential libass-dev libfreetype6-dev \
libgpac-dev libsdl1.2-dev libtheora-dev libtool libva-dev libvdpau-dev \ libgpac-dev libsdl1.2-dev libtheora-dev libtool libva-dev libvdpau-dev \
libvorbis-dev libenchant-dev libxcb1-dev libxcb-image0-dev libxcb-shm0-dev \ libvorbis-dev libenchant-dev libxcb1-dev libxcb-image0-dev libxcb-shm0-dev \

3
.gitmodules vendored
View File

@ -61,3 +61,6 @@
[submodule "Telegram/lib_qr"] [submodule "Telegram/lib_qr"]
path = Telegram/lib_qr path = Telegram/lib_qr
url = https://github.com/desktop-app/lib_qr.git url = https://github.com/desktop-app/lib_qr.git
[submodule "Telegram/ThirdParty/libdbusmenu-qt"]
path = Telegram/ThirdParty/libdbusmenu-qt
url = https://github.com/desktop-app/libdbusmenu-qt.git

View File

@ -77,6 +77,14 @@ if (DESKTOP_APP_USE_PACKAGED)
) )
endif() endif()
if (LINUX AND NOT TDESKTOP_DISABLE_DBUS_INTEGRATION)
target_link_libraries(Telegram
PRIVATE
desktop-app::external_statusnotifieritem
desktop-app::external_dbusmenu_qt
)
endif()
target_link_libraries(Telegram target_link_libraries(Telegram
PRIVATE PRIVATE
tdesktop::lib_mtproto tdesktop::lib_mtproto
@ -1059,28 +1067,6 @@ elseif (LINUX)
find_library(X11_LIBRARY X11) find_library(X11_LIBRARY X11)
target_link_libraries(Telegram PRIVATE ${X11_LIBRARY}) target_link_libraries(Telegram PRIVATE ${X11_LIBRARY})
endif() endif()
set(appindicator_packages
ayatana-appindicator3-0.1
ayatana-appindicator-0.1
appindicator3-0.1
appindicator-0.1
)
set(appindicator_found 0)
foreach (package ${appindicator_packages})
pkg_check_modules(APPIND_${package} ${package})
if (APPIND_${package}_FOUND)
set(appindicator_found 1)
target_include_directories(Telegram PRIVATE "${APPIND_${package}_INCLUDE_DIRS}")
if (${package} MATCHES "ayatana")
target_compile_definitions(Telegram PRIVATE TDESKTOP_USE_AYATANA_INDICATORS)
endif()
break()
endif()
endforeach()
if (NOT ${appindicator_found})
message(FATAL_ERROR "No libappindicator found by pkg-config.")
endif()
endif() endif()
endif() endif()

View File

@ -144,14 +144,10 @@ void MainWindow::firstShow() {
if (Platform::IsLinux()) { if (Platform::IsLinux()) {
trayIconMenu->addAction(tr::lng_open_from_tray(tr::now), this, SLOT(showFromTray())); trayIconMenu->addAction(tr::lng_open_from_tray(tr::now), this, SLOT(showFromTray()));
trayIconMenu->addAction(tr::lng_minimize_to_tray(tr::now), this, SLOT(minimizeToTray()));
trayIconMenu->addAction(notificationActionText, this, SLOT(toggleDisplayNotifyFromTray()));
trayIconMenu->addAction(tr::lng_quit_from_tray(tr::now), this, SLOT(quitFromTray()));
} else {
trayIconMenu->addAction(tr::lng_minimize_to_tray(tr::now), this, SLOT(minimizeToTray()));
trayIconMenu->addAction(notificationActionText, this, SLOT(toggleDisplayNotifyFromTray()));
trayIconMenu->addAction(tr::lng_quit_from_tray(tr::now), this, SLOT(quitFromTray()));
} }
trayIconMenu->addAction(tr::lng_minimize_to_tray(tr::now), this, SLOT(minimizeToTray()));
trayIconMenu->addAction(notificationActionText, this, SLOT(toggleDisplayNotifyFromTray()));
trayIconMenu->addAction(tr::lng_quit_from_tray(tr::now), this, SLOT(quitFromTray()));
Global::RefWorkMode().setForced(Global::WorkMode().value(), true); Global::RefWorkMode().setForced(Global::WorkMode().value(), true);
psFirstShow(); psFirstShow();
@ -564,7 +560,7 @@ void MainWindow::updateTrayMenu(bool force) {
auto actions = iconMenu->actions(); auto actions = iconMenu->actions();
if (Platform::IsLinux()) { if (Platform::IsLinux()) {
auto minimizeAction = actions.at(1); auto minimizeAction = actions.at(1);
minimizeAction->setDisabled(!isVisible()); minimizeAction->setEnabled(isVisible());
} else { } else {
updateIsActive(0); updateIsActive(0);
auto active = isActive(); auto active = isActive();
@ -588,12 +584,6 @@ void MainWindow::updateTrayMenu(bool force) {
: tr::lng_enable_notifications_from_tray(tr::now); : tr::lng_enable_notifications_from_tray(tr::now);
notificationAction->setText(notificationActionText); notificationAction->setText(notificationActionText);
#ifndef Q_OS_WIN
if (trayIcon && trayIcon->contextMenu() != iconMenu) {
trayIcon->setContextMenu(iconMenu);
}
#endif // !Q_OS_WIN
psTrayMenuUpdated(); psTrayMenuUpdated();
} }

View File

@ -9,10 +9,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "platform/linux/specific_linux.h" #include "platform/linux/specific_linux.h"
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
#include <QDBusInterface>
#endif
namespace Platform { namespace Platform {
namespace DesktopEnvironment { namespace DesktopEnvironment {
namespace { namespace {
@ -44,18 +40,15 @@ Type Compute() {
return Type::Gnome; return Type::Gnome;
} }
return Type::Unity; return Type::Unity;
} else if (list.contains("pantheon")) {
return Type::Pantheon;
} else if (list.contains("gnome")) { } else if (list.contains("gnome")) {
if (list.contains("ubuntu"))
return Type::Ubuntu;
return Type::Gnome; return Type::Gnome;
} else if (list.contains("kde")) { } else if (list.contains("kde")) {
if (kdeSession == qstr("5")) { if (kdeSession == qstr("5")) {
return Type::KDE5; return Type::KDE5;
} }
return Type::KDE4; return Type::KDE4;
} else if (list.contains("mate")) {
return Type::MATE;
} }
} }
@ -70,6 +63,8 @@ Type Compute() {
return Type::KDE4; return Type::KDE4;
} }
return Type::KDE3; return Type::KDE3;
} else if (desktopSession == qstr("mate")) {
return Type::MATE;
} }
} }
@ -96,9 +91,8 @@ Type ComputeAndLog() {
case Type::KDE3: return "KDE3"; case Type::KDE3: return "KDE3";
case Type::KDE4: return "KDE4"; case Type::KDE4: return "KDE4";
case Type::KDE5: return "KDE5"; case Type::KDE5: return "KDE5";
case Type::Ubuntu: return "Ubuntu";
case Type::Unity: return "Unity"; case Type::Unity: return "Unity";
case Type::Pantheon: return "Pantheon"; case Type::MATE: return "MATE";
} }
return QString::number(static_cast<int>(result)); return QString::number(static_cast<int>(result));
}; };
@ -114,20 +108,5 @@ Type Get() {
return result; return result;
} }
bool TryQtTrayIcon() {
return !IsPantheon();
}
bool PreferAppIndicatorTrayIcon() {
return (InSandbox() && !IsKDE())
|| IsUnity()
|| IsUbuntu()
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
|| (IsGnome() && QDBusInterface("org.kde.StatusNotifierWatcher", "/").isValid());
#else
|| IsGnome();
#endif
}
} // namespace DesktopEnvironment } // namespace DesktopEnvironment
} // namespace Platform } // namespace Platform

View File

@ -16,9 +16,8 @@ enum class Type {
KDE3, KDE3,
KDE4, KDE4,
KDE5, KDE5,
Ubuntu,
Unity, Unity,
Pantheon, MATE,
}; };
Type Get(); Type Get();
@ -43,20 +42,13 @@ inline bool IsKDE() {
return IsKDE3() || IsKDE4() || IsKDE5(); return IsKDE3() || IsKDE4() || IsKDE5();
} }
inline bool IsUbuntu() {
return Get() == Type::Ubuntu;
}
inline bool IsUnity() { inline bool IsUnity() {
return Get() == Type::Unity; return Get() == Type::Unity;
} }
inline bool IsPantheon() { inline bool IsMATE() {
return Get() == Type::Pantheon; return Get() == Type::MATE;
} }
bool TryQtTrayIcon();
bool PreferAppIndicatorTrayIcon();
} // namespace DesktopEnvironment } // namespace DesktopEnvironment
} // namespace Platform } // namespace Platform

View File

@ -126,16 +126,6 @@ bool setupGtkBase(QLibrary &lib_gtk) {
DEBUG_LOG(("Checked gtk with gtk_init_check!")); DEBUG_LOG(("Checked gtk with gtk_init_check!"));
return true; return true;
} }
bool setupAppIndicator(QLibrary &lib_indicator) {
if (!load(lib_indicator, "app_indicator_new", app_indicator_new)) return false;
if (!load(lib_indicator, "app_indicator_set_status", app_indicator_set_status)) return false;
if (!load(lib_indicator, "app_indicator_set_menu", app_indicator_set_menu)) return false;
if (!load(lib_indicator, "app_indicator_set_icon_full", app_indicator_set_icon_full)) return false;
DEBUG_LOG(("Library appindicator functions loaded!"));
return true;
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION #endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
} // namespace } // namespace
@ -197,10 +187,6 @@ f_g_type_check_instance_cast g_type_check_instance_cast = nullptr;
f_g_type_check_instance_is_a g_type_check_instance_is_a = nullptr; f_g_type_check_instance_is_a g_type_check_instance_is_a = nullptr;
f_g_signal_connect_data g_signal_connect_data = nullptr; f_g_signal_connect_data g_signal_connect_data = nullptr;
f_g_signal_handler_disconnect g_signal_handler_disconnect = nullptr; f_g_signal_handler_disconnect g_signal_handler_disconnect = nullptr;
f_app_indicator_new app_indicator_new = nullptr;
f_app_indicator_set_status app_indicator_set_status = nullptr;
f_app_indicator_set_menu app_indicator_set_menu = nullptr;
f_app_indicator_set_icon_full app_indicator_set_icon_full = nullptr;
f_gdk_init_check gdk_init_check = nullptr; f_gdk_init_check gdk_init_check = nullptr;
f_gdk_pixbuf_new_from_data gdk_pixbuf_new_from_data = nullptr; f_gdk_pixbuf_new_from_data gdk_pixbuf_new_from_data = nullptr;
f_gdk_pixbuf_new_from_file gdk_pixbuf_new_from_file = nullptr; f_gdk_pixbuf_new_from_file gdk_pixbuf_new_from_file = nullptr;
@ -233,33 +219,14 @@ void start() {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION #ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
bool gtkLoaded = false; bool gtkLoaded = false;
bool indicatorLoaded = false;
bool isWayland = QGuiApplication::platformName().startsWith(qsl("wayland"), Qt::CaseInsensitive); bool isWayland = QGuiApplication::platformName().startsWith(qsl("wayland"), Qt::CaseInsensitive);
QLibrary lib_gtk, lib_indicator; QLibrary lib_gtk;
if (loadLibrary(lib_indicator, "ayatana-appindicator3", 1) || loadLibrary(lib_indicator, "appindicator3", 1)) {
if (loadLibrary(lib_gtk, "gtk-3", 0)) {
gtkLoaded = setupGtkBase(lib_gtk);
indicatorLoaded = setupAppIndicator(lib_indicator);
}
}
if ((!gtkLoaded || !indicatorLoaded) && !isWayland) {
if (loadLibrary(lib_indicator, "ayatana-appindicator", 1) || loadLibrary(lib_indicator, "appindicator", 1)) {
if (loadLibrary(lib_gtk, "gtk-x11-2.0", 0)) {
gtkLoaded = indicatorLoaded = false;
gtkLoaded = setupGtkBase(lib_gtk);
indicatorLoaded = setupAppIndicator(lib_indicator);
}
}
}
// If no appindicator, try at least load gtk. if (loadLibrary(lib_gtk, "gtk-3", 0)) {
if (!gtkLoaded && !indicatorLoaded) { gtkLoaded = setupGtkBase(lib_gtk);
if (loadLibrary(lib_gtk, "gtk-3", 0)) { }
gtkLoaded = setupGtkBase(lib_gtk); if (!gtkLoaded && !isWayland && loadLibrary(lib_gtk, "gtk-x11-2.0", 0)) {
} gtkLoaded = setupGtkBase(lib_gtk);
if (!gtkLoaded && !isWayland && loadLibrary(lib_gtk, "gtk-x11-2.0", 0)) {
gtkLoaded = setupGtkBase(lib_gtk);
}
} }
if (gtkLoaded) { if (gtkLoaded) {
@ -287,7 +254,7 @@ void start() {
load(lib_gtk, "gtk_button_set_label", gtk_button_set_label); load(lib_gtk, "gtk_button_set_label", gtk_button_set_label);
load(lib_gtk, "gtk_button_get_type", gtk_button_get_type); load(lib_gtk, "gtk_button_get_type", gtk_button_get_type);
} else { } else {
LOG(("Could not load gtk-x11-2.0!")); LOG(("Could not load gtk-3 or gtk-x11-2.0!"));
} }
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION #endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
} }

View File

@ -13,11 +13,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
extern "C" { extern "C" {
#undef signals #undef signals
#ifdef TDESKTOP_USE_AYATANA_INDICATORS
#include <libayatana-appindicator/app-indicator.h>
#else
#include <libappindicator/app-indicator.h>
#endif
#include <gtk/gtk.h> #include <gtk/gtk.h>
#include <gdk/gdk.h> #include <gdk/gdk.h>
#define signals public #define signals public
@ -278,18 +273,6 @@ inline gulong g_signal_connect_swapped_helper(gpointer instance, const gchar *de
typedef void (*f_g_signal_handler_disconnect)(gpointer instance, gulong handler_id); typedef void (*f_g_signal_handler_disconnect)(gpointer instance, gulong handler_id);
extern f_g_signal_handler_disconnect g_signal_handler_disconnect; extern f_g_signal_handler_disconnect g_signal_handler_disconnect;
typedef AppIndicator* (*f_app_indicator_new)(const gchar *id, const gchar *icon_name, AppIndicatorCategory category);
extern f_app_indicator_new app_indicator_new;
typedef void (*f_app_indicator_set_status)(AppIndicator *self, AppIndicatorStatus status);
extern f_app_indicator_set_status app_indicator_set_status;
typedef void (*f_app_indicator_set_menu)(AppIndicator *self, GtkMenu *menu);
extern f_app_indicator_set_menu app_indicator_set_menu;
typedef void (*f_app_indicator_set_icon_full)(AppIndicator *self, const gchar *icon_name, const gchar *icon_desc);
extern f_app_indicator_set_icon_full app_indicator_set_icon_full;
typedef gboolean (*f_gdk_init_check)(gint *argc, gchar ***argv); typedef gboolean (*f_gdk_init_check)(gint *argc, gchar ***argv);
extern f_gdk_init_check gdk_init_check; extern f_gdk_init_check gdk_init_check;

View File

@ -15,6 +15,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "history/history.h" #include "history/history.h"
#include "mainwindow.h" #include "mainwindow.h"
#include "core/application.h" #include "core/application.h"
#include "core/sandbox.h"
#include "lang/lang_keys.h" #include "lang/lang_keys.h"
#include "storage/localstorage.h" #include "storage/localstorage.h"
#include "facades.h" #include "facades.h"
@ -22,101 +23,77 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
#include <QtDBus> #include <QtDBus>
#endif #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
#include <QtWidgets/QMenu>
#include <QtWidgets/QAction>
namespace Platform { namespace Platform {
namespace { namespace {
bool noQtTrayIcon = false, tryAppIndicator = false; constexpr auto kDisableTrayCounter = "TDESKTOP_DISABLE_TRAY_COUNTER"_cs;
bool useGtkBase = false, useAppIndicator = false, useStatusIcon = false, trayIconChecked = false, useUnityCount = false; constexpr auto kTrayIconName = "telegram"_cs;
constexpr auto kPanelTrayIconName = "telegram-panel"_cs;
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION constexpr auto kMutePanelTrayIconName = "telegram-mute-panel"_cs;
AppIndicator *_trayIndicator = 0; constexpr auto kAttentionPanelTrayIconName = "telegram-attention-panel"_cs;
GtkStatusIcon *_trayIcon = 0; constexpr auto kSNIWatcherService = "org.kde.StatusNotifierWatcher"_cs;
GtkWidget *_trayMenu = 0; constexpr auto kTrayIconFilename = "tdesktop-trayicon-XXXXXX.png"_cs;
GdkPixbuf *_trayPixbuf = 0;
QByteArray _trayPixbufData;
QList<QPair<GtkWidget*, QObject*> > _trayItems;
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
int32 _trayIconSize = 48; int32 _trayIconSize = 48;
bool _trayIconMuted = true; bool _trayIconMuted = true;
int32 _trayIconCount = 0; int32 _trayIconCount = 0;
QImage _trayIconImageBack, _trayIconImage; QImage _trayIconImageBack, _trayIconImage;
QString _trayIconThemeName, _trayIconName; QString _trayIconThemeName, _trayIconName;
QString _desktopFile;
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
QString _dbusPath = "/"; bool UseUnityCount = false;
#endif QString UnityCountDesktopFile;
QString UnityCountDBusPath = "/";
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
void _trayIconPopup(GtkStatusIcon *status_icon, guint button, guint32 activate_time, gpointer popup_menu) {
Libs::gtk_menu_popup(Libs::gtk_menu_cast(popup_menu), NULL, NULL, Libs::gtk_status_icon_position_menu, status_icon, button, activate_time);
}
void _trayIconActivate(GtkStatusIcon *status_icon, gpointer popup_menu) {
if (App::wnd()->isActiveWindow() && App::wnd()->isVisible()) {
Libs::gtk_menu_popup(Libs::gtk_menu_cast(popup_menu), NULL, NULL, Libs::gtk_status_icon_position_menu, status_icon, 0, Libs::gtk_get_current_event_time());
} else {
App::wnd()->showFromTray();
}
}
gboolean _trayIconResized(GtkStatusIcon *status_icon, gint size, gpointer popup_menu) {
_trayIconSize = size;
if (Global::started()) Notify::unreadCounterUpdated();
return FALSE;
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
#define QT_RED 0 #define QT_RED 0
#define QT_GREEN 1 #define QT_GREEN 1
#define QT_BLUE 2 #define QT_BLUE 2
#define QT_ALPHA 3 #define QT_ALPHA 3
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION QString GetTrayIconName() {
#define GTK_RED 2 const auto counter = Core::App().unreadBadge();
#define GTK_GREEN 1 const auto muted = Core::App().unreadBadgeMuted();
#define GTK_BLUE 0
#define GTK_ALPHA 3
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
QImage _trayIconImageGen(bool useSystemIcon) { return (counter > 0)
? (muted
? kMutePanelTrayIconName.utf16()
: kAttentionPanelTrayIconName.utf16())
: kPanelTrayIconName.utf16();
}
QImage TrayIconImageGen() {
const auto counter = Core::App().unreadBadge(); const auto counter = Core::App().unreadBadge();
const auto muted = Core::App().unreadBadgeMuted(); const auto muted = Core::App().unreadBadgeMuted();
const auto counterSlice = (counter >= 1000) const auto counterSlice = (counter >= 1000)
? (1000 + (counter % 100)) ? (1000 + (counter % 100))
: counter; : counter;
QString iconThemeName = QIcon::themeName(); const auto iconThemeName = QIcon::themeName();
QString iconName = (counter > 0) const auto iconName = GetTrayIconName();
? (muted ? "telegram-mute-panel" : "telegram-attention-panel")
: "telegram-panel";
if (_trayIconImage.isNull() || _trayIconImage.width() != _trayIconSize if (_trayIconImage.isNull()
|| (useSystemIcon && (iconThemeName != _trayIconThemeName || _trayIconImage.width() != _trayIconSize
|| iconName != _trayIconName)) || iconThemeName != _trayIconThemeName
|| muted != _trayIconMuted || counterSlice != _trayIconCount) { || iconName != _trayIconName
|| muted != _trayIconMuted
|| counterSlice != _trayIconCount) {
if (_trayIconImageBack.isNull() if (_trayIconImageBack.isNull()
|| _trayIconImageBack.width() != _trayIconSize || _trayIconImageBack.width() != _trayIconSize
|| iconThemeName != _trayIconThemeName || iconThemeName != _trayIconThemeName
|| iconName != _trayIconName) { || iconName != _trayIconName) {
_trayIconImageBack = Core::App().logo(); _trayIconImageBack = Core::App().logo();
if (useSystemIcon) { _trayIconImageBack = QIcon::fromTheme(
_trayIconImageBack = QIcon::fromTheme( iconName,
iconName, QIcon::fromTheme(
QIcon::fromTheme( kTrayIconName.utf16(),
"telegram", QIcon(QPixmap::fromImage(_trayIconImageBack)))
QIcon(QPixmap::fromImage(_trayIconImageBack))) ).pixmap(_trayIconSize, _trayIconSize).toImage();
).pixmap(_trayIconSize, _trayIconSize).toImage();
}
int w = _trayIconImageBack.width(), auto w = _trayIconImageBack.width(),
h = _trayIconImageBack.height(); h = _trayIconImageBack.height();
if (w != _trayIconSize || h != _trayIconSize) { if (w != _trayIconSize || h != _trayIconSize) {
@ -132,8 +109,8 @@ QImage _trayIconImageGen(bool useSystemIcon) {
w = _trayIconImageBack.width(); w = _trayIconImageBack.width();
h = _trayIconImageBack.height(); h = _trayIconImageBack.height();
int perline = _trayIconImageBack.bytesPerLine(); const auto perline = _trayIconImageBack.bytesPerLine();
uchar *bytes = _trayIconImageBack.bits(); auto *bytes = _trayIconImageBack.bits();
for (int32 y = 0; y < h; ++y) { for (int32 y = 0; y < h; ++y) {
for (int32 x = 0; x < w; ++x) { for (int32 x = 0; x < w; ++x) {
@ -189,81 +166,66 @@ QImage _trayIconImageGen(bool useSystemIcon) {
return _trayIconImage; return _trayIconImage;
} }
QString _trayIconImageFile() { #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
const auto counter = Core::App().unreadBadge(); static bool NeedTrayIconFile() {
const auto muted = Core::App().unreadBadgeMuted(); // Hack for indicator-application, which doesn't handle icons sent across D-Bus:
const auto counterSlice = (counter >= 1000) // save the icon to a temp file and set the icon name to that filename.
? (1000 + (counter % 100)) static const auto TrayIconFileNeeded = [&] {
: counter; auto necessary = false;
const auto session = QDBusConnection::sessionBus();
QString iconThemeName = QIcon::themeName(); const auto pid = session.interface()
->servicePid(kSNIWatcherService.utf16()).value();
QString name = cWorkingDir() + qsl("tdata/ticons/icon%1_%2_%3_%4.png") const auto processName = ProcessNameByPID(QString::number(pid));
.arg(muted ? "mute" : "") necessary = processName.endsWith(
.arg(iconThemeName) qsl("indicator-application-service"));
.arg(_trayIconSize) if (!necessary) {
.arg(counterSlice); // Accessing to process name might be not allowed if the application
// is confined, thus we can just rely on the current desktop in use
QFileInfo info(name); necessary = DesktopEnvironment::IsUnity()
if (info.exists()) return name; || DesktopEnvironment::IsMATE();
QImage img = _trayIconImageGen(false);
if (img.save(name, "PNG")) return name;
QDir dir(info.absoluteDir());
if (!dir.exists()) {
dir.mkpath(dir.absolutePath());
if (img.save(name, "PNG")) return name;
}
return QString();
}
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
void loadPixbuf(QImage image) {
int w = image.width(), h = image.height(), perline = image.bytesPerLine(), s = image.byteCount();
_trayPixbufData.resize(w * h * 4);
uchar *result = (uchar*)_trayPixbufData.data(), *bytes = image.bits();
for (int32 y = 0; y < h; ++y) {
for (int32 x = 0; x < w; ++x) {
int32 offset = (y * w + x) * 4, srcoff = y * perline + x * 4;
result[offset + GTK_RED ] = bytes[srcoff + QT_RED ];
result[offset + GTK_GREEN] = bytes[srcoff + QT_GREEN];
result[offset + GTK_BLUE ] = bytes[srcoff + QT_BLUE ];
result[offset + GTK_ALPHA] = bytes[srcoff + QT_ALPHA];
} }
} return necessary;
}();
if (_trayPixbuf) Libs::g_object_unref(_trayPixbuf); return TrayIconFileNeeded;
_trayPixbuf = Libs::gdk_pixbuf_new_from_data(result, GDK_COLORSPACE_RGB, true, 8, w, h, w * 4, 0, 0);
} }
void _trayMenuCallback(GtkMenu *menu, gpointer data) { static inline QString TrayIconFileTemplate() {
for (int32 i = 0, l = _trayItems.size(); i < l; ++i) { static const auto TempFileTemplate = AppRuntimeDirectory()
if ((void*)_trayItems.at(i).first == (void*)menu) { + kTrayIconFilename.utf16();
QMetaObject::invokeMethod(_trayItems.at(i).second, "triggered"); return TempFileTemplate;
}
}
} }
static gboolean _trayIconCheck(gpointer/* pIn*/) { std::unique_ptr<QTemporaryFile> TrayIconFile(
if (useStatusIcon && !trayIconChecked) { const QPixmap &icon, QObject *parent) {
if (Libs::gtk_status_icon_is_embedded(_trayIcon)) { auto ret = std::make_unique<QTemporaryFile>(
trayIconChecked = true; TrayIconFileTemplate(),
cSetSupportTray(true); parent);
if (Global::started()) { ret->open();
Global::RefWorkMode().setForced(Global::WorkMode().value(), true); icon.save(ret.get());
} ret->close();
if (App::wnd()) { return ret;
Notify::unreadCounterUpdated();
App::wnd()->updateTrayMenu();
}
}
}
return FALSE;
} }
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION bool IsSNIAvailable() {
static const auto SNIAvailable = [&] {
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
QDBusInterface systrayHost(
kSNIWatcherService.utf16(),
qsl("/StatusNotifierWatcher"),
kSNIWatcherService.utf16());
return systrayHost.isValid()
&& systrayHost
.property("IsStatusNotifierHostRegistered")
.toBool();
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
return false;
}();
return SNIAvailable;
}
quint32 djbStringHash(QString string) { quint32 djbStringHash(QString string) {
quint32 hash = 5381; quint32 hash = 5381;
@ -278,71 +240,106 @@ quint32 djbStringHash(QString string) {
MainWindow::MainWindow(not_null<Window::Controller*> controller) MainWindow::MainWindow(not_null<Window::Controller*> controller)
: Window::MainWindow(controller) { : Window::MainWindow(controller) {
connect(&_psCheckStatusIconTimer, SIGNAL(timeout()), this, SLOT(psStatusIconCheck()));
_psCheckStatusIconTimer.setSingleShot(false);
connect(&_psUpdateIndicatorTimer, SIGNAL(timeout()), this, SLOT(psUpdateIndicator()));
_psUpdateIndicatorTimer.setSingleShot(true);
} }
bool MainWindow::hasTrayIcon() const { bool MainWindow::hasTrayIcon() const {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
return trayIcon || ((useAppIndicator || (useStatusIcon && trayIconChecked)) && (Global::WorkMode().value() != dbiwmWindowOnly)); return trayIcon || _sniTrayIcon;
#else #else
return trayIcon; return trayIcon;
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
}
void MainWindow::psStatusIconCheck() {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
_trayIconCheck(0);
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
if (cSupportTray() || !--_psCheckStatusIconLeft) {
_psCheckStatusIconTimer.stop();
return;
}
} }
void MainWindow::psShowTrayMenu() { void MainWindow::psShowTrayMenu() {
if (!IsSNIAvailable()) {
_trayIconMenuXEmbed->popup(QCursor::pos());
}
} }
void MainWindow::psTrayMenuUpdated() { void MainWindow::psTrayMenuUpdated() {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
if (noQtTrayIcon && (useAppIndicator || useStatusIcon)) { if (IsSNIAvailable()) {
const QList<QAction*> &actions = trayIconMenu->actions(); if (_sniTrayIcon && trayIconMenu) {
if (_trayItems.isEmpty()) { _sniTrayIcon->setContextMenu(trayIconMenu);
DEBUG_LOG(("Creating tray menu!"));
for (int32 i = 0, l = actions.size(); i != l; ++i) {
GtkWidget *item = Libs::gtk_menu_item_new_with_label(actions.at(i)->text().toUtf8());
Libs::gtk_menu_shell_append(Libs::gtk_menu_shell_cast(_trayMenu), item);
Libs::g_signal_connect_helper(item, "activate", G_CALLBACK(_trayMenuCallback), this);
Libs::gtk_widget_show(item);
Libs::gtk_widget_set_sensitive(item, actions.at(i)->isEnabled());
_trayItems.push_back(qMakePair(item, actions.at(i)));
}
} else {
DEBUG_LOG(("Updating tray menu!"));
for (int32 i = 0, l = actions.size(); i != l; ++i) {
if (i < _trayItems.size()) {
Libs::gtk_menu_item_set_label(reinterpret_cast<GtkMenuItem*>(_trayItems.at(i).first), actions.at(i)->text().toUtf8());
Libs::gtk_widget_set_sensitive(_trayItems.at(i).first, actions.at(i)->isEnabled());
}
}
} }
} }
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
} }
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
void MainWindow::setSNITrayIcon(
const QIcon &icon, const QPixmap &iconPixmap) {
if (!NeedTrayIconFile()) {
_sniTrayIcon->setIconByPixmap(icon);
_sniTrayIcon->setToolTipIconByPixmap(icon);
}
if (qEnvironmentVariableIsSet(kDisableTrayCounter.utf8())) {
const auto iconName = GetTrayIconName();
_sniTrayIcon->setIconByName(iconName);
_sniTrayIcon->setToolTipIconByName(iconName);
} else if (NeedTrayIconFile()) {
_trayIconFile = TrayIconFile(iconPixmap, this);
if (_trayIconFile) {
_sniTrayIcon->setIconByName(_trayIconFile->fileName());
_sniTrayIcon->setToolTipIconByName(_trayIconFile->fileName());
}
}
}
void MainWindow::attachToSNITrayIcon() {
_sniTrayIcon->setToolTipTitle(AppName.utf16());
connect(_sniTrayIcon,
&StatusNotifierItem::activateRequested,
this,
[=](const QPoint &) {
Core::Sandbox::Instance().customEnterFromEventLoop([&] {
handleTrayIconActication(QSystemTrayIcon::Trigger);
});
});
connect(_sniTrayIcon,
&StatusNotifierItem::secondaryActivateRequested,
this,
[=](const QPoint &) {
Core::Sandbox::Instance().customEnterFromEventLoop([&] {
handleTrayIconActication(QSystemTrayIcon::MiddleClick);
});
});
updateTrayMenu();
}
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
void MainWindow::psSetupTrayIcon() { void MainWindow::psSetupTrayIcon() {
if (noQtTrayIcon) { const auto iconPixmap = QPixmap::fromImage(TrayIconImageGen());
if (!cSupportTray()) return; const auto icon = QIcon(iconPixmap);
if (IsSNIAvailable()) {
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
LOG(("Using SNI tray icon."));
if (!_sniTrayIcon) {
_sniTrayIcon = new StatusNotifierItem(
QCoreApplication::applicationName(),
this);
_sniTrayIcon->setTitle(QCoreApplication::applicationName());
setSNITrayIcon(icon, iconPixmap);
attachToSNITrayIcon();
}
updateIconCounters(); updateIconCounters();
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
} else { } else {
LOG(("Using Qt tray icon.")); LOG(("Using Qt tray icon."));
if (!_trayIconMenuXEmbed) {
_trayIconMenuXEmbed = new Ui::PopupMenu(nullptr, trayIconMenu);
_trayIconMenuXEmbed->deleteOnHide(false);
}
if (!trayIcon) { if (!trayIcon) {
trayIcon = new QSystemTrayIcon(this); trayIcon = new QSystemTrayIcon(this);
trayIcon->setIcon(QIcon(QPixmap::fromImage(_trayIconImageGen(true)))); trayIcon->setIcon(icon);
attachToTrayIcon(trayIcon); attachToTrayIcon(trayIcon);
} }
@ -356,14 +353,14 @@ void MainWindow::workmodeUpdated(DBIWorkMode mode) {
if (!cSupportTray()) return; if (!cSupportTray()) return;
if (mode == dbiwmWindowOnly) { if (mode == dbiwmWindowOnly) {
if (noQtTrayIcon) { if (IsSNIAvailable()) {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
if (useAppIndicator) { if (_sniTrayIcon) {
Libs::app_indicator_set_status(_trayIndicator, APP_INDICATOR_STATUS_PASSIVE); _sniTrayIcon->setContextMenu(0);
} else if (useStatusIcon) { _sniTrayIcon->deleteLater();
Libs::gtk_status_icon_set_visible(_trayIcon, false);
} }
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION _sniTrayIcon = 0;
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
} else { } else {
if (trayIcon) { if (trayIcon) {
trayIcon->setContextMenu(0); trayIcon->setContextMenu(0);
@ -372,35 +369,10 @@ void MainWindow::workmodeUpdated(DBIWorkMode mode) {
trayIcon = 0; trayIcon = 0;
} }
} else { } else {
if (noQtTrayIcon) { psSetupTrayIcon();
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
if (useAppIndicator) {
Libs::app_indicator_set_status(_trayIndicator, APP_INDICATOR_STATUS_ACTIVE);
} else if (useStatusIcon) {
Libs::gtk_status_icon_set_visible(_trayIcon, true);
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
} else {
psSetupTrayIcon();
}
} }
} }
void MainWindow::psUpdateIndicator() {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
_psUpdateIndicatorTimer.stop();
_psLastIndicatorUpdate = crl::now();
QFileInfo iconFile(_trayIconImageFile());
if (iconFile.exists()) {
QByteArray path = QFile::encodeName(iconFile.absoluteFilePath()), name = QFile::encodeName(iconFile.fileName());
name = name.mid(0, name.size() - 4);
Libs::app_indicator_set_icon_full(_trayIndicator, path.constData(), name);
} else {
useAppIndicator = false;
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
}
void MainWindow::unreadCounterChangedHook() { void MainWindow::unreadCounterChangedHook() {
setWindowTitle(titleText()); setWindowTitle(titleText());
updateIconCounters(); updateIconCounters();
@ -410,208 +382,91 @@ void MainWindow::updateIconCounters() {
updateWindowIcon(); updateWindowIcon();
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
if (useUnityCount) { if (UseUnityCount) {
const auto counter = Core::App().unreadBadge(); const auto counter = Core::App().unreadBadge();
QVariantMap dbusUnityProperties; QVariantMap dbusUnityProperties;
if (counter > 0) { if (counter > 0) {
// Gnome requires that count is a 64bit integer // Gnome requires that count is a 64bit integer
dbusUnityProperties.insert("count", (qint64) ((counter > 9999) ? 9999 : (counter))); dbusUnityProperties.insert(
"count",
(qint64) ((counter > 9999)
? 9999
: (counter)));
dbusUnityProperties.insert("count-visible", true); dbusUnityProperties.insert("count-visible", true);
} else { } else {
dbusUnityProperties.insert("count-visible", false); dbusUnityProperties.insert("count-visible", false);
} }
QDBusMessage signal = QDBusMessage::createSignal(_dbusPath, "com.canonical.Unity.LauncherEntry", "Update"); QDBusMessage signal = QDBusMessage::createSignal(
signal << "application://" + _desktopFile; UnityCountDBusPath,
"com.canonical.Unity.LauncherEntry",
"Update");
signal << "application://" + UnityCountDesktopFile;
signal << dbusUnityProperties; signal << dbusUnityProperties;
QDBusConnection::sessionBus().send(signal); QDBusConnection::sessionBus().send(signal);
} }
#endif #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
if (noQtTrayIcon) { const auto iconPixmap = QPixmap::fromImage(TrayIconImageGen());
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION const auto icon = QIcon(iconPixmap);
if (useAppIndicator) {
if (crl::now() > _psLastIndicatorUpdate + 1000) { if (IsSNIAvailable()) {
psUpdateIndicator(); #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
} else if (!_psUpdateIndicatorTimer.isActive()) { if (_sniTrayIcon) {
_psUpdateIndicatorTimer.start(100); setSNITrayIcon(icon, iconPixmap);
}
} else if (useStatusIcon && trayIconChecked) {
loadPixbuf(_trayIconImageGen(true));
Libs::gtk_status_icon_set_from_pixbuf(_trayIcon, _trayPixbuf);
} }
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
} else if (trayIcon) { } else if (trayIcon) {
trayIcon->setIcon(QIcon(QPixmap::fromImage(_trayIconImageGen(true)))); trayIcon->setIcon(icon);
} }
} }
void MainWindow::LibsLoaded() { void MainWindow::LibsLoaded() {
noQtTrayIcon = !DesktopEnvironment::TryQtTrayIcon(); #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION qDBusRegisterMetaType<ToolTip>();
tryAppIndicator = DesktopEnvironment::PreferAppIndicatorTrayIcon(); qDBusRegisterMetaType<IconPixmap>();
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION qDBusRegisterMetaType<IconPixmapList>();
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
LOG(("Tray Icon: Try Qt = %1, Prefer appindicator = %2").arg(Logs::b(!noQtTrayIcon)).arg(Logs::b(tryAppIndicator))); if (!IsSNIAvailable()) {
_trayIconSize = 22;
if (noQtTrayIcon) cSetSupportTray(false);
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
useGtkBase = (Libs::gtk_init_check != nullptr)
&& (Libs::gtk_menu_new != nullptr)
&& (Libs::gtk_menu_get_type != nullptr)
&& (Libs::gtk_menu_item_new_with_label != nullptr)
&& (Libs::gtk_menu_item_set_label != nullptr)
&& (Libs::gtk_menu_shell_append != nullptr)
&& (Libs::gtk_menu_shell_get_type != nullptr)
&& (Libs::gtk_widget_show != nullptr)
&& (Libs::gtk_widget_get_toplevel != nullptr)
&& (Libs::gtk_widget_get_visible != nullptr)
&& (Libs::gtk_widget_set_sensitive != nullptr)
&& (Libs::g_type_check_instance_cast != nullptr)
&& (Libs::g_signal_connect_data != nullptr)
&& (Libs::g_object_ref_sink != nullptr)
&& (Libs::g_object_unref != nullptr);
useAppIndicator = useGtkBase
&& (Libs::app_indicator_new != nullptr)
&& (Libs::app_indicator_set_status != nullptr)
&& (Libs::app_indicator_set_menu != nullptr)
&& (Libs::app_indicator_set_icon_full != nullptr);
if (tryAppIndicator && useGtkBase && useAppIndicator) {
noQtTrayIcon = true;
cSetSupportTray(false);
} }
useStatusIcon = (Libs::gdk_init_check != nullptr)
&& (Libs::gdk_pixbuf_new_from_data != nullptr)
&& (Libs::gtk_status_icon_new_from_pixbuf != nullptr)
&& (Libs::gtk_status_icon_set_from_pixbuf != nullptr)
&& (Libs::gtk_status_icon_new_from_file != nullptr)
&& (Libs::gtk_status_icon_set_from_file != nullptr)
&& (Libs::gtk_status_icon_set_title != nullptr)
&& (Libs::gtk_status_icon_set_tooltip_text != nullptr)
&& (Libs::gtk_status_icon_set_visible != nullptr)
&& (Libs::gtk_status_icon_is_embedded != nullptr)
&& (Libs::gtk_status_icon_get_geometry != nullptr)
&& (Libs::gtk_status_icon_position_menu != nullptr)
&& (Libs::gtk_menu_popup != nullptr)
&& (Libs::gtk_get_current_event_time != nullptr)
&& (Libs::g_idle_add != nullptr);
if (useStatusIcon) {
DEBUG_LOG(("Status icon api loaded!"));
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
}
void MainWindow::psCreateTrayIcon() {
if (!noQtTrayIcon) {
LOG(("Tray Icon: Using Qt tray icon, available: %1").arg(Logs::b(QSystemTrayIcon::isSystemTrayAvailable())));
cSetSupportTray(QSystemTrayIcon::isSystemTrayAvailable());
return;
}
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION
if (useAppIndicator) {
DEBUG_LOG(("Trying to create AppIndicator"));
_trayMenu = Libs::gtk_menu_new();
if (_trayMenu) {
DEBUG_LOG(("Created gtk menu for appindicator!"));
QFileInfo iconFile(_trayIconImageFile());
if (iconFile.exists()) {
QByteArray path = QFile::encodeName(iconFile.absoluteFilePath());
_trayIndicator = Libs::app_indicator_new("Telegram Desktop", path.constData(), APP_INDICATOR_CATEGORY_APPLICATION_STATUS);
if (_trayIndicator) {
LOG(("Tray Icon: Using appindicator tray icon."));
} else {
DEBUG_LOG(("Failed to app_indicator_new()!"));
}
} else {
useAppIndicator = false;
DEBUG_LOG(("Failed to create image file!"));
}
} else {
DEBUG_LOG(("Failed to gtk_menu_new()!"));
}
if (_trayMenu && _trayIndicator) {
Libs::app_indicator_set_status(_trayIndicator, APP_INDICATOR_STATUS_ACTIVE);
Libs::app_indicator_set_menu(_trayIndicator, Libs::gtk_menu_cast(_trayMenu));
useStatusIcon = false;
} else {
DEBUG_LOG(("AppIndicator failed!"));
useAppIndicator = false;
}
}
if (useStatusIcon) {
if (Libs::gdk_init_check(0, 0)) {
if (!_trayMenu) _trayMenu = Libs::gtk_menu_new();
if (_trayMenu) {
loadPixbuf(_trayIconImageGen(true));
_trayIcon = Libs::gtk_status_icon_new_from_pixbuf(_trayPixbuf);
if (_trayIcon) {
LOG(("Tray Icon: Using GTK status tray icon."));
Libs::g_signal_connect_helper(_trayIcon, "popup-menu", GCallback(_trayIconPopup), _trayMenu);
Libs::g_signal_connect_helper(_trayIcon, "activate", GCallback(_trayIconActivate), _trayMenu);
Libs::g_signal_connect_helper(_trayIcon, "size-changed", GCallback(_trayIconResized), _trayMenu);
Libs::gtk_status_icon_set_title(_trayIcon, "Telegram Desktop");
Libs::gtk_status_icon_set_tooltip_text(_trayIcon, "Telegram Desktop");
Libs::gtk_status_icon_set_visible(_trayIcon, true);
} else {
useStatusIcon = false;
}
} else {
useStatusIcon = false;
}
} else {
useStatusIcon = false;
}
}
if (!useStatusIcon && !useAppIndicator) {
LOG(("Tray Icon: Not able to use any tray icon :("));
if (_trayMenu) {
Libs::g_object_ref_sink(_trayMenu);
Libs::g_object_unref(_trayMenu);
_trayMenu = nullptr;
}
}
cSetSupportTray(useAppIndicator);
if (useStatusIcon) {
Libs::g_idle_add((GSourceFunc)_trayIconCheck, 0);
_psCheckStatusIconTimer.start(100);
} else {
workmodeUpdated(Global::WorkMode().value());
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
} }
void MainWindow::psFirstShow() { void MainWindow::psFirstShow() {
psCreateTrayIcon(); const auto trayAvailable = IsSNIAvailable()
|| QSystemTrayIcon::isSystemTrayAvailable();
LOG(("System tray available: %1").arg(Logs::b(trayAvailable)));
cSetSupportTray(trayAvailable);
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
if (QDBusInterface("com.canonical.Unity", "/").isValid()) { if (QDBusInterface("com.canonical.Unity", "/").isValid()) {
std::vector<QString> possibleDesktopFiles = { const std::vector<QString> possibleDesktopFiles = {
GetLauncherFilename(), GetLauncherFilename(),
"Telegram.desktop" "Telegram.desktop"
}; };
for (auto it = possibleDesktopFiles.begin(); it != possibleDesktopFiles.end(); it++) { for (auto it = possibleDesktopFiles.begin();
if (!QStandardPaths::locate(QStandardPaths::ApplicationsLocation, *it).isEmpty()) { it != possibleDesktopFiles.end(); it++) {
_desktopFile = *it; if (!QStandardPaths::locate(
LOG(("Found Unity Launcher entry %1!").arg(_desktopFile)); QStandardPaths::ApplicationsLocation, *it).isEmpty()) {
useUnityCount = true; UnityCountDesktopFile = *it;
LOG(("Found Unity Launcher entry %1!")
.arg(UnityCountDesktopFile));
UseUnityCount = true;
break; break;
} }
} }
if (!useUnityCount) { if (!UseUnityCount) {
LOG(("Could not get Unity Launcher entry!")); LOG(("Could not get Unity Launcher entry!"));
} }
_dbusPath = "/com/canonical/unity/launcherentry/" + QString::number(djbStringHash("application://" + _desktopFile)); UnityCountDBusPath = "/com/canonical/unity/launcherentry/"
+ QString::number(
djbStringHash("application://" + UnityCountDesktopFile));
} else { } else {
LOG(("Not using Unity Launcher count.")); LOG(("Not using Unity Launcher count."));
} }
#endif #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
show(); show();
if (cWindowPos().maximized) { if (cWindowPos().maximized) {
@ -619,13 +474,15 @@ void MainWindow::psFirstShow() {
setWindowState(Qt::WindowMaximized); setWindowState(Qt::WindowMaximized);
} }
if ((cLaunchMode() == LaunchModeAutoStart && cStartMinimized()) || cStartInTray()) { if ((cLaunchMode() == LaunchModeAutoStart && cStartMinimized())
|| cStartInTray()) {
// If I call hide() synchronously here after show() then on Ubuntu 14.04 // If I call hide() synchronously here after show() then on Ubuntu 14.04
// it will show a window frame with transparent window body, without content. // it will show a window frame with transparent window body, without content.
// And to be able to "Show from tray" one more hide() will be required. // And to be able to "Show from tray" one more hide() will be required.
crl::on_main(this, [=] { crl::on_main(this, [=] {
setWindowState(Qt::WindowMinimized); setWindowState(Qt::WindowMinimized);
if (Global::WorkMode().value() == dbiwmTrayOnly || Global::WorkMode().value() == dbiwmWindowAndTray) { if (Global::WorkMode().value() == dbiwmTrayOnly
|| Global::WorkMode().value() == dbiwmWindowAndTray) {
hide(); hide();
} else { } else {
show(); show();
@ -637,25 +494,11 @@ void MainWindow::psFirstShow() {
} }
MainWindow::~MainWindow() { MainWindow::~MainWindow() {
#ifndef TDESKTOP_DISABLE_GTK_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
if (_trayIcon) { delete _sniTrayIcon;
Libs::g_object_unref(_trayIcon); #endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
_trayIcon = nullptr;
} delete _trayIconMenuXEmbed;
if (_trayPixbuf) {
Libs::g_object_unref(_trayPixbuf);
_trayPixbuf = nullptr;
}
if (_trayMenu) {
Libs::g_object_ref_sink(_trayMenu);
Libs::g_object_unref(_trayMenu);
_trayMenu = nullptr;
}
if (_trayIndicator) {
Libs::g_object_unref(_trayIndicator);
_trayIndicator = nullptr;
}
#endif // !TDESKTOP_DISABLE_GTK_INTEGRATION
} }
} // namespace Platform } // namespace Platform

View File

@ -9,7 +9,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "platform/platform_main_window.h" #include "platform/platform_main_window.h"
#include <QtCore/QTimer> #include "ui/widgets/popup_menu.h"
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
#include "statusnotifieritem.h"
#include <QtCore/QTemporaryFile>
#endif
namespace Platform { namespace Platform {
@ -21,7 +26,12 @@ public:
void psFirstShow(); void psFirstShow();
virtual QImage iconWithCounter(int size, int count, style::color bg, style::color fg, bool smallIcon) = 0; virtual QImage iconWithCounter(
int size,
int count,
style::color bg,
style::color fg,
bool smallIcon) = 0;
static void LibsLoaded(); static void LibsLoaded();
@ -30,9 +40,6 @@ public:
public slots: public slots:
void psShowTrayMenu(); void psShowTrayMenu();
void psStatusIconCheck();
void psUpdateIndicator();
protected: protected:
void unreadCounterChangedHook() override; void unreadCounterChangedHook() override;
@ -46,17 +53,26 @@ protected:
void psTrayMenuUpdated(); void psTrayMenuUpdated();
void psSetupTrayIcon(); void psSetupTrayIcon();
virtual void placeSmallCounter(QImage &img, int size, int count, style::color bg, const QPoint &shift, style::color color) = 0; virtual void placeSmallCounter(
QImage &img,
int size,
int count,
style::color bg,
const QPoint &shift,
style::color color) = 0;
private: private:
Ui::PopupMenu *_trayIconMenuXEmbed = nullptr;
void updateIconCounters(); void updateIconCounters();
void psCreateTrayIcon();
QTimer _psCheckStatusIconTimer; #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
int _psCheckStatusIconLeft = 100; StatusNotifierItem *_sniTrayIcon = nullptr;
std::unique_ptr<QTemporaryFile> _trayIconFile = nullptr;
QTimer _psUpdateIndicatorTimer; void setSNITrayIcon(const QIcon &icon, const QPixmap &iconPixmap);
crl::time _psLastIndicatorUpdate = 0; void attachToSNITrayIcon();
#endif // !TDESKTOP_DISABLE_DBUS_INTEGRATION
}; };

View File

@ -44,7 +44,7 @@ namespace {
constexpr auto kDesktopFile = ":/misc/telegramdesktop.desktop"_cs; constexpr auto kDesktopFile = ":/misc/telegramdesktop.desktop"_cs;
bool XDGDesktopPortalIsPresent = false; bool XDGDesktopPortalPresent = false;
#ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION #ifndef TDESKTOP_DISABLE_DBUS_INTEGRATION
void SandboxAutostart(bool autostart) { void SandboxAutostart(bool autostart) {
@ -196,13 +196,13 @@ bool InSnap() {
} }
bool IsXDGDesktopPortalPresent() { bool IsXDGDesktopPortalPresent() {
return XDGDesktopPortalIsPresent; return XDGDesktopPortalPresent;
}; }
QString CurrentExecutablePath(int argc, char *argv[]) { QString ProcessNameByPID(const QString &pid) {
constexpr auto kMaxPath = 1024; constexpr auto kMaxPath = 1024;
char result[kMaxPath] = { 0 }; char result[kMaxPath] = { 0 };
auto count = readlink("/proc/self/exe", result, kMaxPath); auto count = readlink("/proc/" + pid.toLatin1() + "/exe", result, kMaxPath);
if (count > 0) { if (count > 0) {
auto filename = QFile::decodeName(result); auto filename = QFile::decodeName(result);
auto deletedPostfix = qstr(" (deleted)"); auto deletedPostfix = qstr(" (deleted)");
@ -212,26 +212,53 @@ QString CurrentExecutablePath(int argc, char *argv[]) {
return filename; return filename;
} }
return QString();
}
QString CurrentExecutablePath(int argc, char *argv[]) {
const auto processName = ProcessNameByPID(qsl("self"));
// Fallback to the first command line argument. // Fallback to the first command line argument.
return argc ? QFile::decodeName(argv[0]) : QString(); return !processName.isEmpty()
? processName
: argc
? QFile::decodeName(argv[0])
: QString();
}
QString AppRuntimeDirectory() {
static const auto RuntimeDirectory = [&] {
auto runtimeDir = QStandardPaths::writableLocation(
QStandardPaths::RuntimeLocation);
if (InSandbox()) {
runtimeDir += qsl("/app/")
+ QString::fromLatin1(qgetenv("FLATPAK_ID"));
}
if (!QFileInfo::exists(runtimeDir)) { // non-systemd distros
runtimeDir = QDir::tempPath();
}
if (runtimeDir.isEmpty()) {
runtimeDir = qsl("/tmp/");
}
if (!runtimeDir.endsWith('/')) {
runtimeDir += '/';
}
return runtimeDir;
}();
return RuntimeDirectory;
} }
QString SingleInstanceLocalServerName(const QString &hash) { QString SingleInstanceLocalServerName(const QString &hash) {
const auto runtimeDir = QStandardPaths::writableLocation( if (InSandbox() || InSnap()) {
QStandardPaths::RuntimeLocation); return AppRuntimeDirectory() + hash;
} else {
if (InSandbox()) { return AppRuntimeDirectory() + hash + '-' + cGUIDStr();
return runtimeDir
+ qsl("/app/")
+ QString::fromLatin1(qgetenv("FLATPAK_ID"))
+ '/' + hash;
} else if (QFileInfo::exists(runtimeDir) && InSnap()) {
return runtimeDir + '/' + hash;
} else if (QFileInfo::exists(runtimeDir)) {
return runtimeDir + '/' + hash + '-' + cGUIDStr();
} else { // non-systemd distros
return QStandardPaths::writableLocation(QStandardPaths::TempLocation)
+ '/' + hash + '-' + cGUIDStr();
} }
} }
@ -381,12 +408,12 @@ void start() {
#if !defined(TDESKTOP_DISABLE_DBUS_INTEGRATION) && defined(TDESKTOP_FORCE_GTK_FILE_DIALOG) #if !defined(TDESKTOP_DISABLE_DBUS_INTEGRATION) && defined(TDESKTOP_FORCE_GTK_FILE_DIALOG)
LOG(("Checking for XDG Desktop Portal...")); LOG(("Checking for XDG Desktop Portal..."));
XDGDesktopPortalIsPresent = QDBusInterface( XDGDesktopPortalPresent = QDBusInterface(
"org.freedesktop.portal.Desktop", "org.freedesktop.portal.Desktop",
"/org/freedesktop/portal/desktop").isValid(); "/org/freedesktop/portal/desktop").isValid();
// this can give us a chance to use a proper file dialog for current session // this can give us a chance to use a proper file dialog for current session
if(XDGDesktopPortalIsPresent) { if(XDGDesktopPortalPresent) {
LOG(("XDG Desktop Portal is present!")); LOG(("XDG Desktop Portal is present!"));
qputenv("QT_QPA_PLATFORMTHEME", "xdgdesktopportal"); qputenv("QT_QPA_PLATFORMTHEME", "xdgdesktopportal");
} else { } else {
@ -467,6 +494,8 @@ bool OpenSystemSettings(SystemSettingsType type) {
add("kcmshell4 phonon"); add("kcmshell4 phonon");
} else if (DesktopEnvironment::IsGnome()) { } else if (DesktopEnvironment::IsGnome()) {
add("gnome-control-center sound"); add("gnome-control-center sound");
} else if (DesktopEnvironment::IsMATE()) {
add("mate-volume-control");
} }
add("pavucontrol-qt"); add("pavucontrol-qt");
add("pavucontrol"); add("pavucontrol");

View File

@ -25,8 +25,10 @@ bool InSnap();
bool IsXDGDesktopPortalPresent(); bool IsXDGDesktopPortalPresent();
QString ProcessNameByPID(const QString &pid);
QString CurrentExecutablePath(int argc, char *argv[]); QString CurrentExecutablePath(int argc, char *argv[]);
QString AppRuntimeDirectory();
QString SingleInstanceLocalServerName(const QString &hash); QString SingleInstanceLocalServerName(const QString &hash);
QString GetLauncherBasename(); QString GetLauncherBasename();

View File

@ -548,6 +548,9 @@ void MainWindow::psShowTrayMenu() {
} }
void MainWindow::psTrayMenuUpdated() { void MainWindow::psTrayMenuUpdated() {
if (trayIcon && trayIconMenu && trayIcon->contextMenu() != trayIconMenu) {
trayIcon->setContextMenu(trayIconMenu);
}
} }
void MainWindow::psSetupTrayIcon() { void MainWindow::psSetupTrayIcon() {

1
Telegram/ThirdParty/libdbusmenu-qt vendored Submodule

@ -0,0 +1 @@
Subproject commit 75afa1003c1d0f6fdfa3a76ce2db689b49f86968

View File

@ -0,0 +1,75 @@
/* BEGIN_COMMON_COPYRIGHT_HEADER
* (c)LGPL2+
*
* LXQt - a lightweight, Qt based, desktop toolset
* https://lxqt.org
*
* Copyright: 2015 LXQt team
* Authors:
* Balázs Béla <balazsbela[at]gmail.com>
* Paulo Lieuthier <paulolieuthier@gmail.com>
*
* This program or library is free software; you can redistribute it
* and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA
*
* END_COMMON_COPYRIGHT_HEADER */
#include "dbustypes.h"
// Marshall the IconPixmap data into a D-Bus argument
QDBusArgument &operator<<(QDBusArgument &argument, const IconPixmap &icon)
{
argument.beginStructure();
argument << icon.width;
argument << icon.height;
argument << icon.bytes;
argument.endStructure();
return argument;
}
// Retrieve the ImageStruct data from the D-Bus argument
const QDBusArgument &operator>>(const QDBusArgument &argument, IconPixmap &icon)
{
argument.beginStructure();
argument >> icon.width;
argument >> icon.height;
argument >> icon.bytes;
argument.endStructure();
return argument;
}
// Marshall the ToolTip data into a D-Bus argument
QDBusArgument &operator<<(QDBusArgument &argument, const ToolTip &toolTip)
{
argument.beginStructure();
argument << toolTip.iconName;
argument << toolTip.iconPixmap;
argument << toolTip.title;
argument << toolTip.description;
argument.endStructure();
return argument;
}
// Retrieve the ToolTip data from the D-Bus argument
const QDBusArgument &operator>>(const QDBusArgument &argument, ToolTip &toolTip)
{
argument.beginStructure();
argument >> toolTip.iconName;
argument >> toolTip.iconPixmap;
argument >> toolTip.title;
argument >> toolTip.description;
argument.endStructure();
return argument;
}

View File

@ -0,0 +1,60 @@
/* BEGIN_COMMON_COPYRIGHT_HEADER
* (c)LGPL2+
*
* LXQt - a lightweight, Qt based, desktop toolset
* https://lxqt.org
*
* Copyright: 2015 LXQt team
* Authors:
* Balázs Béla <balazsbela[at]gmail.com>
* Paulo Lieuthier <paulolieuthier@gmail.com>
*
* This program or library is free software; you can redistribute it
* and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA
*
* END_COMMON_COPYRIGHT_HEADER */
#include <QDBusArgument>
#ifndef DBUSTYPES_H
#define DBUSTYPES_H
struct IconPixmap {
int width;
int height;
QByteArray bytes;
};
typedef QList<IconPixmap> IconPixmapList;
Q_DECLARE_METATYPE(IconPixmap)
Q_DECLARE_METATYPE(IconPixmapList)
struct ToolTip {
QString iconName;
QList<IconPixmap> iconPixmap;
QString title;
QString description;
};
Q_DECLARE_METATYPE(ToolTip)
QDBusArgument &operator<<(QDBusArgument &argument, const IconPixmap &icon);
const QDBusArgument &operator>>(const QDBusArgument &argument, IconPixmap &icon);
QDBusArgument &operator<<(QDBusArgument &argument, const ToolTip &toolTip);
const QDBusArgument &operator>>(const QDBusArgument &argument, ToolTip &toolTip);
#endif // DBUSTYPES_H

View File

@ -0,0 +1,69 @@
<!DOCTYPE node PUBLIC "-//freedesktop//DTD D-BUS Object Introspection 1.0//EN" "http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd">
<node>
<interface name="org.kde.StatusNotifierItem">
<property name="Category" type="s" access="read"/>
<property name="Id" type="s" access="read"/>
<property name="Title" type="s" access="read"/>
<property name="Status" type="s" access="read"/>
<property name="WindowId" type="i" access="read"/>
<property name="IconThemePath" type="s" access="read"/>
<property name="Menu" type="o" access="read"/>
<property name="ItemIsMenu" type="b" access="read"/>
<property name="IconName" type="s" access="read"/>
<property name="IconPixmap" type="a(iiay)" access="read">
<annotation name="org.qtproject.QtDBus.QtTypeName" value="IconPixmapList"/>
</property>
<property name="OverlayIconName" type="s" access="read"/>
<property name="OverlayIconPixmap" type="a(iiay)" access="read">
<annotation name="org.qtproject.QtDBus.QtTypeName" value="IconPixmapList"/>
</property>
<property name="AttentionIconName" type="s" access="read"/>
<property name="AttentionIconPixmap" type="a(iiay)" access="read">
<annotation name="org.qtproject.QtDBus.QtTypeName" value="IconPixmapList"/>
</property>
<property name="AttentionMovieName" type="s" access="read"/>
<property name="ToolTip" type="(sa(iiay)ss)" access="read">
<annotation name="org.qtproject.QtDBus.QtTypeName" value="ToolTip"/>
</property>
<method name="ContextMenu">
<arg name="x" type="i" direction="in"/>
<arg name="y" type="i" direction="in"/>
</method>
<method name="Activate">
<arg name="x" type="i" direction="in"/>
<arg name="y" type="i" direction="in"/>
</method>
<method name="SecondaryActivate">
<arg name="x" type="i" direction="in"/>
<arg name="y" type="i" direction="in"/>
</method>
<method name="Scroll">
<arg name="delta" type="i" direction="in"/>
<arg name="orientation" type="s" direction="in"/>
</method>
<signal name="NewTitle">
</signal>
<signal name="NewIcon">
</signal>
<signal name="NewAttentionIcon">
</signal>
<signal name="NewOverlayIcon">
</signal>
<signal name="NewToolTip">
</signal>
<signal name="NewStatus">
<arg name="status" type="s"/>
</signal>
</interface>
</node>

View File

@ -0,0 +1,331 @@
/* BEGIN_COMMON_COPYRIGHT_HEADER
* (c)LGPL2+
*
* LXQt - a lightweight, Qt based, desktop toolset
* https://lxqt.org/
*
* Copyright: 2015 LXQt team
* Authors:
* Paulo Lieuthier <paulolieuthier@gmail.com>
*
* This program or library is free software; you can redistribute it
* and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA
*
* END_COMMON_COPYRIGHT_HEADER */
#include "statusnotifieritem.h"
#include "statusnotifieritemadaptor.h"
#include <QDBusInterface>
#include <QDBusServiceWatcher>
#include <dbusmenuexporter.h>
int StatusNotifierItem::mServiceCounter = 0;
StatusNotifierItem::StatusNotifierItem(QString id, QObject *parent)
: QObject(parent),
mAdaptor(new StatusNotifierItemAdaptor(this)),
mService(QString::fromLatin1("org.freedesktop.StatusNotifierItem-%1-%2")
.arg(QCoreApplication::applicationPid())
.arg(++mServiceCounter)),
mId(id),
mTitle(QLatin1String("Test")),
mStatus(QLatin1String("Active")),
mMenu(nullptr),
mMenuPath(QLatin1String("/NO_DBUSMENU")),
mMenuExporter(nullptr),
mSessionBus(QDBusConnection::connectToBus(QDBusConnection::SessionBus, mService))
{
// Separate DBus connection to the session bus is created, because QDbus does not provide
// a way to register different objects for different services with the same paths.
// For status notifiers we need different /StatusNotifierItem for each service.
// register service
mSessionBus.registerObject(QLatin1String("/StatusNotifierItem"), this);
registerToHost();
// monitor the watcher service in case the host restarts
QDBusServiceWatcher *watcher = new QDBusServiceWatcher(QLatin1String("org.kde.StatusNotifierWatcher"),
mSessionBus,
QDBusServiceWatcher::WatchForOwnerChange,
this);
connect(watcher, &QDBusServiceWatcher::serviceOwnerChanged,
this, &StatusNotifierItem::onServiceOwnerChanged);
}
StatusNotifierItem::~StatusNotifierItem()
{
mSessionBus.unregisterObject(QLatin1String("/StatusNotifierItem"));
QDBusConnection::disconnectFromBus(mService);
}
void StatusNotifierItem::registerToHost()
{
QDBusInterface interface(QLatin1String("org.kde.StatusNotifierWatcher"),
QLatin1String("/StatusNotifierWatcher"),
QLatin1String("org.kde.StatusNotifierWatcher"),
mSessionBus);
interface.asyncCall(QLatin1String("RegisterStatusNotifierItem"), mSessionBus.baseService());
}
void StatusNotifierItem::onServiceOwnerChanged(const QString& service, const QString& oldOwner,
const QString& newOwner)
{
Q_UNUSED(service);
Q_UNUSED(oldOwner);
if (!newOwner.isEmpty())
registerToHost();
}
void StatusNotifierItem::onMenuDestroyed()
{
mMenu = nullptr;
setMenuPath(QLatin1String("/NO_DBUSMENU"));
mMenuExporter = nullptr; //mMenu is a QObject parent of the mMenuExporter
}
void StatusNotifierItem::setTitle(const QString &title)
{
if (mTitle == title)
return;
mTitle = title;
Q_EMIT mAdaptor->NewTitle();
}
void StatusNotifierItem::setStatus(const QString &status)
{
if (mStatus == status)
return;
mStatus = status;
Q_EMIT mAdaptor->NewStatus(mStatus);
}
void StatusNotifierItem::setMenuPath(const QString& path)
{
mMenuPath.setPath(path);
}
void StatusNotifierItem::setIconByName(const QString &name)
{
if (mIconName == name)
return;
mIconName = name;
Q_EMIT mAdaptor->NewIcon();
}
void StatusNotifierItem::setIconByPixmap(const QIcon &icon)
{
if (mIconCacheKey == icon.cacheKey())
return;
mIconCacheKey = icon.cacheKey();
mIcon = iconToPixmapList(icon);
mIconName.clear();
Q_EMIT mAdaptor->NewIcon();
}
void StatusNotifierItem::setOverlayIconByName(const QString &name)
{
if (mOverlayIconName == name)
return;
mOverlayIconName = name;
Q_EMIT mAdaptor->NewOverlayIcon();
}
void StatusNotifierItem::setOverlayIconByPixmap(const QIcon &icon)
{
if (mOverlayIconCacheKey == icon.cacheKey())
return;
mOverlayIconCacheKey = icon.cacheKey();
mOverlayIcon = iconToPixmapList(icon);
mOverlayIconName.clear();
Q_EMIT mAdaptor->NewOverlayIcon();
}
void StatusNotifierItem::setAttentionIconByName(const QString &name)
{
if (mAttentionIconName == name)
return;
mAttentionIconName = name;
Q_EMIT mAdaptor->NewAttentionIcon();
}
void StatusNotifierItem::setAttentionIconByPixmap(const QIcon &icon)
{
if (mAttentionIconCacheKey == icon.cacheKey())
return;
mAttentionIconCacheKey = icon.cacheKey();
mAttentionIcon = iconToPixmapList(icon);
mAttentionIconName.clear();
Q_EMIT mAdaptor->NewAttentionIcon();
}
void StatusNotifierItem::setToolTipTitle(const QString &title)
{
if (mTooltipTitle == title)
return;
mTooltipTitle = title;
Q_EMIT mAdaptor->NewToolTip();
}
void StatusNotifierItem::setToolTipSubTitle(const QString &subTitle)
{
if (mTooltipSubtitle == subTitle)
return;
mTooltipSubtitle = subTitle;
Q_EMIT mAdaptor->NewToolTip();
}
void StatusNotifierItem::setToolTipIconByName(const QString &name)
{
if (mTooltipIconName == name)
return;
mTooltipIconName = name;
Q_EMIT mAdaptor->NewToolTip();
}
void StatusNotifierItem::setToolTipIconByPixmap(const QIcon &icon)
{
if (mTooltipIconCacheKey == icon.cacheKey())
return;
mTooltipIconCacheKey = icon.cacheKey();
mTooltipIcon = iconToPixmapList(icon);
mTooltipIconName.clear();
Q_EMIT mAdaptor->NewToolTip();
}
void StatusNotifierItem::setContextMenu(QMenu* menu)
{
if (mMenu == menu)
return;
if (nullptr != mMenu)
{
disconnect(mMenu, &QObject::destroyed, this, &StatusNotifierItem::onMenuDestroyed);
}
mMenu = menu;
if (nullptr != mMenu)
setMenuPath(QLatin1String("/MenuBar"));
else
setMenuPath(QLatin1String("/NO_DBUSMENU"));
//Note: we need to destroy menu exporter before creating new one -> to free the DBus object path for new menu
delete mMenuExporter;
if (nullptr != mMenu)
{
connect(mMenu, &QObject::destroyed, this, &StatusNotifierItem::onMenuDestroyed);
mMenuExporter = new DBusMenuExporter{this->menu().path(), mMenu, mSessionBus};
}
}
void StatusNotifierItem::Activate(int x, int y)
{
if (mStatus == QLatin1String("NeedsAttention"))
mStatus = QLatin1String("Active");
Q_EMIT activateRequested(QPoint(x, y));
}
void StatusNotifierItem::SecondaryActivate(int x, int y)
{
if (mStatus == QLatin1String("NeedsAttention"))
mStatus = QLatin1String("Active");
Q_EMIT secondaryActivateRequested(QPoint(x, y));
}
void StatusNotifierItem::ContextMenu(int x, int y)
{
if (mMenu)
{
if (mMenu->isVisible())
mMenu->popup(QPoint(x, y));
else
mMenu->hide();
}
}
void StatusNotifierItem::Scroll(int delta, const QString &orientation)
{
Qt::Orientation orient = Qt::Vertical;
if (orientation.toLower() == QLatin1String("horizontal"))
orient = Qt::Horizontal;
Q_EMIT scrollRequested(delta, orient);
}
void StatusNotifierItem::showMessage(const QString& title, const QString& msg,
const QString& iconName, int secs)
{
QDBusInterface interface(QLatin1String("org.freedesktop.Notifications"), QLatin1String("/org/freedesktop/Notifications"),
QLatin1String("org.freedesktop.Notifications"), mSessionBus);
interface.call(QLatin1String("Notify"), mTitle, (uint) 0, iconName, title,
msg, QStringList(), QVariantMap(), secs);
}
IconPixmapList StatusNotifierItem::iconToPixmapList(const QIcon& icon)
{
IconPixmapList pixmapList;
// long live KDE!
const QList<QSize> sizes = icon.availableSizes();
for (const QSize &size : sizes)
{
QImage image = icon.pixmap(size).toImage();
IconPixmap pix;
pix.height = image.height();
pix.width = image.width();
if (image.format() != QImage::Format_ARGB32)
image = image.convertToFormat(QImage::Format_ARGB32);
pix.bytes = QByteArray((char *) image.bits(),
#if QT_VERSION < QT_VERSION_CHECK(5, 10, 0)
image.byteCount());
#else
image.sizeInBytes());
#endif
// swap to network byte order if we are little endian
if (QSysInfo::ByteOrder == QSysInfo::LittleEndian)
{
quint32 *uintBuf = (quint32 *) pix.bytes.data();
for (uint i = 0; i < pix.bytes.size() / sizeof(quint32); ++i)
{
*uintBuf = qToBigEndian(*uintBuf);
++uintBuf;
}
}
pixmapList.append(pix);
}
return pixmapList;
}

View File

@ -0,0 +1,185 @@
/* BEGIN_COMMON_COPYRIGHT_HEADER
* (c)LGPL2+
*
* LXQt - a lightweight, Qt based, desktop toolset
* https://lxqt.org/
*
* Copyright: 2015 LXQt team
* Authors:
* Paulo Lieuthier <paulolieuthier@gmail.com>
*
* This program or library is free software; you can redistribute it
* and/or modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 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
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General
* Public License along with this library; if not, write to the
* Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA
*
* END_COMMON_COPYRIGHT_HEADER */
#ifndef STATUS_NOTIFIER_ITEM_H
#define STATUS_NOTIFIER_ITEM_H
#include <QObject>
#include <QIcon>
#include <QMenu>
#include <QDBusConnection>
#include "dbustypes.h"
class StatusNotifierItemAdaptor;
class DBusMenuExporter;
class StatusNotifierItem : public QObject
{
Q_OBJECT
Q_PROPERTY(QString Title READ title)
Q_PROPERTY(QString Id READ id)
Q_PROPERTY(QString Status READ status)
Q_PROPERTY(QDBusObjectPath Menu READ menu)
Q_PROPERTY(QString IconName READ iconName)
Q_PROPERTY(IconPixmapList IconPixmap READ iconPixmap)
Q_PROPERTY(QString OverlayIconName READ overlayIconName)
Q_PROPERTY(IconPixmapList OverlayIconPixmap READ overlayIconPixmap)
Q_PROPERTY(QString AttentionIconName READ attentionIconName)
Q_PROPERTY(IconPixmapList AttentionIconPixmap READ attentionIconPixmap)
Q_PROPERTY(ToolTip ToolTip READ toolTip)
public:
StatusNotifierItem(QString id, QObject *parent = nullptr);
~StatusNotifierItem() override;
QString id() const
{ return mId; }
QString title() const
{ return mTitle; }
void setTitle(const QString &title);
QString status() const
{ return mStatus; }
void setStatus(const QString &status);
QDBusObjectPath menu() const
{ return mMenuPath; }
void setMenuPath(const QString &path);
QString iconName() const
{ return mIconName; }
void setIconByName(const QString &name);
IconPixmapList iconPixmap() const
{ return mIcon; }
void setIconByPixmap(const QIcon &icon);
QString overlayIconName() const
{ return mOverlayIconName; }
void setOverlayIconByName(const QString &name);
IconPixmapList overlayIconPixmap() const
{ return mOverlayIcon; }
void setOverlayIconByPixmap(const QIcon &icon);
QString attentionIconName() const
{ return mAttentionIconName; }
void setAttentionIconByName(const QString &name);
IconPixmapList attentionIconPixmap() const
{ return mAttentionIcon; }
void setAttentionIconByPixmap(const QIcon &icon);
QString toolTipTitle() const
{ return mTooltipTitle; }
void setToolTipTitle(const QString &title);
QString toolTipSubTitle() const
{ return mTooltipSubtitle; }
void setToolTipSubTitle(const QString &subTitle);
QString toolTipIconName() const
{ return mTooltipIconName; }
void setToolTipIconByName(const QString &name);
IconPixmapList toolTipIconPixmap() const
{ return mTooltipIcon; }
void setToolTipIconByPixmap(const QIcon &icon);
ToolTip toolTip() const
{
ToolTip tt;
tt.title = mTooltipTitle;
tt.description = mTooltipSubtitle;
tt.iconName = mTooltipIconName;
tt.iconPixmap = mTooltipIcon;
return tt;
}
/*!
* \Note: we don't take ownership for the \param menu
*/
void setContextMenu(QMenu *menu);
public Q_SLOTS:
void Activate(int x, int y);
void SecondaryActivate(int x, int y);
void ContextMenu(int x, int y);
void Scroll(int delta, const QString &orientation);
void showMessage(const QString &title, const QString &msg, const QString &iconName, int secs);
private:
void registerToHost();
IconPixmapList iconToPixmapList(const QIcon &icon);
private Q_SLOTS:
void onServiceOwnerChanged(const QString &service, const QString &oldOwner,
const QString &newOwner);
void onMenuDestroyed();
Q_SIGNALS:
void activateRequested(const QPoint &pos);
void secondaryActivateRequested(const QPoint &pos);
void scrollRequested(int delta, Qt::Orientation orientation);
private:
StatusNotifierItemAdaptor *mAdaptor;
QString mService;
QString mId;
QString mTitle;
QString mStatus;
// icons
QString mIconName, mOverlayIconName, mAttentionIconName;
IconPixmapList mIcon, mOverlayIcon, mAttentionIcon;
qint64 mIconCacheKey, mOverlayIconCacheKey, mAttentionIconCacheKey;
// tooltip
QString mTooltipTitle, mTooltipSubtitle, mTooltipIconName;
IconPixmapList mTooltipIcon;
qint64 mTooltipIconCacheKey;
// menu
QMenu *mMenu;
QDBusObjectPath mMenuPath;
DBusMenuExporter *mMenuExporter;
QDBusConnection mSessionBus;
static int mServiceCounter;
};
#endif

View File

@ -8,7 +8,7 @@ option(TDESKTOP_FORCE_GTK_FILE_DIALOG "Force using GTK file dialog (Linux only).
option(TDESKTOP_DISABLE_REGISTER_CUSTOM_SCHEME "Disable automatic 'tg://' URL scheme handler registration." ${DESKTOP_APP_USE_PACKAGED}) option(TDESKTOP_DISABLE_REGISTER_CUSTOM_SCHEME "Disable automatic 'tg://' URL scheme handler registration." ${DESKTOP_APP_USE_PACKAGED})
option(TDESKTOP_DISABLE_NETWORK_PROXY "Disable all code for working through Socks5 or MTProxy." OFF) option(TDESKTOP_DISABLE_NETWORK_PROXY "Disable all code for working through Socks5 or MTProxy." OFF)
option(TDESKTOP_DISABLE_DESKTOP_FILE_GENERATION "Disable automatic '.desktop' file generation (Linux only)." ${DESKTOP_APP_USE_PACKAGED}) option(TDESKTOP_DISABLE_DESKTOP_FILE_GENERATION "Disable automatic '.desktop' file generation (Linux only)." ${DESKTOP_APP_USE_PACKAGED})
option(TDESKTOP_DISABLE_GTK_INTEGRATION "Disable all code for GTK integration (Linux only)." OFF) option(TDESKTOP_DISABLE_GTK_INTEGRATION "Disable all code for GTK integration (Linux only)." ON)
option(TDESKTOP_DISABLE_DBUS_INTEGRATION "Disable all code for D-Bus integration (Linux only)." OFF) option(TDESKTOP_DISABLE_DBUS_INTEGRATION "Disable all code for D-Bus integration (Linux only)." OFF)
option(TDESKTOP_USE_PACKAGED_TGVOIP "Find libtgvoip using CMake instead of bundled one." ${DESKTOP_APP_USE_PACKAGED}) option(TDESKTOP_USE_PACKAGED_TGVOIP "Find libtgvoip using CMake instead of bundled one." ${DESKTOP_APP_USE_PACKAGED})
option(TDESKTOP_API_TEST "Use test API credentials." OFF) option(TDESKTOP_API_TEST "Use test API credentials." OFF)
@ -48,8 +48,8 @@ if (NOT DESKTOP_APP_SPECIAL_TARGET STREQUAL "")
set(TDESKTOP_FORCE_GTK_FILE_DIALOG ON) set(TDESKTOP_FORCE_GTK_FILE_DIALOG ON)
endif() endif()
if (TDESKTOP_FORCE_GTK_FILE_DIALOG AND TDESKTOP_DISABLE_GTK_INTEGRATION) if (TDESKTOP_FORCE_GTK_FILE_DIALOG)
message(FATAL_ERROR "Option TDESKTOP_FORCE_GTK_FILE_DIALOG conflicts with option TDESKTOP_DISABLE_GTK_INTEGRATION.") set(TDESKTOP_DISABLE_GTK_INTEGRATION OFF)
endif() endif()
if (DESKTOP_APP_DISABLE_SPELLCHECK) if (DESKTOP_APP_DISABLE_SPELLCHECK)

View File

@ -14,7 +14,7 @@ You will need GCC 8 installed. To install them and all the required dependencies
sudo apt-get install software-properties-common -y && \ sudo apt-get install software-properties-common -y && \
sudo apt-get install git libexif-dev liblzma-dev libz-dev libssl-dev \ sudo apt-get install git libexif-dev liblzma-dev libz-dev libssl-dev \
libappindicator-dev libicu-dev libdee-dev libdrm-dev dh-autoreconf \ libgtk2.0-dev libice-dev libsm-dev libicu-dev libdrm-dev dh-autoreconf \
autoconf automake build-essential libass-dev libfreetype6-dev \ autoconf automake build-essential libass-dev libfreetype6-dev \
libgpac-dev libsdl1.2-dev libtheora-dev libtool libva-dev libvdpau-dev \ libgpac-dev libsdl1.2-dev libtheora-dev libtool libva-dev libvdpau-dev \
libvorbis-dev libenchant-dev libxcb1-dev libxcb-image0-dev libxcb-shm0-dev \ libvorbis-dev libenchant-dev libxcb1-dev libxcb-image0-dev libxcb-shm0-dev \

View File

@ -52,15 +52,13 @@ parts:
- g++-8 - g++-8
- qtbase5-private-dev - qtbase5-private-dev
- libmapbox-variant-dev - libmapbox-variant-dev
- libmsgsl-dev
- libayatana-appindicator3-dev
- libgtk-3-dev
- libasound2-dev - libasound2-dev
- libavcodec-dev - libavcodec-dev
- libavformat-dev - libavformat-dev
- libavutil-dev - libavutil-dev
- libswscale-dev - libswscale-dev
- libswresample-dev - libswresample-dev
- libdbusmenu-qt5-dev
- liblz4-dev - liblz4-dev
- liblzma-dev - liblzma-dev
- libminizip-dev - libminizip-dev
@ -71,13 +69,13 @@ parts:
- zlib1g-dev - zlib1g-dev
stage-packages: stage-packages:
- qt5-image-formats-plugins - qt5-image-formats-plugins
- libayatana-appindicator3-1
- libasound2 - libasound2
- libavcodec57 - libavcodec57
- libavformat57 - libavformat57
- libavutil55 - libavutil55
- libswscale4 - libswscale4
- libswresample2 - libswresample2
- libdbusmenu-qt5-2
- liblz4-1 - liblz4-1
- liblzma5 - liblzma5
- libminizip1 - libminizip1
@ -116,6 +114,7 @@ parts:
- cmake - cmake
- desktop-qt5 - desktop-qt5
- enchant - enchant
- gsl
- range-v3 - range-v3
- xxhash - xxhash
@ -205,6 +204,15 @@ parts:
- --enable-relocatable - --enable-relocatable
prime: [-./bin/*] prime: [-./bin/*]
gsl:
source: https://github.com/microsoft/GSL.git
source-depth: 1
source-tag: v2.1.0
plugin: cmake
configflags:
- -DGSL_TEST=OFF
prime: [-./*]
range-v3: range-v3:
source: https://github.com/ericniebler/range-v3.git source: https://github.com/ericniebler/range-v3.git
source-depth: 1 source-depth: 1