summaryrefslogtreecommitdiff
path: root/tray/src
diff options
context:
space:
mode:
Diffstat (limited to 'tray/src')
-rw-r--r--tray/src/cli.vala54
-rw-r--r--tray/src/config.vala.in6
-rw-r--r--tray/src/meson.build100
-rw-r--r--tray/src/tray.vala135
-rw-r--r--tray/src/trayItem.vala363
-rw-r--r--tray/src/watcher.vala59
6 files changed, 0 insertions, 717 deletions
diff --git a/tray/src/cli.vala b/tray/src/cli.vala
deleted file mode 100644
index 3147fb5..0000000
--- a/tray/src/cli.vala
+++ /dev/null
@@ -1,54 +0,0 @@
-static bool version;
-static bool daemonize;
-
-const OptionEntry[] options = {
- { "version", 'v', OptionFlags.NONE, OptionArg.NONE, ref version, "Print version number", null },
- { "daemonize", 'd', OptionFlags.NONE, OptionArg.NONE, ref daemonize, "Monitor the systemtray", null },
- { null },
-};
-
-int main(string[] argv) {
- try {
- var opts = new OptionContext();
- opts.add_main_entries(options, null);
- opts.set_help_enabled(true);
- opts.set_ignore_unknown_options(false);
- opts.parse(ref argv);
- } catch (OptionError err) {
- printerr (err.message);
- return 1;
- }
-
- if (version) {
- print(AstalTray.VERSION);
- return 0;
- }
-
- if (daemonize) {
- var loop = new MainLoop();
- var tray = new AstalTray.Tray();
-
- tray.item_added.connect((id) => {
- AstalTray.TrayItem item = tray.get_item(id);
-
- stdout.printf("{\"event\":\"item_added\",\"id\":\"%s\",\"item\":%s}\n",
- id, item.to_json_string());
- stdout.flush();
-
- item.changed.connect(() => {
- stdout.printf("{\"event\":\"item_changed\",\"id\":\"%s\",\"item\":%s}\n",
- id, item.to_json_string());
- stdout.flush();
- });
- });
-
- tray.item_removed.connect((id) => {
- stdout.printf("{\"event\":\"item_removed\",\"id\":\"%s\"}\n", id);
- stdout.flush();
- });
-
- loop.run();
- }
-
- return 0;
-}
diff --git a/tray/src/config.vala.in b/tray/src/config.vala.in
deleted file mode 100644
index 8ef8498..0000000
--- a/tray/src/config.vala.in
+++ /dev/null
@@ -1,6 +0,0 @@
-namespace AstalTray {
- public const int MAJOR_VERSION = @MAJOR_VERSION@;
- public const int MINOR_VERSION = @MINOR_VERSION@;
- public const int MICRO_VERSION = @MICRO_VERSION@;
- public const string VERSION = "@VERSION@";
-}
diff --git a/tray/src/meson.build b/tray/src/meson.build
deleted file mode 100644
index b2229b0..0000000
--- a/tray/src/meson.build
+++ /dev/null
@@ -1,100 +0,0 @@
-version_split = meson.project_version().split('.')
-api_version = version_split[0] + '.' + version_split[1]
-gir = 'AstalTray-' + api_version + '.gir'
-typelib = 'AstalTray-' + api_version + '.typelib'
-
-config = configure_file(
- input: 'config.vala.in',
- output: 'config.vala',
- configuration: {
- 'VERSION': meson.project_version(),
- 'MAJOR_VERSION': version_split[0],
- 'MINOR_VERSION': version_split[1],
- 'MICRO_VERSION': version_split[2],
- },
-)
-
-deps = [
- dependency('glib-2.0'),
- dependency('gobject-2.0'),
- dependency('gio-2.0'),
- dependency('json-glib-1.0'),
- dependency('gdk-pixbuf-2.0'),
- dependency('gtk+-3.0'),
-]
-
-dbusmenu_cflags = run_command(
- find_program('pkg-config', required: true),
- '--cflags', 'dbusmenu-gtk3-0.4',
- 'gobject-introspection-1.0',
- 'gobject-2.0',
- 'glib-2.0',
- capture: true,
- check: true,
-).stdout().strip()
-
-dbusmenu_libs = run_command(
- find_program('pkg-config', required: true),
- '--libs', 'dbusmenu-gtk3-0.4',
- 'gobject-introspection-1.0',
- 'gobject-2.0',
- 'glib-2.0',
- capture: true,
- check: true,
-).stdout().strip()
-
-sources = [config, 'tray.vala', 'watcher.vala', 'trayItem.vala']
-
-if get_option('lib')
- lib = library(
- meson.project_name(),
- sources,
- dependencies: deps,
- vala_header: meson.project_name() + '.h',
- vala_vapi: meson.project_name() + '-' + api_version + '.vapi',
- vala_gir: gir,
- vala_args: ['--pkg', 'DbusmenuGtk3-0.4', '--pkg', 'Dbusmenu-0.4'],
- version: meson.project_version(),
- c_args: dbusmenu_cflags.split(' '),
- link_args: dbusmenu_libs.split(' '),
- install: true,
- install_dir: [true, true, true, true],
- )
-
- import('pkgconfig').generate(
- lib,
- name: meson.project_name(),
- filebase: meson.project_name() + '-' + api_version,
- version: meson.project_version(),
- subdirs: meson.project_name(),
- requires: deps,
- install_dir: get_option('libdir') / 'pkgconfig',
- )
-
- custom_target(
- typelib,
- command: [
- find_program('g-ir-compiler'),
- '--output', '@OUTPUT@',
- '--shared-library', get_option('prefix') / get_option('libdir') / '@PLAINNAME@',
- meson.current_build_dir() / gir,
- ],
- input: lib,
- output: typelib,
- depends: lib,
- install: true,
- install_dir: get_option('libdir') / 'girepository-1.0',
- )
-endif
-
-if get_option('cli')
- executable(
- meson.project_name(),
- ['cli.vala', sources],
- dependencies: deps,
- vala_args: ['--pkg', 'DbusmenuGtk3-0.4', '--pkg', 'Dbusmenu-0.4'],
- c_args: dbusmenu_cflags.split(' '),
- link_args: dbusmenu_libs.split(' '),
- install: true,
- )
-endif
diff --git a/tray/src/tray.vala b/tray/src/tray.vala
deleted file mode 100644
index 09b0643..0000000
--- a/tray/src/tray.vala
+++ /dev/null
@@ -1,135 +0,0 @@
-namespace AstalTray {
-[DBus (name="org.kde.StatusNotifierWatcher")]
-internal interface IWatcher : Object {
- public abstract string[] RegisteredStatusNotifierItems { owned get; }
- public abstract int ProtocolVersion { owned get; }
-
- public abstract void RegisterStatusNotifierItem(string service, BusName sender) throws DBusError, IOError;
- public abstract void RegisterStatusNotifierHost(string service) throws DBusError, IOError;
-
- public signal void StatusNotifierItemRegistered(string service);
- public signal void StatusNotifierItemUnregistered(string service);
- public signal void StatusNotifierHostRegistered();
- public signal void StatusNotifierHostUnregistered();
-}
-
-public Tray get_default() {
- return Tray.get_default();
-}
-
-public class Tray : Object {
- private static Tray? instance;
- public static unowned Tray get_default() {
- if (instance == null)
- instance = new Tray();
-
- return instance;
- }
-
- private StatusNotifierWatcher watcher;
- private IWatcher proxy;
-
- private HashTable<string, TrayItem> _items =
- new HashTable<string, TrayItem>(str_hash, str_equal);
-
- public List<weak TrayItem> items { owned get { return _items.get_values(); }}
-
- public signal void item_added(string service) {
- notify_property("items");
- }
-
- public signal void item_removed(string service) {
- notify_property("items");
- }
-
- construct {
- try {
- Bus.own_name(
- BusType.SESSION,
- "org.kde.StatusNotifierWatcher",
- BusNameOwnerFlags.NONE,
- start_watcher,
- () => {
- if (proxy != null) {
- proxy = null;
- }
- },
- start_host
- );
- } catch (Error err) {
- critical(err.message);
- }
-
- }
-
- private void start_watcher(DBusConnection conn) {
- try {
- watcher = new StatusNotifierWatcher();
- conn.register_object("/StatusNotifierWatcher", watcher);
- watcher.StatusNotifierItemRegistered.connect(on_item_register);
- watcher.StatusNotifierItemUnregistered.connect(on_item_unregister);
- } catch (Error err) {
- critical(err.message);
- }
- }
-
- private void start_host() {
- if (proxy != null)
- return;
-
- try {
- proxy = Bus.get_proxy_sync(BusType.SESSION,
- "org.kde.StatusNotifierWatcher",
- "/StatusNotifierWatcher");
-
- proxy.StatusNotifierItemRegistered.connect(on_item_register);
- proxy.StatusNotifierItemUnregistered.connect(on_item_unregister);
-
- proxy.notify["g-name-owner"].connect(() => {
- _items.foreach((service, _) => {
- item_removed(service);
- });
-
- _items.remove_all();
-
- if(proxy != null) {
- foreach (string item in proxy.RegisteredStatusNotifierItems) {
- on_item_register(item);
- }
- } else {
- foreach (string item in watcher.RegisteredStatusNotifierItems) {
- on_item_register(item);
- }
- }
- });
-
- foreach (string item in proxy.RegisteredStatusNotifierItems) {
- on_item_register(item);
- }
- } catch (Error err) {
- critical("cannot get proxy: %s", err.message);
- }
- }
-
- private void on_item_register(string service) {
- if (_items.contains(service))
- return;
-
- var parts = service.split("/", 2);
- TrayItem item = new TrayItem(parts[0], "/" + parts[1]);
- item.ready.connect(() => {
- _items.set(service, item);
- item_added(service);
- });
- }
-
- private void on_item_unregister(string service) {
- _items.remove(service);
- item_removed(service);
- }
-
- public TrayItem get_item(string service) {
- return _items.get(service);
- }
-}
-}
diff --git a/tray/src/trayItem.vala b/tray/src/trayItem.vala
deleted file mode 100644
index b6b9da0..0000000
--- a/tray/src/trayItem.vala
+++ /dev/null
@@ -1,363 +0,0 @@
-using DbusmenuGtk;
-
-namespace AstalTray {
-public struct Pixmap {
- int width;
- int height;
- uint8[] bytes;
-}
-
-public struct Tooltip {
- string icon_name;
- Pixmap[] icon;
- string title;
- string description;
-}
-
-[DBus (use_string_marshalling = true)]
-public enum Category {
- [DBus (value = "ApplicationStatus"), Description (nick = "ApplicationStatus")]
- APPLICATION,
-
- [DBus (value = "Communications"), Description (nick = "Communications")]
- COMMUNICATIONS,
-
- [DBus (value = "SystemServices"), Description (nick = "SystemServices")]
- SYSTEM,
-
- [DBus (value = "Hardware"), Description (nick = "Hardware")]
- HARDWARE;
-
- public string to_nick () {
- var enumc = (EnumClass)typeof (Category).class_ref();
- unowned var eval = enumc.get_value(this);
- return eval.value_nick;
- }
-}
-
-
-[DBus (use_string_marshalling = true)]
-public enum Status {
- [DBus (value = "Passive"), Description (nick = "Passive")]
- PASSIVE,
-
- [DBus (value = "Active"), Description (nick = "Active")]
- ACTIVE,
-
- [DBus (value = "NeedsAttention"), Description (nick = "NeedsAttention")]
- NEEDS_ATTENTION;
-
- public string to_nick () {
- var enumc = (EnumClass)typeof (Status).class_ref();
- unowned var eval = enumc.get_value(this);
- return eval.value_nick;
- }
-}
-
-[DBus (name="org.kde.StatusNotifierItem")]
-internal interface IItem : DBusProxy {
- public abstract string Title { owned get; }
- public abstract Category Category { owned get; }
- public abstract Status Status { owned get; }
- public abstract Tooltip? ToolTip { owned get; }
- public abstract string Id { owned get; }
- public abstract string? IconThemePath { owned get; }
- public abstract bool ItemIsMenu { owned get; }
- public abstract ObjectPath? Menu { owned get; }
- public abstract string IconName { owned get; }
- public abstract Pixmap[] IconPixmap { owned get; }
- public abstract string AttentionIconName { owned get; }
- public abstract Pixmap[] AttentionIconPixmap { owned get; }
- public abstract string OverlayIconName { owned get; }
- public abstract Pixmap[] OverlayIconPixmap { owned get; }
-
- public abstract void ContexMenu(int x, int y) throws DBusError, IOError;
- public abstract void Activate(int x, int y) throws DBusError, IOError;
- public abstract void SecondaryActivate(int x, int y) throws DBusError, IOError;
- public abstract void Scroll(int delta, string orientation) throws DBusError, IOError;
-
- public signal void NewTitle();
- public signal void NewIcon();
- public signal void NewAttentionIcon();
- public signal void NewOverlayIcon();
- public signal void NewToolTip();
- public signal void NewStatus(string status);
-}
-
-public class TrayItem : Object {
- private IItem proxy;
- private List<ulong> connection_ids;
-
- public string title { owned get { return proxy.Title; } }
- public Category category { get { return proxy.Category; } }
- public Status status { get { return proxy.Status; } }
- public Tooltip? tooltip { owned get { return proxy.ToolTip; } }
-
- public string tooltip_markup {
- owned get {
- if (proxy.ToolTip == null)
- return "";
-
- var tt = proxy.ToolTip.title;
- if (proxy.ToolTip.description != "")
- tt += "\n" + proxy.ToolTip.description;
-
- return tt;
- }
- }
-
- public string id { owned get { return proxy.Id ;} }
- public string icon_theme_path { owned get { return proxy.IconThemePath ;} }
- public bool is_menu { get { return proxy.ItemIsMenu ;} }
-
- public string icon_name {
- owned get {
- return proxy.Status == Status.NEEDS_ATTENTION
- ? proxy.AttentionIconName
- : proxy.IconName;
- }
- }
-
- public Gdk.Pixbuf icon_pixbuf { owned get { return _get_icon_pixbuf(); } }
-
- public GLib.Icon gicon { get; private set; }
-
- public string item_id { get; private set; }
-
- public signal void changed();
- public signal void ready();
-
- public TrayItem(string service, string path) {
- connection_ids = new List<ulong>();
- item_id = service + path;
- setup_proxy.begin(service, path, (_, res) => setup_proxy.end(res));
- }
-
- private async void setup_proxy(string service, string path) {
- try {
- proxy = yield Bus.get_proxy(
- BusType.SESSION,
- service,
- path);
-
- connection_ids.append(proxy.NewStatus.connect(refresh_all_properties));
- connection_ids.append(proxy.NewToolTip.connect(refresh_all_properties));
- connection_ids.append(proxy.NewTitle.connect(refresh_all_properties));
- connection_ids.append(proxy.NewIcon.connect(refresh_all_properties));
-
- proxy.notify["g-name-owner"].connect(() => {
- if (proxy.g_name_owner == null) {
- foreach (var id in connection_ids)
- SignalHandler.disconnect(proxy, id);
- }
- });
-
- update_gicon();
-
- ready();
- } catch (Error err) {
- critical(err.message);
- }
- }
-
- private void _notify() {
- string[] props = { "category", "id", "title", "status", "is-menu", "tooltip-markup", "icon-name", "icon-pixbuf" };
-
- foreach (string prop in props)
- notify_property(prop);
-
- changed();
- }
-
- private void update_gicon() {
- if(icon_name != null && icon_name != "") {
- if(icon_theme_path != null && icon_theme_path != "") {
-
- Gtk.IconTheme icon_theme = new Gtk.IconTheme();
- string[] paths = {icon_theme_path};
- icon_theme.set_search_path(paths);
-
- int size = icon_theme.get_icon_sizes(icon_name)[0];
- Gtk.IconInfo icon_info = icon_theme.lookup_icon(
- icon_name, size, Gtk.IconLookupFlags.FORCE_SIZE);
-
- if (icon_info != null)
- gicon = new GLib.FileIcon(GLib.File.new_for_path(icon_info.get_filename()));
- } else {
- gicon = new GLib.ThemedIcon(icon_name);
- }
- }
- else {
- Pixmap[] pixmaps = proxy.Status == Status.NEEDS_ATTENTION
- ? proxy.AttentionIconPixmap
- : proxy.IconPixmap;
- gicon = pixmap_to_pixbuf(pixmaps);
- }
- }
-
-
- private void refresh_all_properties() {
- proxy.g_connection.call.begin(
- proxy.g_name,
- proxy.g_object_path,
- "org.freedesktop.DBus.Properties",
- "GetAll",
- new Variant("(s)", proxy.g_interface_name),
- new VariantType("(a{sv})"),
- DBusCallFlags.NONE,
- -1,
- null,
- (_, result) => {
- try {
- Variant parameters = proxy.g_connection.call.end(result);
- VariantIter prop_iter;
- parameters.get("(a{sv})", out prop_iter);
-
- string prop_key;
- Variant prop_value;
-
- while (prop_iter.next ("{sv}", out prop_key, out prop_value)) {
- proxy.set_cached_property(prop_key, prop_value);
- }
-
- update_gicon();
-
- _notify();
- } catch(Error e) {
- //silently ignore
- }
- }
- );
- }
-
- public void activate(int x, int y) {
- try {
- proxy.Activate(x, y);
- } catch (Error e) {
- if(e.domain != DBusError.quark() || e.code != DBusError.UNKNOWN_METHOD)
- warning(e.message);
- }
- }
-
- public void secondary_activate(int x, int y) {
- try {
- proxy.SecondaryActivate(x, y);
- } catch (Error e) {
- if(e.domain != DBusError.quark() || e.code != DBusError.UNKNOWN_METHOD)
- warning(e.message);
- }
- }
-
- public void scroll(int delta, string orientation) {
- try {
- proxy.Scroll(delta, orientation);
- } catch (Error e) {
- if(e.domain != DBusError.quark() || e.code != DBusError.UNKNOWN_METHOD)
- warning("%s\n", e.message);
- }
- }
-
-
- public DbusmenuGtk.Menu? create_menu() {
- if (proxy.Menu == null)
- return null;
-
- return new DbusmenuGtk.Menu(
- proxy.get_name_owner(),
- proxy.Menu);
- }
-
- public Gdk.Pixbuf? _get_icon_pixbuf() {
- Pixmap[] pixmaps = proxy.Status == Status.NEEDS_ATTENTION
- ? proxy.AttentionIconPixmap
- : proxy.IconPixmap;
-
-
- string icon_name = proxy.Status == Status.NEEDS_ATTENTION
- ? proxy.AttentionIconName
- : proxy.IconName;
-
- Gdk.Pixbuf pixbuf = null;
-
- if (icon_name != null && proxy.IconThemePath != null)
- pixbuf = load_from_theme(icon_name, proxy.IconThemePath);
-
- if (pixbuf == null)
- pixbuf = pixmap_to_pixbuf(pixmaps);
-
- return pixbuf;
- }
-
- private Gdk.Pixbuf? load_from_theme(string icon_name, string theme_path) {
- if (theme_path == "" || theme_path == null)
- return null;
-
- if (icon_name == "" || icon_name == null)
- return null;
-
- Gtk.IconTheme icon_theme = new Gtk.IconTheme();
- string[] paths = {theme_path};
- icon_theme.set_search_path(paths);
-
- int size = icon_theme.get_icon_sizes(icon_name)[0];
- Gtk.IconInfo icon_info = icon_theme.lookup_icon(
- icon_name, size, Gtk.IconLookupFlags.FORCE_SIZE);
-
- if (icon_info != null)
- return icon_info.load_icon();
-
- return null;
- }
-
- private Gdk.Pixbuf? pixmap_to_pixbuf(Pixmap[] pixmaps) {
- if (pixmaps == null || pixmaps.length == 0)
- return null;
-
- Pixmap pixmap = pixmaps[0];
- uint8[] image_data = pixmap.bytes.copy();
-
- for (int i = 0; i < pixmap.width * pixmap.height * 4; i += 4) {
- uint8 alpha = image_data[i];
- image_data[i] = image_data[i + 1];
- image_data[i + 1] = image_data[i + 2];
- image_data[i + 2] = image_data[i + 3];
- image_data[i + 3] = alpha;
- }
-
- return new Gdk.Pixbuf.from_bytes(
- new Bytes(image_data),
- Gdk.Colorspace.RGB,
- true,
- 8,
- (int)pixmap.width,
- (int)pixmap.height,
- (int)(pixmap.width * 4)
- );
- }
-
- public string to_json_string() {
- var generator = new Json.Generator();
- generator.set_root(to_json());
- return generator.to_data(null);
- }
-
- internal Json.Node to_json() {
- return new Json.Builder()
- .begin_object()
- .set_member_name("item_id").add_string_value(item_id)
- .set_member_name("id").add_string_value(id)
- .set_member_name("bus_name").add_string_value(proxy.g_name)
- .set_member_name("object_path").add_string_value(proxy.g_object_path)
- .set_member_name("title").add_string_value(title)
- .set_member_name("status").add_string_value(status.to_nick())
- .set_member_name("category").add_string_value(category.to_nick())
- .set_member_name("tooltip").add_string_value(tooltip_markup)
- .set_member_name("icon_theme_path").add_string_value(proxy.IconThemePath)
- .set_member_name("icon_name").add_string_value(icon_name)
- .set_member_name("menu_path").add_string_value(proxy.Menu)
- .set_member_name("is_menu").add_boolean_value(is_menu)
- .end_object()
- .get_root();
- }
-}
-}
diff --git a/tray/src/watcher.vala b/tray/src/watcher.vala
deleted file mode 100644
index 974cd02..0000000
--- a/tray/src/watcher.vala
+++ /dev/null
@@ -1,59 +0,0 @@
-namespace AstalTray {
-[DBus (name="org.kde.StatusNotifierWatcher")]
-internal class StatusNotifierWatcher : Object {
- private HashTable<string, string> _items =
- new HashTable<string, string>(str_hash, str_equal);
-
- public string[] RegisteredStatusNotifierItems { owned get { return _items.get_values_as_ptr_array().data; } }
- public bool IsStatusNotifierHostRegistered { get; default = true; }
- public int ProtocolVersion { get; default = 0; }
-
- public signal void StatusNotifierItemRegistered(string service);
- public signal void StatusNotifierItemUnregistered(string service);
- public signal void StatusNotifierHostRegistered();
- public signal void StatusNotifierHostUnregistered();
-
- public void RegisterStatusNotifierItem(string service, BusName sender) throws DBusError, IOError {
- string busName;
- string path;
- if (service[0] == '/') {
- path = service;
- busName = sender;
- } else {
- busName = service;
- path = "/StatusNotifierItem";
- }
-
- Bus.get_sync(BusType.SESSION).signal_subscribe(
- null,
- "org.freedesktop.DBus",
- "NameOwnerChanged",
- null,
- null,
- DBusSignalFlags.NONE,
- (connection, sender_name, path, interface_name, signal_name, parameters) => {
- string name = null;
- string new_owner = null;
- string old_owner = null;
- parameters.get("(sss)", &name, &old_owner, &new_owner);
- if (new_owner == "" && _items.contains(old_owner)) {
- string full_path = _items.take(old_owner);
- StatusNotifierItemUnregistered(full_path);
- }
- }
- );
-
- _items.set(busName, busName+path);
- StatusNotifierItemRegistered(busName+path);
- }
-
- public void RegisterStatusNotifierHost(string service) throws DBusError, IOError {
- /* NOTE:
- usually the watcher should keep track of registered host
- but some tray applications do net register their trayitem properly
- when hosts register/deregister. This is fixed by setting isHostRegistered
- always to true, this also make host handling logic unneccessary.
- */
- }
-}
-}