/************************************************************************** * Tint2 : launcher * * Copyright (C) 2010 (mrovi@interfete-web-club.com) * * SVG support: https://github.com/ixxra/tint2-svg * Copyright (C) 2010 Rene Garcia (garciamx@gmail.com) * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License version 2 * as published by the Free Software Foundation. * * This program 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 this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. **************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include "window.h" #include "server.h" #include "area.h" #include "panel.h" #include "taskbar.h" #include "launcher.h" #include "apps-common.h" #include "icon-theme-common.h" int launcher_enabled; int launcher_max_icon_size; int launcher_tooltip_enabled; int launcher_alpha; int launcher_saturation; int launcher_brightness; char *icon_theme_name_config; char *icon_theme_name_xsettings; int launcher_icon_theme_override; int startup_notifications; Background *launcher_icon_bg; Imlib_Image scale_icon(Imlib_Image original, int icon_size); void free_icon(Imlib_Image icon); void default_launcher() { launcher_enabled = 0; launcher_max_icon_size = 0; launcher_tooltip_enabled = 0; launcher_alpha = 100; launcher_saturation = 0; launcher_brightness = 0; icon_theme_name_config = NULL; icon_theme_name_xsettings = NULL; launcher_icon_theme_override = 0; startup_notifications = 0; launcher_icon_bg = NULL; } void init_launcher() { } void init_launcher_panel(void *p) { Panel *panel = (Panel *)p; Launcher *launcher = &panel->launcher; launcher->area.parent = p; launcher->area.panel = p; launcher->area._draw_foreground = NULL; launcher->area.size_mode = LAYOUT_FIXED; launcher->area._resize = resize_launcher; launcher->area.resize_needed = 1; schedule_redraw(&launcher->area); if (!launcher->area.bg) launcher->area.bg = &g_array_index(backgrounds, Background, 0); if (!launcher_icon_bg) launcher_icon_bg = &g_array_index(backgrounds, Background, 0); // check consistency if (launcher->list_apps == NULL) return; launcher->area.on_screen = TRUE; panel_refresh = TRUE; launcher_load_themes(launcher); launcher_load_icons(launcher); } void cleanup_launcher() { for (int i = 0; i < num_panels; i++) { Panel *panel = &panels[i]; Launcher *launcher = &panel->launcher; cleanup_launcher_theme(launcher); } for (GSList *l = panel_config.launcher.list_apps; l; l = l->next) { free(l->data); } g_slist_free(panel_config.launcher.list_apps); panel_config.launcher.list_apps = NULL; free(icon_theme_name_config); icon_theme_name_config = NULL; free(icon_theme_name_xsettings); icon_theme_name_xsettings = NULL; launcher_enabled = FALSE; } void cleanup_launcher_theme(Launcher *launcher) { free_area(&launcher->area); for (GSList *l = launcher->list_icons; l; l = l->next) { LauncherIcon *launcherIcon = (LauncherIcon *)l->data; if (launcherIcon) { free_icon(launcherIcon->image); free_icon(launcherIcon->image_hover); free_icon(launcherIcon->image_pressed); free(launcherIcon->icon_name); free(launcherIcon->icon_path); free(launcherIcon->cmd); g_free(launcherIcon->icon_tooltip); } free(launcherIcon); } g_slist_free(launcher->list_icons); launcher->list_icons = NULL; free_themes(launcher->icon_theme_wrapper); launcher->icon_theme_wrapper = NULL; } gboolean resize_launcher(void *obj) { Launcher *launcher = obj; int icons_per_column = 1, icons_per_row = 1, margin = 0; int icon_size; if (panel_horizontal) { icon_size = launcher->area.height; } else { icon_size = launcher->area.width; } icon_size = icon_size - (2 * launcher->area.bg->border.width) - (2 * launcher->area.paddingy); if (launcher_max_icon_size > 0 && icon_size > launcher_max_icon_size) icon_size = launcher_max_icon_size; // Resize icons if necessary for (GSList *l = launcher->list_icons; l; l = l->next) { LauncherIcon *launcherIcon = (LauncherIcon *)l->data; if (launcherIcon->icon_size != icon_size || !launcherIcon->image) { launcherIcon->icon_size = icon_size; launcherIcon->area.width = launcherIcon->icon_size; launcherIcon->area.height = launcherIcon->icon_size; // Get the path for an icon file with the new size char *new_icon_path = get_icon_path(launcher->icon_theme_wrapper, launcherIcon->icon_name, launcherIcon->icon_size); if (!new_icon_path) { // Draw a blank icon free_icon(launcherIcon->image); free_icon(launcherIcon->image_hover); free_icon(launcherIcon->image_pressed); launcherIcon->image = NULL; continue; } // Free the old files free_icon(launcherIcon->image); free_icon(launcherIcon->image_hover); free_icon(launcherIcon->image_pressed); // Load the new file launcherIcon->image = load_image(new_icon_path, 1); // On loading error, fallback to default if (!launcherIcon->image) { free(new_icon_path); new_icon_path = get_icon_path(launcher->icon_theme_wrapper, DEFAULT_ICON, launcherIcon->icon_size); if (new_icon_path) launcherIcon->image = imlib_load_image_immediately(new_icon_path); } if (!launcherIcon->image) { // Loading default icon failed, draw a blank icon free(new_icon_path); } else { // Loaded icon successfully, rescale it Imlib_Image original = launcherIcon->image; launcherIcon->image = scale_icon(launcherIcon->image, launcherIcon->icon_size); free_icon(original); free(launcherIcon->icon_path); launcherIcon->icon_path = new_icon_path; fprintf(stderr, "launcher.c %d: Using icon %s\n", __LINE__, launcherIcon->icon_path); } } if (panel_config.mouse_effects) { launcherIcon->image_hover = adjust_icon(launcherIcon->image, panel_config.mouse_over_alpha, panel_config.mouse_over_saturation, panel_config.mouse_over_brightness); launcherIcon->image_pressed = adjust_icon(launcherIcon->image, panel_config.mouse_pressed_alpha, panel_config.mouse_pressed_saturation, panel_config.mouse_pressed_brightness); } } save_icon_cache(launcher->icon_theme_wrapper); int count = g_slist_length(launcher->list_icons); if (panel_horizontal) { if (!count) { launcher->area.width = 0; } else { int height = launcher->area.height - 2 * launcher->area.bg->border.width - 2 * launcher->area.paddingy; // here icons_per_column always higher than 0 icons_per_column = (height + launcher->area.paddingx) / (icon_size + launcher->area.paddingx); margin = height - (icons_per_column - 1) * (icon_size + launcher->area.paddingx) - icon_size; icons_per_row = count / icons_per_column + (count % icons_per_column != 0); launcher->area.width = (2 * launcher->area.bg->border.width) + (2 * launcher->area.paddingxlr) + (icon_size * icons_per_row) + ((icons_per_row - 1) * launcher->area.paddingx); } } else { if (!count) { launcher->area.height = 0; } else { int width = launcher->area.width - 2 * launcher->area.bg->border.width - 2 * launcher->area.paddingy; // here icons_per_row always higher than 0 icons_per_row = (width + launcher->area.paddingx) / (icon_size + launcher->area.paddingx); margin = width - (icons_per_row - 1) * (icon_size + launcher->area.paddingx) - icon_size; icons_per_column = count / icons_per_row + (count % icons_per_row != 0); launcher->area.height = (2 * launcher->area.bg->border.width) + (2 * launcher->area.paddingxlr) + (icon_size * icons_per_column) + ((icons_per_column - 1) * launcher->area.paddingx); } } int posx, posy; int start = launcher->area.bg->border.width + launcher->area.paddingy + margin / 2; if (panel_horizontal) { posy = start; posx = launcher->area.bg->border.width + launcher->area.paddingxlr; } else { posx = start; posy = launcher->area.bg->border.width + launcher->area.paddingxlr; } int i; GSList *l; for (i = 1, l = launcher->list_icons; l; i++, l = l->next) { LauncherIcon *launcherIcon = (LauncherIcon *)l->data; launcherIcon->y = posy; launcherIcon->x = posx; launcherIcon->area.posy = ((Area *)launcherIcon->area.parent)->posy + launcherIcon->y; launcherIcon->area.posx = ((Area *)launcherIcon->area.parent)->posx + launcherIcon->x; launcherIcon->area.width = launcherIcon->icon_size; launcherIcon->area.height = launcherIcon->icon_size; // printf("launcher %d : %d,%d\n", i, posx, posy); if (panel_horizontal) { if (i % icons_per_column) { posy += icon_size + launcher->area.paddingx; } else { posy = start; posx += (icon_size + launcher->area.paddingx); } } else { if (i % icons_per_row) { posx += icon_size + launcher->area.paddingx; } else { posx = start; posy += (icon_size + launcher->area.paddingx); } } } return TRUE; } // Here we override the default layout of the icons; normally Area layouts its children // in a stack; we need to layout them in a kind of table void launcher_icon_on_change_layout(void *obj) { LauncherIcon *launcherIcon = (LauncherIcon *)obj; launcherIcon->area.posy = ((Area *)launcherIcon->area.parent)->posy + launcherIcon->y; launcherIcon->area.posx = ((Area *)launcherIcon->area.parent)->posx + launcherIcon->x; launcherIcon->area.width = launcherIcon->icon_size; launcherIcon->area.height = launcherIcon->icon_size; } char *launcher_icon_get_tooltip_text(void *obj) { LauncherIcon *launcherIcon = (LauncherIcon *)obj; return strdup(launcherIcon->icon_tooltip); } void draw_launcher_icon(void *obj, cairo_t *c) { LauncherIcon *launcherIcon = (LauncherIcon *)obj; Imlib_Image image; // Render if (panel_config.mouse_effects) { if (launcherIcon->area.mouse_state == MOUSE_OVER) image = launcherIcon->image_hover ? launcherIcon->image_hover : launcherIcon->image; else if (launcherIcon->area.mouse_state == MOUSE_DOWN) image = launcherIcon->image_pressed ? launcherIcon->image_pressed : launcherIcon->image; else image = launcherIcon->image; } else { image = launcherIcon->image; } imlib_context_set_image(image); render_image(launcherIcon->area.pix, 0, 0); } Imlib_Image scale_icon(Imlib_Image original, int icon_size) { Imlib_Image icon_scaled; if (original) { imlib_context_set_image(original); icon_scaled = imlib_create_cropped_scaled_image(0, 0, imlib_image_get_width(), imlib_image_get_height(), icon_size, icon_size); imlib_context_set_image(icon_scaled); imlib_image_set_has_alpha(1); DATA32 *data = imlib_image_get_data(); adjust_asb(data, icon_size, icon_size, launcher_alpha, launcher_saturation / 100.0, launcher_brightness / 100.0); imlib_image_put_back_data(data); imlib_context_set_image(icon_scaled); } else { icon_scaled = imlib_create_image(icon_size, icon_size); imlib_context_set_image(icon_scaled); imlib_context_set_color(255, 255, 255, 255); imlib_image_fill_rectangle(0, 0, icon_size, icon_size); } return icon_scaled; } void free_icon(Imlib_Image icon) { if (icon) { imlib_context_set_image(icon); imlib_free_image(); } } void launcher_action(LauncherIcon *icon, XEvent *evt) { char *cmd = calloc(strlen(icon->cmd) + 10, 1); sprintf(cmd, "(%s&)", icon->cmd); #if HAVE_SN SnLauncherContext *ctx = 0; Time time; if (startup_notifications) { ctx = sn_launcher_context_new(server.sn_display, server.screen); sn_launcher_context_set_name(ctx, icon->icon_tooltip); sn_launcher_context_set_description(ctx, "Application launched from tint2"); sn_launcher_context_set_binary_name(ctx, icon->cmd); // Get a timestamp from the X event if (evt->type == ButtonPress || evt->type == ButtonRelease) { time = evt->xbutton.time; } else { fprintf(stderr, "Unknown X event: %d\n", evt->type); free(cmd); return; } sn_launcher_context_initiate(ctx, "tint2", icon->cmd, time); } #endif /* HAVE_SN */ pid_t pid; pid = fork(); if (pid < 0) { fprintf(stderr, "Could not fork\n"); } else if (pid == 0) { // Child process #if HAVE_SN if (startup_notifications) { sn_launcher_context_setup_child_process(ctx); } #endif // HAVE_SN // Allow children to exist after parent destruction setsid(); // Run the command execl("/bin/sh", "/bin/sh", "-c", icon->cmd, NULL); fprintf(stderr, "Failed to execlp %s\n", icon->cmd); #if HAVE_SN if (startup_notifications) { sn_launcher_context_unref(ctx); } #endif // HAVE_SN exit(1); } else { // Parent process #if HAVE_SN if (startup_notifications) { g_tree_insert(server.pids, GINT_TO_POINTER(pid), ctx); } #endif // HAVE_SN } free(cmd); } // Populates the list_icons list from the list_apps list void launcher_load_icons(Launcher *launcher) { // Load apps (.desktop style launcher items) GSList *app = launcher->list_apps; while (app != NULL) { DesktopEntry entry; read_desktop_file(app->data, &entry); if (entry.exec) { LauncherIcon *launcherIcon = calloc(1, sizeof(LauncherIcon)); launcherIcon->area.panel = launcher->area.panel; launcherIcon->area._draw_foreground = draw_launcher_icon; launcherIcon->area.size_mode = LAYOUT_FIXED; launcherIcon->area._resize = NULL; launcherIcon->area.resize_needed = 0; schedule_redraw(&launcherIcon->area); launcherIcon->area.has_mouse_over_effect = panel_config.mouse_effects; launcherIcon->area.has_mouse_press_effect = launcherIcon->area.has_mouse_over_effect; launcherIcon->area.bg = launcher_icon_bg; launcherIcon->area.on_screen = TRUE; launcherIcon->area._on_change_layout = launcher_icon_on_change_layout; if (launcher_tooltip_enabled) { launcherIcon->area._get_tooltip_text = launcher_icon_get_tooltip_text; } else { launcherIcon->area._get_tooltip_text = NULL; } launcherIcon->is_app_desktop = 1; launcherIcon->cmd = strdup(entry.exec); launcherIcon->icon_name = entry.icon ? strdup(entry.icon) : strdup(DEFAULT_ICON); launcherIcon->icon_size = 1; if (entry.name) { if (entry.generic_name) { launcherIcon->icon_tooltip = g_strdup_printf("%s (%s)", entry.name, entry.generic_name); } else { launcherIcon->icon_tooltip = g_strdup_printf("%s", entry.name); } } else { if (entry.generic_name) { launcherIcon->icon_tooltip = g_strdup_printf("%s", entry.generic_name); } else if (entry.exec) { launcherIcon->icon_tooltip = g_strdup_printf("%s", entry.exec); } } launcher->list_icons = g_slist_append(launcher->list_icons, launcherIcon); add_area(&launcherIcon->area, (Area *)launcher); } free_desktop_entry(&entry); app = g_slist_next(app); } } // Populates the icon_theme_wrapper list void launcher_load_themes(Launcher *launcher) { launcher->icon_theme_wrapper = load_themes(launcher_icon_theme_override ? (icon_theme_name_config ? icon_theme_name_config : icon_theme_name_xsettings ? icon_theme_name_xsettings : "hicolor") : (icon_theme_name_xsettings ? icon_theme_name_xsettings : icon_theme_name_config ? icon_theme_name_config : "hicolor")); } void launcher_default_icon_theme_changed() { if (!launcher_enabled) return; if (launcher_icon_theme_override && icon_theme_name_config) return; for (int i = 0; i < num_panels; i++) { Launcher *launcher = &panels[i].launcher; cleanup_launcher_theme(launcher); launcher_load_themes(launcher); launcher_load_icons(launcher); launcher->area.resize_needed = 1; } panel_refresh = TRUE; }