mirror of https://github.com/procxx/kepka.git
Use LXQt's StatusNotifierItem implementation instead of appindicator
This commit is contained in:
parent
1b1f9d9985
commit
3b4dfa1381
|
@ -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 \
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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()
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -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");
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 75afa1003c1d0f6fdfa3a76ce2db689b49f86968
|
|
@ -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;
|
||||||
|
}
|
|
@ -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
|
|
@ -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>
|
|
@ -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;
|
||||||
|
}
|
|
@ -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
|
|
@ -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)
|
||||||
|
|
|
@ -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 \
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue