123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120 |
- /*
- * linux_notify.c
- * Copyright (C) 2019 Kovid Goyal <kovid at kovidgoyal.net>
- *
- * Distributed under terms of the GPL3 license.
- */
- #include "internal.h"
- #include "linux_notify.h"
- #include <stdlib.h>
- #define NOTIFICATIONS_SERVICE "org.freedesktop.Notifications"
- #define NOTIFICATIONS_PATH "/org/freedesktop/Notifications"
- #define NOTIFICATIONS_IFACE "org.freedesktop.Notifications"
- static inline void cleanup_free(void *p) { free(*(void**)p); }
- #define RAII_ALLOC(type, name, initializer) __attribute__((cleanup(cleanup_free))) type *name = initializer
- static notification_id_type notification_id = 0;
- typedef struct {
- notification_id_type next_id;
- GLFWDBusnotificationcreatedfun callback;
- void *data;
- } NotificationCreatedData;
- static GLFWDBusnotificationactivatedfun activated_handler = NULL;
- void
- glfw_dbus_set_user_notification_activated_handler(GLFWDBusnotificationactivatedfun handler) {
- activated_handler = handler;
- }
- void
- notification_created(DBusMessage *msg, const char* errmsg, void *data) {
- if (errmsg) {
- _glfwInputError(GLFW_PLATFORM_ERROR, "Notify: Failed to create notification error: %s", errmsg);
- if (data) free(data);
- return;
- }
- uint32_t id;
- if (!glfw_dbus_get_args(msg, "Failed to get Notification uid", DBUS_TYPE_UINT32, &id, DBUS_TYPE_INVALID)) return;
- NotificationCreatedData *ncd = (NotificationCreatedData*)data;
- if (ncd->callback) ncd->callback(ncd->next_id, id, ncd->data);
- if (data) free(data);
- }
- static DBusHandlerResult
- message_handler(DBusConnection *conn UNUSED, DBusMessage *msg, void *user_data UNUSED) {
- /* printf("session_bus message_handler invoked interface: %s member: %s\n", dbus_message_get_interface(msg), dbus_message_get_member(msg)); */
- if (dbus_message_is_signal(msg, NOTIFICATIONS_IFACE, "ActionInvoked")) {
- uint32_t id;
- const char *action;
- if (glfw_dbus_get_args(msg, "Failed to get args from ActionInvoked notification signal",
- DBUS_TYPE_UINT32, &id, DBUS_TYPE_STRING, &action, DBUS_TYPE_INVALID)) {
- if (activated_handler) {
- activated_handler(id, action);
- return DBUS_HANDLER_RESULT_HANDLED;
- }
- }
- }
- return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
- }
- notification_id_type
- glfw_dbus_send_user_notification(const char *app_name, const char* icon, const char *summary, const char *body, const char* action_name, int32_t timeout, int urgency, GLFWDBusnotificationcreatedfun callback, void *user_data) {
- DBusConnection *session_bus = glfw_dbus_session_bus();
- static DBusConnection *added_signal_match = NULL;
- if (!session_bus) return 0;
- if (added_signal_match != session_bus) {
- dbus_bus_add_match(session_bus, "type='signal',interface='" NOTIFICATIONS_IFACE "',member='ActionInvoked'", NULL);
- dbus_connection_add_filter(session_bus, message_handler, NULL, NULL);
- added_signal_match = session_bus;
- }
- RAII_ALLOC(NotificationCreatedData, data, malloc(sizeof(NotificationCreatedData)));
- if (!data) return 0;
- data->next_id = ++notification_id;
- data->callback = callback; data->data = user_data;
- if (!data->next_id) data->next_id = ++notification_id;
- uint32_t replaces_id = 0;
- RAII_MSG(msg, dbus_message_new_method_call(NOTIFICATIONS_SERVICE, NOTIFICATIONS_PATH, NOTIFICATIONS_IFACE, "Notify"));
- if (!msg) { return 0; }
- DBusMessageIter args, array, variant, dict;
- dbus_message_iter_init_append(msg, &args);
- #define check_call(func, ...) if (!func(__VA_ARGS__)) { _glfwInputError(GLFW_PLATFORM_ERROR, "%s", "Out of memory allocating DBUS message for notification\n"); return 0; }
- #define APPEND(to, type, val) check_call(dbus_message_iter_append_basic, &to, type, &val);
- APPEND(args, DBUS_TYPE_STRING, app_name)
- APPEND(args, DBUS_TYPE_UINT32, replaces_id)
- APPEND(args, DBUS_TYPE_STRING, icon)
- APPEND(args, DBUS_TYPE_STRING, summary)
- APPEND(args, DBUS_TYPE_STRING, body)
- check_call(dbus_message_iter_open_container, &args, DBUS_TYPE_ARRAY, "s", &array);
- if (action_name) {
- static const char* default_action = "default";
- APPEND(array, DBUS_TYPE_STRING, default_action);
- APPEND(array, DBUS_TYPE_STRING, action_name);
- }
- check_call(dbus_message_iter_close_container, &args, &array);
- check_call(dbus_message_iter_open_container, &args, DBUS_TYPE_ARRAY, "{sv}", &array);
- check_call(dbus_message_iter_open_container, &array, DBUS_TYPE_DICT_ENTRY, NULL, &dict);
- static const char* urgency_key = "urgency";
- APPEND(dict, DBUS_TYPE_STRING, urgency_key);
- check_call(dbus_message_iter_open_container, &dict, DBUS_TYPE_VARIANT, DBUS_TYPE_BYTE_AS_STRING, &variant);
- uint8_t urgencyb = urgency & 3;
- APPEND(variant, DBUS_TYPE_BYTE, urgencyb);
- check_call(dbus_message_iter_close_container, &dict, &variant);
- check_call(dbus_message_iter_close_container, &array, &dict);
- check_call(dbus_message_iter_close_container, &args, &array);
- APPEND(args, DBUS_TYPE_INT32, timeout)
- #undef check_call
- #undef APPEND
- if (!call_method_with_msg(session_bus, msg, 5000, notification_created, data)) return 0;
- notification_id_type ans = data->next_id;
- data = NULL;
- return ans;
- }
|