493 lines
15 KiB
C
493 lines
15 KiB
C
/**************************************************************************
|
|
* 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 <string.h>
|
|
#include <stdio.h>
|
|
#include <cairo.h>
|
|
#include <cairo-xlib.h>
|
|
#include <pango/pangocairo.h>
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <stdlib.h>
|
|
#include <glib/gi18n.h>
|
|
#include <glib.h>
|
|
#include <glib/gstdio.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <sys/types.h>
|
|
|
|
#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;
|
|
XSettingsClient *xsettings_client;
|
|
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;
|
|
xsettings_client = NULL;
|
|
startup_notifications = 0;
|
|
launcher_icon_bg = NULL;
|
|
}
|
|
|
|
|
|
void init_launcher()
|
|
{
|
|
if (launcher_enabled) {
|
|
// if XSETTINGS manager running, tint2 read the icon_theme_name.
|
|
xsettings_client = xsettings_client_new(server.dsp, server.screen, xsettings_notify_cb, NULL, NULL);
|
|
}
|
|
}
|
|
|
|
|
|
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;
|
|
launcher->area.redraw_needed = 1;
|
|
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 = 1;
|
|
panel_refresh = 1;
|
|
|
|
launcher_load_themes(launcher);
|
|
launcher_load_icons(launcher);
|
|
}
|
|
|
|
|
|
void cleanup_launcher()
|
|
{
|
|
int i;
|
|
GSList *l;
|
|
|
|
if (xsettings_client)
|
|
xsettings_client_destroy(xsettings_client);
|
|
xsettings_client = NULL;
|
|
|
|
for (i = 0; i < nb_panel; i++) {
|
|
Panel *panel = &panel1[i];
|
|
Launcher *launcher = &panel->launcher;
|
|
cleanup_launcher_theme(launcher);
|
|
}
|
|
|
|
for (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 = 0;
|
|
}
|
|
|
|
|
|
void cleanup_launcher_theme(Launcher *launcher)
|
|
{
|
|
free_area(&launcher->area);
|
|
GSList *l;
|
|
for (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);
|
|
free(launcherIcon->icon_tooltip);
|
|
}
|
|
free(launcherIcon);
|
|
}
|
|
g_slist_free(launcher->list_icons);
|
|
launcher->list_icons = NULL;
|
|
|
|
free_themes(launcher->list_themes);
|
|
launcher->list_themes = NULL;
|
|
}
|
|
|
|
|
|
int resize_launcher(void *obj)
|
|
{
|
|
Launcher *launcher = obj;
|
|
GSList *l;
|
|
int count, icon_size;
|
|
int icons_per_column=1, icons_per_row=1, marging=0;
|
|
|
|
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 (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->list_themes, 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->list_themes, 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);
|
|
}
|
|
}
|
|
|
|
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);
|
|
marging = 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);
|
|
marging = 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 i, posx, posy;
|
|
int start = launcher->area.bg->border.width + launcher->area.paddingy + marging/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;
|
|
}
|
|
|
|
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 1;
|
|
}
|
|
|
|
// 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, (float)launcher_saturation/100, (float)launcher_brightness/100);
|
|
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_dsp, 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;
|
|
launcherIcon->area.redraw_needed = 1;
|
|
launcherIcon->area.has_mouse_over_effect = 1;
|
|
launcherIcon->area.has_mouse_press_effect = 1;
|
|
launcherIcon->area.bg = launcher_icon_bg;
|
|
launcherIcon->area.on_screen = 1;
|
|
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;
|
|
launcherIcon->icon_tooltip = entry.name ? strdup(entry.name) : strdup(entry.exec);
|
|
free_desktop_entry(&entry);
|
|
launcher->list_icons = g_slist_append(launcher->list_icons, launcherIcon);
|
|
add_area(&launcherIcon->area, (Area*)launcher);
|
|
}
|
|
app = g_slist_next(app);
|
|
}
|
|
}
|
|
|
|
|
|
// Populates the list_themes list
|
|
void launcher_load_themes(Launcher *launcher)
|
|
{
|
|
launcher->list_themes = 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"));
|
|
}
|