fix rearrange
[awesome.git] / tag.c
blobc5dde1ede116058b874a6836034a85f2c00dd0f9
1 /*
2 * tag.c - tag management
4 * Copyright © 2007 Julien Danjou <julien@danjou.info>
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
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.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 #include <stdio.h>
23 #include <X11/Xutil.h>
25 #include "screen.h"
26 #include "tag.h"
27 #include "util.h"
28 #include "rules.h"
29 #include "client.h"
30 #include "ewmh.h"
32 extern AwesomeConf globalconf;
34 static void
35 detach_tagclientlink(TagClientLink *tc)
37 TagClientLink *tmp;
39 if(globalconf.tclink == tc)
40 globalconf.tclink = tc->next;
41 else
43 for(tmp = globalconf.tclink; tmp && tmp->next != tc; tmp = tmp->next);
44 tmp->next = tc->next;
47 p_delete(&tc);
50 void
51 tag_client(Client *c, Tag *t)
53 TagClientLink *tc, *new_tc;
55 /* don't tag twice */
56 if(is_client_tagged(c, t))
57 return;
59 new_tc = p_new(TagClientLink, 1);
61 if(!globalconf.tclink)
62 globalconf.tclink = new_tc;
63 else
65 for(tc = globalconf.tclink; tc->next; tc = tc->next);
66 tc->next = new_tc;
69 new_tc->client = c;
70 new_tc->tag = t;
73 void
74 untag_client(Client *c, Tag *t)
76 TagClientLink *tc;
78 for(tc = globalconf.tclink; tc; tc = tc->next)
79 if(tc->client == c && tc->tag == t)
81 detach_tagclientlink(tc);
82 break;
86 Bool
87 is_client_tagged(Client *c, Tag *t)
89 TagClientLink *tc;
91 if(!c)
92 return False;
94 for(tc = globalconf.tclink; tc; tc = tc->next)
95 if(tc->client == c && tc->tag == t)
96 return True;
98 return False;
101 void
102 tag_client_with_current_selected(Client *c)
104 Tag *tag;
105 VirtScreen vscreen = globalconf.screens[c->screen];
107 for(tag = vscreen.tags; tag; tag = tag->next)
108 if(tag->selected)
109 tag_client(c, tag);
110 else
111 untag_client(c, tag);
114 void
115 tag_client_with_rules(Client *c)
117 Rule *r;
118 Tag *tag;
119 Bool matched = False;
121 for(r = globalconf.rules; r; r = r->next)
122 if(client_match_rule(c, r))
124 switch(r->isfloating)
126 case Tile:
127 c->isfloating = False;
128 break;
129 case Float:
130 c->isfloating = True;
131 break;
132 default:
133 break;
136 if(r->screen != RULE_NOSCREEN && r->screen != c->screen)
137 move_client_to_screen(c, r->screen, True);
139 for(tag = globalconf.screens[c->screen].tags; tag; tag = tag->next)
140 if(is_tag_match_rules(tag, r))
142 matched = True;
143 tag_client(c, tag);
145 else
146 untag_client(c, tag);
148 if(!matched)
149 tag_client_with_current_selected(c);
150 break;
155 Tag **
156 get_current_tags(int screen)
158 Tag *tag, **tags = NULL;
159 int n = 1;
161 tags = p_new(Tag *, n);
162 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
163 if(tag->selected)
165 p_realloc(&tags, ++n);
166 tags[n - 2] = tag;
169 /* finish with null */
170 tags[n - 1] = NULL;
172 return tags;
175 /** Tag selected window with tag
176 * \param screen Screen ID
177 * \param arg Tag name
178 * \ingroup ui_callback
180 void
181 uicb_client_tag(int screen, char *arg)
183 int tag_id = -1;
184 Tag *tag, *target_tag;
185 Client *sel = globalconf.focus->client;
187 if(!sel)
188 return;
190 if(arg)
192 tag_id = atoi(arg) - 1;
193 if(tag_id != -1)
195 for(target_tag = globalconf.screens[screen].tags; target_tag && tag_id > 0;
196 target_tag = target_tag->next, tag_id--);
197 if(target_tag)
199 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
200 untag_client(sel, tag);
201 tag_client(sel, target_tag);
205 else
206 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
207 tag_client(sel, tag);
209 client_saveprops(sel);
210 arrange(screen);
213 /** Toggle floating state of a client
214 * \param screen Screen ID
215 * \param arg unused
216 * \ingroup ui_callback
218 void
219 uicb_client_togglefloating(int screen, char *arg __attribute__((unused)))
221 Client *sel = globalconf.focus->client;
223 if(!sel)
224 return;
226 if((sel->isfloating = !sel->isfloating))
227 client_resize(sel, sel->f_geometry, True);
229 client_saveprops(sel);
230 arrange(screen);
233 /** Toggle a tag on client
234 * \param screen Screen ID
235 * \param arg Tag name
236 * \ingroup ui_callback
238 void
239 uicb_client_toggletag(int screen, char *arg)
241 Client *sel = globalconf.focus->client;
242 int i;
243 Tag *tag, *target_tag;
245 if(!sel)
246 return;
248 if(arg)
250 i = atoi(arg) - 1;
251 for(target_tag = globalconf.screens[screen].tags; target_tag && i > 0;
252 target_tag = target_tag->next, i--);
253 if(target_tag)
255 if(is_client_tagged(sel, target_tag))
256 untag_client(sel, target_tag);
257 else
258 tag_client(sel, target_tag);
261 /* check that there's at least one tag selected for this client*/
262 for(tag = globalconf.screens[screen].tags; tag
263 && !is_client_tagged(sel, tag); tag = tag->next)
265 if(!tag)
266 tag_client(sel, target_tag);
268 else
269 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
270 if(is_client_tagged(sel, tag))
271 tag_client(sel, tag);
272 else
273 untag_client(sel, tag);
275 client_saveprops(sel);
276 arrange(screen);
279 /** Add a tag to viewed tags
280 * \param screen Screen ID
281 * \param arg Tag name
282 * \ingroup ui_callback
284 void
285 uicb_tag_toggleview(int screen, char *arg)
287 int i;
288 Tag *tag, *target_tag;
290 if(arg)
292 i = atoi(arg) - 1;
293 for(target_tag = globalconf.screens[screen].tags; target_tag && i > 0;
294 target_tag = target_tag->next, i--);
296 if(target_tag)
297 target_tag->selected = !target_tag->selected;
299 /* check that there's at least one tag selected */
300 for(tag = globalconf.screens[screen].tags; tag && !tag->selected; tag = tag->next);
301 if(!tag)
302 target_tag->selected = True;
304 else
305 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
306 tag->selected = !tag->selected;
308 saveawesomeprops(screen);
309 arrange(screen);
310 ewmh_update_net_current_desktop(get_phys_screen(screen));
313 void
314 tag_view(int screen, int dindex)
316 Tag *target_tag, *tag;
318 if(dindex < 0)
319 return;
321 for(target_tag = globalconf.screens[screen].tags; target_tag && dindex > 0;
322 target_tag = target_tag->next, dindex--);
323 if(target_tag)
325 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
326 tag->selected = False;
327 target_tag->selected = True;
329 saveawesomeprops(screen);
330 arrange(screen);
331 ewmh_update_net_current_desktop(get_phys_screen(screen));
334 /** View tag
335 * \param screen Screen ID
336 * \param arg tag to view
337 * \ingroup ui_callback
339 void
340 uicb_tag_view(int screen, char *arg)
342 Tag *tag;
344 if(arg)
345 tag_view(screen, atoi(arg) - 1);
346 else
348 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
349 tag->selected = True;
350 saveawesomeprops(screen);
351 arrange(screen);
352 ewmh_update_net_current_desktop(get_phys_screen(screen));
356 /** View previously selected tags
357 * \param screen Screen ID
358 * \param arg unused
359 * \ingroup ui_callback
361 void
362 uicb_tag_prev_selected(int screen, char *arg __attribute__ ((unused)))
364 Tag *tag;
365 Bool t;
367 for(tag = globalconf.screens[screen].tags; tag; tag = tag->next)
369 t = tag->selected;
370 tag->selected = tag->was_selected;
371 tag->was_selected = t;
373 arrange(screen);
374 ewmh_update_net_current_desktop(get_phys_screen(screen));
377 /** View next tag
378 * \param screen Screen ID
379 * \param arg unused
380 * \ingroup ui_callback
382 void
383 uicb_tag_viewnext(int screen, char *arg __attribute__ ((unused)))
385 Tag **curtags = get_current_tags(screen);
387 if(!curtags[0]->next)
388 return;
390 curtags[0]->selected = False;
391 curtags[0]->next->selected = True;
393 p_delete(&curtags);
395 saveawesomeprops(screen);
396 arrange(screen);
397 ewmh_update_net_current_desktop(get_phys_screen(screen));
400 /** View previous tag
401 * \param screen Screen ID
402 * \param arg unused
403 * \ingroup ui_callback
405 void
406 uicb_tag_viewprev(int screen, char *arg __attribute__ ((unused)))
408 Tag *tag, **curtags = get_current_tags(screen);
410 for(tag = globalconf.screens[screen].tags; tag && tag->next != curtags[0]; tag = tag->next);
411 if(tag)
413 tag->selected = True;
414 curtags[0]->selected = False;
415 saveawesomeprops(screen);
416 arrange(screen);
418 p_delete(&curtags);
419 ewmh_update_net_current_desktop(get_phys_screen(screen));
422 void
423 uicb_tag_create(int screen, char *arg)
425 Tag *last_tag, *tag;
427 if(!a_strlen(arg))
428 return;
430 for(last_tag = globalconf.screens[screen].tags; last_tag && last_tag->next; last_tag = last_tag->next);
431 last_tag->next = tag = p_new(Tag, 1);
432 tag->name = a_strdup(arg);
433 tag->layout = globalconf.screens[screen].layouts;
434 tag->mwfact = 0.5;
435 tag->nmaster = 1;
436 tag->ncol = 1;
439 // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80