1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
#include "openbox/actions.h"
#include "openbox/event.h"
#include "openbox/startupnotify.h"
#include "openbox/screen.h"
#include "gettext.h"
typedef struct {
gchar *cmd;
gboolean sn;
gchar *sn_name;
gchar *sn_icon;
gchar *sn_wmclass;
} Options;
static gpointer setup_func(xmlNodePtr node);
static void free_func(gpointer options);
static gboolean run_func(ObActionsData *data, gpointer options);
/*
static gboolean i_input_func(guint initial_state,
XEvent *e,
gpointer options,
gboolean *used);
static void i_cancel_func(gpointer options);
*/
void action_execute_startup(void)
{
actions_register("Execute", setup_func, free_func, run_func, NULL, NULL);
}
static gpointer setup_func(xmlNodePtr node)
{
xmlNodePtr n;
Options *o;
o = g_new0(Options, 1);
if ((n = obt_parse_find_node(node, "command")) ||
(n = obt_parse_find_node(node, "execute")))
{
gchar *s = obt_parse_node_string(n);
o->cmd = parse_expand_tilde(s);
g_free(s);
}
if ((n = obt_parse_find_node(node, "startupnotify"))) {
xmlNodePtr m;
if ((m = obt_parse_find_node(n->xmlChildrenNode, "enabled")))
o->sn = obt_parse_node_bool(m);
if ((m = obt_parse_find_node(n->xmlChildrenNode, "name")))
o->sn_name = obt_parse_node_string(m);
if ((m = obt_parse_find_node(n->xmlChildrenNode, "icon")))
o->sn_icon = obt_parse_node_string(m);
if ((m = obt_parse_find_node(n->xmlChildrenNode, "wmclass")))
o->sn_wmclass = obt_parse_node_string(m);
}
return o;
}
static void free_func(gpointer options)
{
Options *o = options;
if (o) {
g_free(o->cmd);
g_free(o->sn_name);
g_free(o->sn_icon);
g_free(o->sn_wmclass);
g_free(o);
}
}
/* Always return FALSE because its not interactive */
static gboolean run_func(ObActionsData *data, gpointer options)
{
GError *e = NULL;
gchar **argv = NULL;
gchar *cmd;
Options *o = options;
if (!o->cmd) return FALSE;
cmd = g_filename_from_utf8(o->cmd, -1, NULL, NULL, NULL);
if (!cmd) {
g_message(_("Failed to convert the path '%s' from utf8"), o->cmd);
return FALSE;
}
/* If there is a keyboard grab going on then we need to cancel
it so the application can grab things */
event_cancel_all_key_grabs();
if (!g_shell_parse_argv(cmd, NULL, &argv, &e)) {
g_message(_("Failed to execute '%s': %s"), o->cmd, e->message);
g_error_free(e);
}
else {
gchar *program = NULL;
if (o->sn) {
program = g_path_get_basename(argv[0]);
/* sets up the environment */
sn_setup_spawn_environment(program, o->sn_name, o->sn_icon,
/* launch it on the current desktop */
screen_desktop);
}
if (!g_spawn_async(NULL, argv, NULL,
G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD,
NULL, NULL, NULL, &e))
{
g_message(_("Failed to execute '%s': %s"), o->cmd, e->message);
g_error_free(e);
if (o->sn)
sn_spawn_cancel();
}
if (o->sn)
unsetenv("DESKTOP_STARTUP_ID");
g_free(program);
g_strfreev(argv);
}
g_free(cmd);
return FALSE;
}
|