dwm/tag.c

140 lines
2.5 KiB
C
Raw Normal View History

2006-07-15 16:30:50 +02:00
/*
* (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com>
* See LICENSE file for license details.
*/
#include "dwm.h"
2006-07-19 17:42:08 +02:00
#include <regex.h>
#include <stdio.h>
2006-08-04 14:40:32 +02:00
#include <stdlib.h>
2006-07-15 16:30:50 +02:00
#include <string.h>
2006-07-19 17:42:08 +02:00
#include <sys/types.h>
2006-07-15 16:30:50 +02:00
#include <X11/Xutil.h>
2006-07-19 17:42:08 +02:00
typedef struct {
2006-08-04 14:40:32 +02:00
const char *clpattern;
const char *tpattern;
2006-07-19 17:42:08 +02:00
Bool isfloat;
} Rule;
2006-08-04 14:40:32 +02:00
typedef struct {
regex_t *clregex;
regex_t *tregex;
} RReg;
/* static */
TAGS
RULES
2006-07-15 16:30:50 +02:00
2006-08-04 14:40:32 +02:00
static RReg *rreg = NULL;
static unsigned int len = 0;
2006-07-20 15:17:52 +02:00
/* extern */
Client *
getnext(Client *c) {
for(; c && !isvisible(c); c = c->next);
return c;
2006-07-15 16:30:50 +02:00
}
Client *
getprev(Client *c) {
for(; c && !isvisible(c); c = c->prev);
return c;
}
2006-08-04 14:40:32 +02:00
void
initrregs() {
2006-08-04 14:40:32 +02:00
unsigned int i;
regex_t *reg;
if(rreg)
return;
len = sizeof(rule) / sizeof(rule[0]);
rreg = emallocz(len * sizeof(RReg));
for(i = 0; i < len; i++) {
if(rule[i].clpattern) {
reg = emallocz(sizeof(regex_t));
if(regcomp(reg, rule[i].clpattern, 0))
free(reg);
else
rreg[i].clregex = reg;
}
if(rule[i].tpattern) {
reg = emallocz(sizeof(regex_t));
if(regcomp(reg, rule[i].tpattern, 0))
free(reg);
else
rreg[i].tregex = reg;
}
}
}
void
settags(Client *c, Client *trans) {
2006-08-23 12:08:37 +02:00
char prop[512];
2006-08-04 14:40:32 +02:00
unsigned int i, j;
2006-07-19 17:42:08 +02:00
regmatch_t tmp;
Bool matched = trans != NULL;
2006-07-19 17:42:08 +02:00
XClassHint ch;
if(matched) {
for(i = 0; i < ntags; i++)
c->tags[i] = trans->tags[i];
}
else if(XGetClassHint(dpy, c->win, &ch)) {
2006-08-23 12:08:37 +02:00
snprintf(prop, sizeof(prop), "%s:%s:%s",
2006-07-19 17:42:08 +02:00
ch.res_class ? ch.res_class : "",
2006-08-23 12:08:37 +02:00
ch.res_name ? ch.res_name : "", c->name);
2006-08-04 14:40:32 +02:00
for(i = 0; !matched && i < len; i++)
2006-08-23 12:08:37 +02:00
if(rreg[i].clregex && !regexec(rreg[i].clregex, prop, 1, &tmp, 0)) {
2006-08-04 14:40:32 +02:00
c->isfloat = rule[i].isfloat;
for(j = 0; rreg[i].tregex && j < ntags; j++) {
if(!regexec(rreg[i].tregex, tags[j], 1, &tmp, 0)) {
matched = True;
c->tags[j] = True;
}
}
2006-07-19 17:42:08 +02:00
}
if(ch.res_class)
XFree(ch.res_class);
if(ch.res_name)
XFree(ch.res_name);
}
if(!matched)
for(i = 0; i < ntags; i++)
c->tags[i] = seltag[i];
for(c->weight = 0; c->weight < ntags && !c->tags[c->weight]; c->weight++);
}
2006-08-14 16:59:18 +02:00
void
tag(Arg *arg) {
2006-08-14 16:59:18 +02:00
unsigned int i;
if(!sel)
return;
for(i = 0; i < ntags; i++)
sel->tags[i] = False;
sel->tags[arg->i] = True;
sel->weight = arg->i;
arrange(NULL);
2006-08-14 16:59:18 +02:00
}
void
toggletag(Arg *arg) {
unsigned int i;
2006-08-14 16:59:18 +02:00
if(!sel)
return;
sel->tags[arg->i] = !sel->tags[arg->i];
for(i = 0; i < ntags && !sel->tags[i]; i++);
if(i == ntags)
sel->tags[arg->i] = True;
sel->weight = (i == ntags) ? arg->i : i;
arrange(NULL);
}