]> Dogcows Code - chaz/tint2/blob - src/taskbar/taskbar.c
fixed issue 110 and issue 107
[chaz/tint2] / src / taskbar / taskbar.c
1 /**************************************************************************
2 *
3 * Tint2 : taskbar
4 *
5 * Copyright (C) 2008 thierry lorthiois (lorthiois@bbsoft.fr)
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License version 2
9 * as published by the Free Software Foundation.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
18 **************************************************************************/
19
20 #include <X11/Xlib.h>
21 #include <X11/Xutil.h>
22 #include <X11/Xatom.h>
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26 #include <glib.h>
27 #include <Imlib2.h>
28
29 #include "task.h"
30 #include "taskbar.h"
31 #include "server.h"
32 #include "window.h"
33 #include "panel.h"
34
35
36
37 void init_taskbar()
38 {
39 Panel *panel;
40 int i, j;
41
42 for (i=0 ; i < nb_panel ; i++) {
43 panel = &panel1[i];
44
45 if (panel->taskbar) {
46 free(panel->taskbar);
47 panel->taskbar = 0;
48 }
49
50 // taskbar
51 panel->g_taskbar._resize = resize_taskbar;
52 panel->g_taskbar.redraw = 1;
53 panel->g_taskbar.on_screen = 1;
54 if (panel_horizontal) {
55 panel->g_taskbar.posy = panel->area.pix.border.width + panel->area.paddingy;
56 panel->g_taskbar.height = panel->area.height - (2 * panel->g_taskbar.posy);
57 }
58 else {
59 panel->g_taskbar.posx = panel->area.pix.border.width + panel->area.paddingy;
60 panel->g_taskbar.width = panel->area.width - (2 * panel->g_taskbar.posx);
61 }
62
63 // task
64 panel->g_task.area._draw_foreground = draw_task;
65 panel->g_task.area.use_active = 1;
66 panel->g_task.area.redraw = 1;
67 panel->g_task.area.on_screen = 1;
68 if (panel_horizontal) {
69 panel->g_task.area.posy = panel->g_taskbar.posy + panel->g_taskbar.pix.border.width + panel->g_taskbar.paddingy;
70 panel->g_task.area.height = panel->area.height - (2 * panel->g_task.area.posy);
71 }
72 else {
73 panel->g_task.area.posx = panel->g_taskbar.posx + panel->g_taskbar.pix.border.width + panel->g_taskbar.paddingy;
74 panel->g_task.area.width = panel->area.width - (2 * panel->g_task.area.posx);
75 panel->g_task.area.height = panel->g_task.maximum_height;
76 }
77
78 if (panel->g_task.area.pix.border.rounded > panel->g_task.area.height/2) {
79 panel->g_task.area.pix.border.rounded = panel->g_task.area.height/2;
80 panel->g_task.area.pix_active.border.rounded = panel->g_task.area.pix.border.rounded;
81 }
82
83 // compute vertical position : text and icon
84 int height_ink, height;
85 get_text_size(panel->g_task.font_desc, &height_ink, &height, panel->area.height, "TAjpg", 5);
86
87 if (!panel->g_task.maximum_width && panel_horizontal)
88 panel->g_task.maximum_width = server.monitor[panel->monitor].width;
89
90 panel->g_task.text_posx = panel->g_task.area.pix.border.width + panel->g_task.area.paddingxlr;
91 panel->g_task.text_posy = (panel->g_task.area.height - height) / 2.0;
92 if (panel->g_task.icon) {
93 panel->g_task.icon_size1 = panel->g_task.area.height - (2 * panel->g_task.area.paddingy);
94 panel->g_task.text_posx += panel->g_task.icon_size1;
95 panel->g_task.icon_posy = (panel->g_task.area.height - panel->g_task.icon_size1) / 2;
96 }
97 //printf("monitor %d, task_maximum_width %d\n", panel->monitor, panel->g_task.maximum_width);
98
99 Taskbar *tskbar;
100 panel->nb_desktop = server.nb_desktop;
101 panel->taskbar = calloc(panel->nb_desktop, sizeof(Taskbar));
102 for (j=0 ; j < panel->nb_desktop ; j++) {
103 tskbar = &panel->taskbar[j];
104 memcpy(&tskbar->area, &panel->g_taskbar, sizeof(Area));
105 tskbar->desktop = j;
106
107 // add taskbar to the panel
108 panel->area.list = g_slist_append(panel->area.list, tskbar);
109 }
110 }
111 }
112
113
114 void cleanup_taskbar()
115 {
116 Panel *panel;
117 Taskbar *tskbar;
118 int i, j;
119 GSList *l0;
120 Task *tsk;
121
122 for (i=0 ; i < nb_panel ; i++) {
123 panel = &panel1[i];
124
125 for (j=0 ; j < panel->nb_desktop ; j++) {
126 tskbar = &panel->taskbar[j];
127 l0 = tskbar->area.list;
128 while (l0) {
129 tsk = l0->data;
130 l0 = l0->next;
131 // careful : remove_task change l0->next
132 remove_task (tsk);
133 }
134 free_area (&tskbar->area);
135
136 // remove taskbar from the panel
137 panel->area.list = g_slist_remove(panel->area.list, tskbar);
138 }
139 }
140
141 for (i=0 ; i < nb_panel ; i++) {
142 panel = &panel1[i];
143 if (panel->taskbar) {
144 free(panel->taskbar);
145 panel->taskbar = 0;
146 }
147 }
148 }
149
150
151 Task *task_get_task (Window win)
152 {
153 Task *tsk;
154 GSList *l0;
155 int i, j;
156
157 for (i=0 ; i < nb_panel ; i++) {
158 for (j=0 ; j < panel1[i].nb_desktop ; j++) {
159 for (l0 = panel1[i].taskbar[j].area.list; l0 ; l0 = l0->next) {
160 tsk = l0->data;
161 if (win == tsk->win)
162 return tsk;
163 }
164 }
165 }
166 return 0;
167 }
168
169
170 void task_refresh_tasklist ()
171 {
172 Window *win, active_win;
173 int num_results, i, j, k;
174 GSList *l0;
175 Task *tsk;
176
177 win = server_get_property (server.root_win, server.atom._NET_CLIENT_LIST, XA_WINDOW, &num_results);
178 if (!win) return;
179
180 // Remove any old and set active win
181 active_win = window_get_active ();
182 if (task_active) {
183 task_active->area.is_active = 0;
184 task_active = 0;
185 }
186
187 for (i=0 ; i < nb_panel ; i++) {
188 for (j=0 ; j < panel1[i].nb_desktop ; j++) {
189 l0 = panel1[i].taskbar[j].area.list;
190 while (l0) {
191 tsk = l0->data;
192 l0 = l0->next;
193
194 if (tsk->win == active_win) {
195 tsk->area.is_active = 1;
196 task_active = tsk;
197 }
198
199 for (k = 0; k < num_results; k++) {
200 if (tsk->win == win[k]) break;
201 }
202 // careful : remove_task change l0->next
203 if (k == num_results) remove_task (tsk);
204 }
205 }
206 }
207
208 // Add any new
209 for (i = 0; i < num_results; i++)
210 if (!task_get_task (win[i]))
211 add_task (win[i]);
212
213 XFree (win);
214 }
215
216
217 void resize_taskbar(void *obj)
218 {
219 Taskbar *taskbar = (Taskbar*)obj;
220 Panel *panel = (Panel*)taskbar->area.panel;
221 Task *tsk;
222 GSList *l;
223 int task_count;
224
225 //printf("resize_taskbar : posx et width des taches\n");
226
227 taskbar->area.redraw = 1;
228
229 if (panel_horizontal) {
230 int pixel_width, modulo_width=0;
231 int x, taskbar_width;
232
233 // new task width for 'desktop'
234 task_count = g_slist_length(taskbar->area.list);
235 if (!task_count) pixel_width = panel->g_task.maximum_width;
236 else {
237 taskbar_width = taskbar->area.width - (2 * panel->g_taskbar.pix.border.width) - (2 * panel->g_taskbar.paddingxlr);
238 if (task_count>1) taskbar_width -= ((task_count-1) * panel->g_taskbar.paddingx);
239
240 pixel_width = taskbar_width / task_count;
241 if (pixel_width > panel->g_task.maximum_width)
242 pixel_width = panel->g_task.maximum_width;
243 else
244 modulo_width = taskbar_width % task_count;
245 }
246
247 if ((taskbar->task_width == pixel_width) && (taskbar->task_modulo == modulo_width)) {
248 }
249 else {
250 taskbar->task_width = pixel_width;
251 taskbar->task_modulo = modulo_width;
252 taskbar->text_width = pixel_width - panel->g_task.text_posx - panel->g_task.area.pix.border.width - panel->g_task.area.paddingx;
253 }
254
255 // change pos_x and width for all tasks
256 x = taskbar->area.posx + taskbar->area.pix.border.width + taskbar->area.paddingxlr;
257 for (l = taskbar->area.list; l ; l = l->next) {
258 tsk = l->data;
259 tsk->area.posx = x;
260 tsk->area.width = pixel_width;
261 tsk->area.redraw = 1;
262 if (modulo_width) {
263 tsk->area.width++;
264 modulo_width--;
265 }
266
267 x += tsk->area.width + panel->g_taskbar.paddingx;
268 }
269 }
270 else {
271 int pixel_height, modulo_height=0;
272 int y, taskbar_height;
273
274 // new task width for 'desktop'
275 task_count = g_slist_length(taskbar->area.list);
276 if (!task_count) pixel_height = panel->g_task.maximum_height;
277 else {
278 taskbar_height = taskbar->area.height - (2 * panel->g_taskbar.pix.border.width) - (2 * panel->g_taskbar.paddingxlr);
279 if (task_count>1) taskbar_height -= ((task_count-1) * panel->g_taskbar.paddingx);
280
281 pixel_height = taskbar_height / task_count;
282 if (pixel_height > panel->g_task.maximum_height)
283 pixel_height = panel->g_task.maximum_height;
284 else
285 modulo_height = taskbar_height % task_count;
286 }
287
288 if ((taskbar->task_width == pixel_height) && (taskbar->task_modulo == modulo_height)) {
289 }
290 else {
291 taskbar->task_width = pixel_height;
292 taskbar->task_modulo = modulo_height;
293 taskbar->text_width = taskbar->area.width - (2 * panel->g_taskbar.paddingy) - panel->g_task.text_posx - panel->g_task.area.pix.border.width - panel->g_task.area.paddingx;
294 }
295
296 // change pos_y and height for all tasks
297 y = taskbar->area.posy + taskbar->area.pix.border.width + taskbar->area.paddingxlr;
298 for (l = taskbar->area.list; l ; l = l->next) {
299 tsk = l->data;
300 tsk->area.posy = y;
301 tsk->area.height = pixel_height;
302 tsk->area.redraw = 1;
303 if (modulo_height) {
304 tsk->area.height++;
305 modulo_height--;
306 }
307
308 y += tsk->area.height + panel->g_taskbar.paddingx;
309 }
310 }
311 }
312
313
314
This page took 0.044082 seconds and 5 git commands to generate.