4f915648351a9c9ceb1adfeb847790de0b9fc74f
[dana/urxvt.git] / src / rxvttoolkit.h
1 /*----------------------------------------------------------------------*
2  * File:        rxvttoolkit.h - provide toolkit-functionality for rxvt.
3  *----------------------------------------------------------------------*
4  *
5  * All portions of code are copyright by their respective author/s.
6  * Copyright (c) 2003-2006 Marc Lehmann <pcg@goof.com>
7  *
8  * This program is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation; either version 2 of the License, or
11  * (at your option) any later version.
12  *
13  * This program is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program; if not, write to the Free Software
20  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21  *----------------------------------------------------------------------*/
22
23 #ifndef RXVT_TOOLKIT_H
24 #define RXVT_TOOLKIT_H
25
26 #include <X11/Xlib.h>
27
28 #if XFT
29 # include <X11/Xft/Xft.h>
30 #endif
31
32 #include "ev_cpp.h"
33
34 #include "rxvtutil.h"
35
36 #include "callback.h"
37
38 // see rxvttoolkit.C:xa_names, which must be kept in sync
39 enum {
40   XA_TEXT,
41   XA_COMPOUND_TEXT,
42   XA_UTF8_STRING,
43   XA_MULTIPLE,
44   XA_TARGETS,
45   XA_TIMESTAMP,
46   XA_VT_SELECTION,
47   XA_INCR,
48   XA_WM_PROTOCOLS,
49   XA_WM_DELETE_WINDOW,
50   XA_CLIPBOARD,
51   XA_AVERAGE_WIDTH,
52   XA_WEIGHT_NAME,
53   XA_SLANT,
54   XA_CHARSET_REGISTRY,
55   XA_CHARSET_ENCODING,
56 #if ENABLE_FRILLS
57   XA_MOTIF_WM_HINTS,
58 #endif
59 #if ENABLE_EWMH
60   XA_NET_WM_PID,
61   XA_NET_WM_NAME,
62   XA_NET_WM_ICON_NAME,
63   XA_NET_WM_PING,
64   XA_NET_WM_ICON,
65 #endif
66 #if USE_XIM
67   XA_WM_LOCALE_NAME,
68   XA_XIM_SERVERS,
69 #endif
70 #if ENABLE_TRANSPARENCY
71   XA_XROOTPMAP_ID,
72   XA_ESETROOT_PMAP_ID,
73 #endif
74 #if ENABLE_XEMBED
75   XA_XEMBED,
76   XA_XEMBED_INFO,
77 #endif
78 #if !ENABLE_MINIMAL
79   // these are usually allocated by other subsystems, but we do it
80   // here to avoid a server roundtrip.
81   XA_SCREEN_RESOURCES,
82   XA_XDCCC_LINEAR_RGB_CORRECTION,
83   XA_XDCCC_LINEAR_RGB_MATRICES,
84   XA_WM_COLORMAP_WINDOWS,
85   XA_WM_STATE,
86   XA_cursor,
87 # if USE_XIM
88   // various selection targets used by XIM
89   XA_TRANSPORT,
90   XA_LOCALES,
91   XA__XIM_PROTOCOL,
92   XA__XIM_XCONNECT,
93   XA__XIM_MOREDATA,
94 # endif
95 #endif
96   NUM_XA
97 };
98
99 struct rxvt_term;
100 struct rxvt_display;
101
102 struct im_watcher;
103 struct xevent_watcher;
104
105 template<class watcher>
106 struct event_vec : vector<watcher *>
107 {
108   void erase_unordered (unsigned int pos)
109   {
110     watcher *w = (*this)[this->size () - 1];
111     this->pop_back ();
112
113     if (!this->empty ())
114       if (((*this)[pos] = w)) // '=' is correct!
115         w->active = pos + 1;
116   }
117 };
118
119 struct rxvt_watcher
120 {
121   int active; /* 0 == inactive, else index into respective vector */
122
123   bool is_active () { return active; }
124
125   rxvt_watcher () : active (0) { }
126 };
127
128 struct refcounted
129 {
130   int referenced;
131   char *id;
132
133   refcounted (const char *id);
134   bool ref_init () { return false; }
135   void ref_next () { }
136   ~refcounted ();
137 };
138
139 template<class T>
140 struct refcache : vector<T *>
141 {
142   T *get (const char *id);
143   void put (T *obj);
144   void clear ();
145
146   ~refcache ()
147   {
148     clear ();
149   }
150 };
151
152 /////////////////////////////////////////////////////////////////////////////
153
154 struct rxvt_screen;
155
156 struct rxvt_drawable
157 {
158   rxvt_screen *screen;
159   Drawable drawable;
160   operator Drawable() { return drawable; }
161
162 #if XFT
163   XftDraw *xftdrawable;
164   operator XftDraw *();
165 #endif
166
167   rxvt_drawable (rxvt_screen *screen, Drawable drawable)
168   : screen(screen),
169 #if XFT
170     xftdrawable(0),
171 #endif
172     drawable(drawable)
173   { }
174
175 #if XFT
176   ~rxvt_drawable ();
177 #endif
178 };
179
180 /////////////////////////////////////////////////////////////////////////////
181
182 #ifdef USE_XIM
183 struct rxvt_xim : refcounted
184 {
185   void destroy ();
186   rxvt_display *display;
187
188 //public
189   XIM xim;
190
191   rxvt_xim (const char *id) : refcounted (id) { }
192   bool ref_init ();
193   ~rxvt_xim ();
194 };
195 #endif
196
197 struct rxvt_screen
198 {
199   rxvt_display *display;
200   Display *dpy;
201   int depth;
202   Visual *visual;
203   Colormap cmap;
204
205 #if XFT
206   // scratch pixmap
207   rxvt_drawable *scratch_area;
208   int scratch_w, scratch_h;
209
210   rxvt_drawable &scratch_drawable (int w, int h);
211
212   rxvt_screen ();
213 #endif
214
215   void set (rxvt_display *disp);
216   void select_visual (int bitdepth);
217   void clear ();
218 };
219
220 struct rxvt_display : refcounted
221 {
222   event_vec<xevent_watcher> xw;
223
224   ev::prepare flush_ev; void flush_cb (ev::prepare &w, int revents);
225   ev::io      x_ev    ; void x_cb     (ev::io      &w, int revents);
226
227 #ifdef USE_XIM
228   refcache<rxvt_xim> xims;
229   vector<im_watcher *> imw;
230
231   void im_change_cb ();
232   void im_change_check ();
233 #endif
234
235 //public
236   Display   *dpy;
237   int       screen;
238   Window    root;
239   rxvt_term *selection_owner;
240   rxvt_term *clipboard_owner;
241   Atom      xa[NUM_XA];
242   bool      is_local;
243 #ifdef POINTER_BLANK
244   Cursor    blank_cursor;
245 #endif
246
247   rxvt_display (const char *id);
248   XrmDatabase get_resources (bool refresh);
249   bool ref_init ();
250   void ref_next ();
251   ~rxvt_display ();
252
253   void flush ()
254   {
255     flush_ev.start ();
256   }
257
258   Atom atom (const char *name);
259   void set_selection_owner (rxvt_term *owner, bool clipboard);
260
261   void reg (xevent_watcher *w);
262   void unreg (xevent_watcher *w);
263
264 #ifdef USE_XIM
265   void reg (im_watcher *w);
266   void unreg (im_watcher *w);
267
268   rxvt_xim *get_xim (const char *locale, const char *modifiers);
269   void put_xim (rxvt_xim *xim);
270 #endif
271 };
272
273 #ifdef USE_XIM
274 struct im_watcher : rxvt_watcher, callback<void (void)>
275 {
276   void start (rxvt_display *display)
277   {
278     display->reg (this);
279   }
280
281   void stop (rxvt_display *display)
282   {
283     display->unreg (this);
284   }
285 };
286 #endif
287
288 struct xevent_watcher : rxvt_watcher, callback<void (XEvent &)>
289 {
290   Window window;
291
292   void start (rxvt_display *display, Window window)
293   {
294     this->window = window;
295     display->reg (this);
296   }
297
298   void stop (rxvt_display *display)
299   {
300     display->unreg (this);
301   }
302 };
303
304 extern refcache<rxvt_display> displays;
305
306 /////////////////////////////////////////////////////////////////////////////
307
308 typedef unsigned long Pixel;
309
310 struct rgba
311 {
312   unsigned short r, g, b, a;
313
314   enum { MIN_CC = 0x0000, MAX_CC  = 0xffff };
315
316   rgba ()
317   { }
318
319   rgba (unsigned short r, unsigned short g, unsigned short b, unsigned short a = MAX_CC)
320   : r(r), g(g), b(b), a(a)
321   { }
322 };
323
324 struct rxvt_color
325 {
326 #if XFT
327   XftColor c;
328 #else
329   XColor c;
330 #endif
331
332   operator Pixel () const { return c.pixel; }
333
334   bool operator == (const rxvt_color &b) const { return Pixel (*this) == Pixel (b); }
335   bool operator != (const rxvt_color &b) const { return Pixel (*this) != Pixel (b); }
336
337   bool is_opaque () const
338   {
339 #if XFT
340     return c.color.alpha == rgba::MAX_CC;
341 #else
342     return 1;
343 #endif
344   }
345
346   bool alloc (rxvt_screen *screen, const rgba &color);
347   void free (rxvt_screen *screen);
348
349   void get (rgba &color);
350   void get (XColor &color);
351
352   bool set (rxvt_screen *screen, const char *name);
353   bool set (rxvt_screen *screen, const rgba &color);
354
355   void fade (rxvt_screen *screen, int percent, rxvt_color &result, const rgba &to = rgba (0, 0, 0));
356 };
357
358 #if TRACE_PIXMAPS
359 Pixmap trace_XCreatePixmap (const char *file, int line, Display *dpy, Window r, unsigned int w, unsigned int h, unsigned int d);
360 void trace_XFreePixmap (const char *file, int line, Display *dpy, Pixmap p);
361
362 # define XCreatePixmap(dpy,r,w,h,d) trace_XCreatePixmap (__FILE__,__LINE__,dpy,r,w,h,d)
363 # define XFreePixmap(dpy,p) trace_XFreePixmap (__FILE__,__LINE__,dpy,p)
364 #endif
365
366 #endif
367