mirror of
https://github.com/qTox/qTox.git
synced 2024-03-22 14:00:36 +08:00
d4ac13dbf4
Revert needed, since otherwise there is no way to do automatic sorting of includes. Also reverted change to the docs, as leaving it would make incorrect docs. In case of conflicts, includes were sorted according to the coding standards from #3839. This reverts commitb4a9f04f92
. This reverts commit5921122960
.
364 lines
13 KiB
C++
364 lines
13 KiB
C++
/*
|
|
Copyright © 2015 by The qTox Project Contributors
|
|
|
|
This file is part of qTox, a Qt-based graphical interface for Tox.
|
|
|
|
qTox is libre software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
qTox 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with qTox. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
#include "systemtrayicon.h"
|
|
#include <QString>
|
|
#include <QSystemTrayIcon>
|
|
#include <QMenu>
|
|
#include <QFile>
|
|
#include <QDebug>
|
|
#include "src/persistence/settings.h"
|
|
|
|
SystemTrayIcon::SystemTrayIcon()
|
|
{
|
|
QString desktop = getenv("XDG_CURRENT_DESKTOP");
|
|
if (desktop.isEmpty())
|
|
desktop = getenv("DESKTOP_SESSION");
|
|
desktop = desktop.toLower();
|
|
if (false);
|
|
#ifdef ENABLE_SYSTRAY_UNITY_BACKEND
|
|
else if (desktop == "unity")
|
|
{
|
|
qDebug() << "Using Unity backend";
|
|
gtk_init(nullptr, nullptr);
|
|
QString settingsDir = Settings::getInstance().getSettingsDirPath();
|
|
QFile iconFile(settingsDir+"/icon.png");
|
|
if (iconFile.open(QIODevice::Truncate | QIODevice::WriteOnly))
|
|
{
|
|
QFile resIconFile(":/img/icon.png");
|
|
if (resIconFile.open(QIODevice::ReadOnly))
|
|
iconFile.write(resIconFile.readAll());
|
|
resIconFile.close();
|
|
iconFile.close();
|
|
}
|
|
backendType = SystrayBackendType::Unity;
|
|
unityMenu = gtk_menu_new();
|
|
unityIndicator = app_indicator_new_with_path(
|
|
"qTox",
|
|
"icon",
|
|
APP_INDICATOR_CATEGORY_APPLICATION_STATUS,
|
|
settingsDir.toStdString().c_str()
|
|
);
|
|
app_indicator_set_menu(unityIndicator, GTK_MENU(unityMenu));
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_GTK_BACKEND
|
|
else if (desktop == "xfce" || desktop.contains("gnome") || desktop == "mate" || desktop == "x-cinnamon")
|
|
{
|
|
qDebug() << "Using GTK backend";
|
|
backendType = SystrayBackendType::GTK;
|
|
gtk_init(nullptr, nullptr);
|
|
|
|
gtkIcon = gtk_status_icon_new();
|
|
|
|
gtkMenu = gtk_menu_new();
|
|
|
|
void (*callbackTrigger)(GtkStatusIcon*, gpointer) =
|
|
[](GtkStatusIcon*, gpointer data)
|
|
{
|
|
static_cast<SystemTrayIcon*>(data)->activated(QSystemTrayIcon::Trigger);
|
|
};
|
|
g_signal_connect(gtkIcon, "activate", G_CALLBACK(callbackTrigger), this);
|
|
void (*callbackButtonClick)(GtkStatusIcon*, GdkEvent*, gpointer) =
|
|
[](GtkStatusIcon*, GdkEvent* event, gpointer data)
|
|
{
|
|
if (event->button.button == 2)
|
|
static_cast<SystemTrayIcon*>(data)->activated(QSystemTrayIcon::MiddleClick);
|
|
};
|
|
g_signal_connect(gtkIcon, "button-release-event", G_CALLBACK(callbackButtonClick), this);
|
|
}
|
|
#endif
|
|
else
|
|
{
|
|
qDebug() << "Using the Qt backend";
|
|
qtIcon = new QSystemTrayIcon;
|
|
backendType = SystrayBackendType::Qt;
|
|
connect(qtIcon, &QSystemTrayIcon::activated, this, &SystemTrayIcon::activated);
|
|
}
|
|
}
|
|
|
|
SystemTrayIcon::~SystemTrayIcon()
|
|
{
|
|
qDebug() << "Deleting SystemTrayIcon";
|
|
delete qtIcon;
|
|
}
|
|
|
|
QString SystemTrayIcon::extractIconToFile(QIcon icon, QString name)
|
|
{
|
|
QString iconPath;
|
|
(void) icon;
|
|
(void) name;
|
|
#ifdef ENABLE_SYSTRAY_UNITY_BACKEND
|
|
iconPath = Settings::getInstance().getSettingsDirPath()+"/"+name+".png";
|
|
QSize iconSize = icon.actualSize(QSize{64,64});
|
|
icon.pixmap(iconSize).save(iconPath);
|
|
#endif
|
|
return iconPath;
|
|
}
|
|
|
|
#if defined(ENABLE_SYSTRAY_GTK_BACKEND) || defined(ENABLE_SYSTRAY_STATUSNOTIFIER_BACKEND)
|
|
GdkPixbuf* SystemTrayIcon::convertQIconToPixbuf(const QIcon &icon)
|
|
{
|
|
void (*callbackFreeImage)(guchar*, gpointer) =
|
|
[](guchar* image_bytes, gpointer)
|
|
{
|
|
delete[] image_bytes;
|
|
};
|
|
QImage image = icon.pixmap(64, 64).toImage();
|
|
if (image.format() != QImage::Format_RGBA8888_Premultiplied)
|
|
image = image.convertToFormat(QImage::Format_RGBA8888_Premultiplied);
|
|
guchar* image_bytes = new guchar[image.byteCount()];
|
|
memcpy(image_bytes, image.bits(), image.byteCount());
|
|
|
|
return gdk_pixbuf_new_from_data(image_bytes, GDK_COLORSPACE_RGB, image.hasAlphaChannel(),
|
|
8, image.width(), image.height(), image.bytesPerLine(),
|
|
callbackFreeImage, NULL);
|
|
}
|
|
#endif
|
|
|
|
void SystemTrayIcon::setContextMenu(QMenu* menu)
|
|
{
|
|
if (false);
|
|
#ifdef ENABLE_SYSTRAY_STATUSNOTIFIER_BACKEND
|
|
else if (backendType == SystrayBackendType::StatusNotifier)
|
|
{
|
|
void (*callbackClick)(StatusNotifier*, gint, gint, gpointer) =
|
|
[](StatusNotifier*, gint, gint, gpointer data)
|
|
{
|
|
static_cast<SystemTrayIcon*>(data)->activated(QSystemTrayIcon::Trigger);
|
|
};
|
|
g_signal_connect(statusNotifier, "activate", G_CALLBACK(callbackClick), this);
|
|
void (*callbackMiddleClick)(StatusNotifier*, gint, gint, gpointer) =
|
|
[](StatusNotifier*, gint, gint, gpointer data)
|
|
{
|
|
static_cast<SystemTrayIcon*>(data)->activated(QSystemTrayIcon::MiddleClick);
|
|
};
|
|
g_signal_connect(statusNotifier, "secondary_activate", G_CALLBACK(callbackMiddleClick), this);
|
|
|
|
for (QAction* a : menu->actions())
|
|
{
|
|
QString aText = a->text().replace('&',"");
|
|
GtkWidget* item;
|
|
if (a->isSeparator())
|
|
item = gtk_menu_item_new();
|
|
else if (a->icon().isNull())
|
|
item = gtk_menu_item_new_with_label(aText.toStdString().c_str());
|
|
else
|
|
{
|
|
GdkPixbuf* pixbuf = convertQIconToPixbuf(a->icon());
|
|
item = gtk_image_menu_item_new_with_label(aText.toStdString().c_str());
|
|
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), gtk_image_new_from_pixbuf(pixbuf));
|
|
gtk_image_menu_item_set_always_show_image(GTK_IMAGE_MENU_ITEM(item),TRUE);
|
|
g_object_unref(pixbuf);
|
|
}
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(snMenu), item);
|
|
void (*callback)(GtkMenu*, gpointer data) = [](GtkMenu*, gpointer a)
|
|
{
|
|
((QAction*)a)->activate(QAction::Trigger);
|
|
};
|
|
g_signal_connect(item, "activate", G_CALLBACK(callback), a);
|
|
gtk_widget_show(item);
|
|
}
|
|
void (*callbackMenu)(StatusNotifier*, gint, gint, gpointer) =
|
|
[](StatusNotifier*, gint, gint, gpointer data)
|
|
{
|
|
gtk_widget_show_all(static_cast<SystemTrayIcon*>(data)->snMenu);
|
|
gtk_menu_popup(GTK_MENU(static_cast<SystemTrayIcon*>(data)->snMenu), 0, 0, 0, 0, 3, gtk_get_current_event_time());
|
|
};
|
|
g_signal_connect(statusNotifier, "context-menu", G_CALLBACK(callbackMenu), this);
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_GTK_BACKEND
|
|
else if (backendType == SystrayBackendType::GTK)
|
|
{
|
|
for (QAction* a : menu->actions())
|
|
{
|
|
QString aText = a->text().replace('&',"");
|
|
GtkWidget* item;
|
|
if (a->isSeparator())
|
|
item = gtk_menu_item_new();
|
|
else if (a->icon().isNull())
|
|
item = gtk_menu_item_new_with_label(aText.toStdString().c_str());
|
|
else
|
|
{
|
|
GdkPixbuf* pixbuf = convertQIconToPixbuf(a->icon());
|
|
item = gtk_image_menu_item_new_with_label(aText.toStdString().c_str());
|
|
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), gtk_image_new_from_pixbuf(pixbuf));
|
|
gtk_image_menu_item_set_always_show_image(GTK_IMAGE_MENU_ITEM(item),TRUE);
|
|
g_object_unref(pixbuf);
|
|
}
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(gtkMenu), item);
|
|
void (*callback)(GtkMenu*, gpointer data) = [](GtkMenu*, gpointer a)
|
|
{
|
|
((QAction*)a)->activate(QAction::Trigger);
|
|
};
|
|
g_signal_connect(item, "activate", G_CALLBACK(callback), a);
|
|
gtk_widget_show(item);
|
|
}
|
|
void (*callbackMenu)(GtkMenu*, gint, gint, gpointer) =
|
|
[](GtkMenu*, gint, gint, gpointer data)
|
|
{
|
|
gtk_widget_show_all(static_cast<SystemTrayIcon*>(data)->gtkMenu);
|
|
gtk_menu_popup(GTK_MENU(static_cast<SystemTrayIcon*>(data)->gtkMenu), 0, 0, 0, 0, 3, gtk_get_current_event_time());
|
|
};
|
|
g_signal_connect(gtkIcon, "popup-menu", G_CALLBACK(callbackMenu), this);
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_UNITY_BACKEND
|
|
else if (backendType == SystrayBackendType::Unity)
|
|
{
|
|
for (QAction* a : menu->actions())
|
|
{
|
|
QString aText = a->text().replace('&',"");
|
|
GtkWidget* item;
|
|
if (a->isSeparator())
|
|
item = gtk_menu_item_new();
|
|
else if (a->icon().isNull())
|
|
item = gtk_menu_item_new_with_label(aText.toStdString().c_str());
|
|
else
|
|
{
|
|
QString iconPath = extractIconToFile(a->icon(),"iconmenu"+a->icon().name());
|
|
GtkWidget* image = gtk_image_new_from_file(iconPath.toStdString().c_str());
|
|
item = gtk_image_menu_item_new_with_label(aText.toStdString().c_str());
|
|
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), image);
|
|
gtk_image_menu_item_set_always_show_image(GTK_IMAGE_MENU_ITEM(item),TRUE);
|
|
}
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(unityMenu), item);
|
|
void (*callback)(GtkMenu*, gpointer data) = [](GtkMenu*, gpointer a)
|
|
{
|
|
static_cast<QAction*>(a)->activate(QAction::Trigger);
|
|
};
|
|
g_signal_connect(item, "activate", G_CALLBACK(callback), a);
|
|
gtk_widget_show(item);
|
|
}
|
|
app_indicator_set_menu(unityIndicator, GTK_MENU(unityMenu));
|
|
DbusmenuServer *menuServer;
|
|
DbusmenuMenuitem *rootMenuItem;
|
|
g_object_get(unityIndicator, "dbus-menu-server", &menuServer, NULL);
|
|
g_object_get(menuServer, "root-node", &rootMenuItem, NULL);
|
|
void (*callback)(DbusmenuMenuitem *, gpointer) =
|
|
[](DbusmenuMenuitem *, gpointer data)
|
|
{
|
|
static_cast<SystemTrayIcon*>(data)->activated(QSystemTrayIcon::Unknown);
|
|
};
|
|
g_signal_connect(rootMenuItem, "about-to-show", G_CALLBACK(callback), this);
|
|
}
|
|
#endif
|
|
else if (backendType == SystrayBackendType::Qt)
|
|
{
|
|
qtIcon->setContextMenu(menu);
|
|
}
|
|
}
|
|
|
|
void SystemTrayIcon::show()
|
|
{
|
|
setVisible(true);
|
|
}
|
|
|
|
void SystemTrayIcon::hide()
|
|
{
|
|
setVisible(false);
|
|
}
|
|
|
|
void SystemTrayIcon::setVisible(bool newState)
|
|
{
|
|
if (false);
|
|
#ifdef ENABLE_SYSTRAY_STATUSNOTIFIER_BACKEND
|
|
else if (backendType == SystrayBackendType::StatusNotifier)
|
|
{
|
|
if (newState)
|
|
status_notifier_set_status(statusNotifier, STATUS_NOTIFIER_STATUS_ACTIVE);
|
|
else
|
|
status_notifier_set_status(statusNotifier, STATUS_NOTIFIER_STATUS_PASSIVE);
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_GTK_BACKEND
|
|
else if (backendType == SystrayBackendType::GTK)
|
|
{
|
|
if (newState)
|
|
gtk_status_icon_set_visible(gtkIcon, true);
|
|
else
|
|
gtk_status_icon_set_visible(gtkIcon, false);
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_UNITY_BACKEND
|
|
else if (backendType == SystrayBackendType::Unity)
|
|
{
|
|
if (newState)
|
|
app_indicator_set_status(unityIndicator, APP_INDICATOR_STATUS_ACTIVE);
|
|
else
|
|
app_indicator_set_status(unityIndicator, APP_INDICATOR_STATUS_PASSIVE);
|
|
}
|
|
#endif
|
|
else if (backendType == SystrayBackendType::Qt)
|
|
{
|
|
if (newState)
|
|
qtIcon->show();
|
|
else
|
|
qtIcon->hide();
|
|
}
|
|
}
|
|
|
|
void SystemTrayIcon::setIcon(QIcon &icon)
|
|
{
|
|
if (false);
|
|
#ifdef ENABLE_SYSTRAY_STATUSNOTIFIER_BACKEND
|
|
else if (backendType == SystrayBackendType::StatusNotifier)
|
|
{
|
|
GdkPixbuf* pixbuf = convertQIconToPixbuf(icon);
|
|
status_notifier_set_from_pixbuf(statusNotifier, STATUS_NOTIFIER_ICON, pixbuf);
|
|
g_object_unref(pixbuf);
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_GTK_BACKEND
|
|
else if (backendType == SystrayBackendType::GTK)
|
|
{
|
|
GdkPixbuf* pixbuf = convertQIconToPixbuf(icon);
|
|
gtk_status_icon_set_from_pixbuf(gtkIcon, pixbuf);
|
|
g_object_unref(pixbuf);
|
|
}
|
|
#endif
|
|
#ifdef ENABLE_SYSTRAY_UNITY_BACKEND
|
|
else if (backendType == SystrayBackendType::Unity)
|
|
{
|
|
// Alternate file names or appindicator will not reload the icon
|
|
if (app_indicator_get_icon(unityIndicator) == QString("icon2"))
|
|
{
|
|
extractIconToFile(icon,"icon");
|
|
app_indicator_set_icon_full(unityIndicator, "icon","qtox");
|
|
}
|
|
else
|
|
{
|
|
extractIconToFile(icon,"icon2");
|
|
app_indicator_set_icon_full(unityIndicator, "icon2","qtox");
|
|
}
|
|
}
|
|
#endif
|
|
else if (backendType == SystrayBackendType::Qt)
|
|
{
|
|
qtIcon->setIcon(icon);
|
|
}
|
|
}
|
|
|
|
SystrayBackendType SystemTrayIcon::backend() const
|
|
{
|
|
return backendType;
|
|
}
|