Xinqi Bao's Git
   2  * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> 
   3  * See LICENSE file for license details. 
  12 #include <sys/select.h> 
  13 #include <X11/cursorfont.h> 
  14 #include <X11/keysym.h> 
  15 #include <X11/Xatom.h> 
  16 #include <X11/Xproto.h> 
  22 int screen
, sx
, sy
, sw
, sh
, bx
, by
, bw
, bh
, mw
; 
  23 unsigned int ntags
, numlockmask
; 
  24 Atom wmatom
[WMLast
], netatom
[NetLast
]; 
  27 Client 
*clients 
= NULL
; 
  29 Cursor cursor
[CurLast
]; 
  36 static int (*xerrorxlib
)(Display 
*, XErrorEvent 
*); 
  37 static Bool otherwm
, readin
; 
  44                 resize(sel
, True
, TopLeft
); 
  48                 XFreeFontSet(dpy
, dc
.font
.set
); 
  50                 XFreeFont(dpy
, dc
.font
.xfont
); 
  51         XUngrabKey(dpy
, AnyKey
, AnyModifier
, root
); 
  52         XFreePixmap(dpy
, dc
.drawable
); 
  54         XDestroyWindow(dpy
, barwin
); 
  55         XSetInputFocus(dpy
, PointerRoot
, RevertToPointerRoot
, CurrentTime
); 
  67         if(XQueryTree(dpy
, root
, &d1
, &d2
, &wins
, &num
)) { 
  68                 for(i 
= 0; i 
< num
; i
++) { 
  69                         if(!XGetWindowAttributes(dpy
, wins
[i
], &wa
)) 
  71                         if(wa
.override_redirect 
|| XGetTransientForHint(dpy
, wins
[i
], &d1
)) 
  73                         if(wa
.map_state 
== IsViewable
) 
  87         XModifierKeymap 
*modmap
; 
  88         XSetWindowAttributes wa
; 
  91         wmatom
[WMProtocols
] = XInternAtom(dpy
, "WM_PROTOCOLS", False
); 
  92         wmatom
[WMDelete
] = XInternAtom(dpy
, "WM_DELETE_WINDOW", False
); 
  93         netatom
[NetSupported
] = XInternAtom(dpy
, "_NET_SUPPORTED", False
); 
  94         netatom
[NetWMName
] = XInternAtom(dpy
, "_NET_WM_NAME", False
); 
  95         XChangeProperty(dpy
, root
, netatom
[NetSupported
], XA_ATOM
, 32, 
  96                         PropModeReplace
, (unsigned char *) netatom
, NetLast
); 
  99         cursor
[CurNormal
] = XCreateFontCursor(dpy
, XC_left_ptr
); 
 100         cursor
[CurResize
] = XCreateFontCursor(dpy
, XC_sizing
); 
 101         cursor
[CurMove
] = XCreateFontCursor(dpy
, XC_fleur
); 
 103         modmap 
= XGetModifierMapping(dpy
); 
 104         for (i 
= 0; i 
< 8; i
++) { 
 105                 for (j 
= 0; j 
< modmap
->max_keypermod
; j
++) { 
 106                         if(modmap
->modifiermap
[i 
* modmap
->max_keypermod 
+ j
] == XKeysymToKeycode(dpy
, XK_Num_Lock
)) 
 107                                 numlockmask 
= (1 << i
); 
 112         wa
.event_mask 
= SubstructureRedirectMask 
| SubstructureNotifyMask 
| EnterWindowMask 
| LeaveWindowMask
; 
 113         wa
.cursor 
= cursor
[CurNormal
]; 
 114         XChangeWindowAttributes(dpy
, root
, CWEventMask 
| CWCursor
, &wa
); 
 119         for(ntags 
= 0; tags
[ntags
]; ntags
++); 
 120         seltag 
= emallocz(sizeof(Bool
) * ntags
); 
 124         dc
.bg 
= getcolor(BGCOLOR
); 
 125         dc
.fg 
= getcolor(FGCOLOR
); 
 126         dc
.border 
= getcolor(BORDERCOLOR
); 
 130         sw 
= DisplayWidth(dpy
, screen
); 
 131         sh 
= DisplayHeight(dpy
, screen
); 
 132         mw 
= (sw 
* MASTERW
) / 100; 
 136         dc
.h 
= bh 
= dc
.font
.height 
+ 4; 
 137         wa
.override_redirect 
= 1; 
 138         wa
.background_pixmap 
= ParentRelative
; 
 139         wa
.event_mask 
= ButtonPressMask 
| ExposureMask
; 
 140         barwin 
= XCreateWindow(dpy
, root
, bx
, by
, bw
, bh
, 0, DefaultDepth(dpy
, screen
), 
 141                         CopyFromParent
, DefaultVisual(dpy
, screen
), 
 142                         CWOverrideRedirect 
| CWBackPixmap 
| CWEventMask
, &wa
); 
 143         XDefineCursor(dpy
, barwin
, cursor
[CurNormal
]); 
 144         XMapRaised(dpy
, barwin
); 
 146         dc
.drawable 
= XCreatePixmap(dpy
, root
, sw
, bh
, DefaultDepth(dpy
, screen
)); 
 147         dc
.gc 
= XCreateGC(dpy
, root
, 0, 0); 
 149         issel 
= XQueryPointer(dpy
, root
, &w
, &w
, &i
, &i
, &i
, &i
, &mask
); 
 151         strcpy(stext
, "dwm-"VERSION
); 
 155  * Startup Error handler to check if another window manager 
 156  * is already running. 
 159 xerrorstart(Display 
*dsply
, XErrorEvent 
*ee
) 
 170         int i
, format
, protos
, status
; 
 171         unsigned long extra
, res
; 
 172         Atom 
*protocols
, real
; 
 175         status 
= XGetWindowProperty(dpy
, w
, wmatom
[WMProtocols
], 0L, 20L, False
, 
 176                         XA_ATOM
, &real
, &format
, &res
, &extra
, (unsigned char **)&protocols
); 
 177         if(status 
!= Success 
|| protocols 
== 0) 
 179         for(i 
= 0; i 
< res
; i
++) 
 180                 if(protocols
[i
] == wmatom
[WMDelete
]) 
 181                         protos 
|= PROTODELWIN
; 
 187 sendevent(Window w
, Atom a
, long value
) 
 191         e
.type 
= ClientMessage
; 
 192         e
.xclient
.window 
= w
; 
 193         e
.xclient
.message_type 
= a
; 
 194         e
.xclient
.format 
= 32; 
 195         e
.xclient
.data
.l
[0] = value
; 
 196         e
.xclient
.data
.l
[1] = CurrentTime
; 
 197         XSendEvent(dpy
, w
, False
, NoEventMask
, &e
); 
 204         readin 
= running 
= False
; 
 208  * There's no way to check accesses to destroyed windows, thus those cases are 
 209  * ignored (especially on UnmapNotify's).  Other types of errors call Xlibs 
 210  * default error handler, which calls exit(). 
 213 xerror(Display 
*dpy
, XErrorEvent 
*ee
) 
 215         if(ee
->error_code 
== BadWindow
 
 216         || (ee
->request_code 
== X_SetInputFocus 
&& ee
->error_code 
== BadMatch
) 
 217         || (ee
->request_code 
== X_PolyText8 
&& ee
->error_code 
== BadDrawable
) 
 218         || (ee
->request_code 
== X_PolyFillRectangle 
&& ee
->error_code 
== BadDrawable
) 
 219         || (ee
->request_code 
== X_PolySegment 
&& ee
->error_code 
== BadDrawable
) 
 220         || (ee
->request_code 
== X_ConfigureWindow 
&& ee
->error_code 
== BadMatch
) 
 221         || (ee
->request_code 
== X_GrabKey 
&& ee
->error_code 
== BadAccess
)) 
 223         fprintf(stderr
, "dwm: fatal error: request code=%d, error code=%d\n", 
 224                 ee
->request_code
, ee
->error_code
); 
 225         return xerrorxlib(dpy
, ee
); /* may call exit() */ 
 229 main(int argc
, char *argv
[]) 
 234         if(argc 
== 2 && !strncmp("-v", argv
[1], 3)) { 
 235                 fputs("dwm-"VERSION
", (C)opyright MMVI Anselm R. Garbe\n", stdout
); 
 239                 eprint("usage: dwm [-v]\n"); 
 241         dpy 
= XOpenDisplay(0); 
 243                 eprint("dwm: cannot open display\n"); 
 245         xfd 
= ConnectionNumber(dpy
); 
 246         screen 
= DefaultScreen(dpy
); 
 247         root 
= RootWindow(dpy
, screen
); 
 250         XSetErrorHandler(xerrorstart
); 
 251         /* this causes an error if some other window manager is running */ 
 252         XSelectInput(dpy
, root
, SubstructureRedirectMask
); 
 256                 eprint("dwm: another window manager is already running\n"); 
 259         XSetErrorHandler(NULL
); 
 260         xerrorxlib 
= XSetErrorHandler(xerror
); 
 267         /* main event loop, also reads status text from stdin */ 
 274                         FD_SET(STDIN_FILENO
, &rd
); 
 276                 r 
= select(xfd 
+ 1, &rd
, NULL
, NULL
, NULL
); 
 277                 if((r 
== -1) && (errno 
== EINTR
)) 
 280                         if(readin 
&& FD_ISSET(STDIN_FILENO
, &rd
)) { 
 281                                 readin 
= NULL 
!= fgets(stext
, sizeof(stext
), stdin
); 
 283                                         stext
[strlen(stext
) - 1] = 0; 
 285                                         strcpy(stext
, "broken pipe"); 
 290                         eprint("select failed\n");