2
* Copyright (c) 1999 Alfredo K. Kojima
3
* Copyright (c) 2001, 2002 Seiichi SATO <ssato@sh.rim.or.jp>
5
* Permission is hereby granted, free of charge, to any person obtaining a copy
6
* of this software and associated documentation files (the "Software"), to deal
7
* in the Software without restriction, including without limitation the rights
8
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
* copies of the Software, and to permit persons to whom the Software is
10
* furnished to do so, subject to the following conditions:
12
* The above copyright notice and this permission notice shall be included in
13
* all copies or substantial portions of the Software.
15
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
* AUTHOR BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
19
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
20
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22
* This code is based on libdockapp-0.4.0
23
* modified by Seiichi SATO <ssato@sh.rim.or.jp>
28
#define WINDOWED_SIZE_W 64
29
#define WINDOWED_SIZE_H 64
32
Display *display = NULL;
33
Bool dockapp_iswindowed = False;
34
Bool dockapp_isbrokenwm = False;
37
static Window window = None;
38
static Window icon_window = None;
41
static Atom delete_win;
42
static int width, height;
43
static int offset_w, offset_h;
46
dockapp_open_window(char *display_specified, char *appname,
47
unsigned w, unsigned h, int argc, char **argv)
49
XClassHint *classhint;
57
/* Open Connection to X Server */
58
display = XOpenDisplay(display_specified);
60
fprintf(stderr, "%s: could not open display %s!\n", argv[0],
61
XDisplayName(display_specified));
64
root = DefaultRootWindow(display);
69
if (dockapp_iswindowed) {
70
offset_w = (WINDOWED_SIZE_W - w) / 2;
71
offset_h = (WINDOWED_SIZE_H - h) / 2;
75
offset_w = offset_h = 0;
81
icon_window = XCreateSimpleWindow(display, root, 0, 0, ww, wh, 0, 0, 0);
82
if (dockapp_isbrokenwm) {
83
window = XCreateSimpleWindow(display, root, 0, 0, ww, wh, 0, 0, 0);
85
window = XCreateSimpleWindow(display, root, 0, 0, 1, 1, 0, 0, 0);
89
classhint = XAllocClassHint();
90
if (classhint == NULL) {
91
fprintf(stderr, "%s: can't allocate memory for wm hints!\n", argv[0]);
94
classhint->res_class = "DockApp";
95
classhint->res_name = appname;
96
XSetClassHint(display, window, classhint);
100
wmhints = XAllocWMHints();
101
if (wmhints == NULL) {
102
fprintf(stderr, "%s: can't allocate memory for wm hints!\n", argv[0]);
105
wmhints->flags = IconWindowHint | WindowGroupHint;
106
if (!dockapp_iswindowed) {
107
wmhints->flags |= StateHint;
108
wmhints->initial_state = WithdrawnState;
110
wmhints->window_group = window;
111
wmhints->icon_window = icon_window;
112
XSetWMHints(display, window, wmhints);
115
/* Set WM Protocols */
116
delete_win = XInternAtom(display, "WM_DELETE_WINDOW", False);
117
XSetWMProtocols (display, icon_window, &delete_win, 1);
120
sizehints.flags = USSize;
121
if (!dockapp_iswindowed) {
122
sizehints.flags |= USPosition;
123
sizehints.x = sizehints.y = 0;
125
sizehints.flags |= PMinSize | PMaxSize;
126
sizehints.min_width = sizehints.max_width = WINDOWED_SIZE_W;
127
sizehints.min_height = sizehints.max_height = WINDOWED_SIZE_H;
129
sizehints.width = ww;
130
sizehints.height = wh;
131
XSetWMNormalHints(display, icon_window, &sizehints);
133
/* Set WindowTitle for AfterStep Wharf */
134
stat = XStringListToTextProperty(&appname, 1, &title);
135
XSetWMName(display, window, &title);
136
XSetWMName(display, icon_window, &title);
138
/* Set Command to start the app so it can be docked properly */
139
XSetCommand(display, window, argv, argc);
141
depth = DefaultDepth(display, DefaultScreen(display));
142
gc = DefaultGC(display, DefaultScreen(display));
149
dockapp_set_eventmask(long mask)
151
XSelectInput(display, icon_window, mask);
152
XSelectInput(display, window, mask);
157
create_bg_pixmap(void)
161
bg = XCreatePixmap(display, icon_window, WINDOWED_SIZE_W, WINDOWED_SIZE_H,
163
XSetForeground(display, gc, dockapp_getcolor("rgb:ae/aa/ae"));
164
XFillRectangle(display, bg, gc, 0, 0, WINDOWED_SIZE_W, WINDOWED_SIZE_H);
165
XSetForeground(display, gc, dockapp_getcolor("rgb:ff/ff/ff"));
166
XDrawLine(display, bg, gc, 0, 0, 0, 63);
167
XDrawLine(display, bg, gc, 1, 0, 1, 62);
168
XDrawLine(display, bg, gc, 2, 0, 63, 0);
169
XDrawLine(display, bg, gc, 2, 1, 62, 1);
170
XSetForeground(display, gc, dockapp_getcolor("rgb:52/55/52"));
171
XDrawLine(display, bg, gc, 1, 63, 63, 63);
172
XDrawLine(display, bg, gc, 2, 62, 63, 62);
173
XDrawLine(display, bg, gc, 63, 1, 63, 61);
174
XDrawLine(display, bg, gc, 62, 2, 62, 61);
181
dockapp_set_background(Pixmap pixmap)
183
if (dockapp_iswindowed) {
185
bg = create_bg_pixmap();
186
XCopyArea(display, pixmap, bg, gc, 0, 0, width, height,
188
XSetWindowBackgroundPixmap(display, icon_window, bg);
189
XSetWindowBackgroundPixmap(display, window, bg);
190
XFreePixmap(display, bg);
192
XSetWindowBackgroundPixmap(display, icon_window, pixmap);
193
XSetWindowBackgroundPixmap(display, window, pixmap);
195
XClearWindow(display, icon_window);
203
if (!dockapp_iswindowed)
204
XMapRaised(display, window);
206
XMapRaised(display, icon_window);
213
dockapp_xpm2pixmap(char **data, Pixmap *pixmap, Pixmap *mask,
214
XpmColorSymbol * colorSymbol, unsigned int nsymbols)
216
XpmAttributes xpmAttr;
217
xpmAttr.valuemask = XpmCloseness;
218
xpmAttr.closeness = 40000;
221
xpmAttr.colorsymbols = colorSymbol;
222
xpmAttr.numsymbols = nsymbols;
223
xpmAttr.valuemask |= XpmColorSymbols;
226
if (XpmCreatePixmapFromData(display, icon_window, data, pixmap, mask, &xpmAttr) != 0)
234
dockapp_XCreatePixmap(int w, int h)
236
return (XCreatePixmap(display, icon_window, w, h, depth));
241
dockapp_setshape(Pixmap mask, int x_ofs, int y_ofs)
243
XShapeCombineMask(display, icon_window, ShapeBounding, -x_ofs, -y_ofs,
245
XShapeCombineMask(display, window, ShapeBounding, -x_ofs, -y_ofs,
252
dockapp_copyarea(Pixmap src, Pixmap dist, int x_src, int y_src, int w, int h,
253
int x_dist, int y_dist)
255
XCopyArea(display, src, dist, gc, x_src, y_src, w, h, x_dist, y_dist);
260
dockapp_copy2window (Pixmap src)
262
if (dockapp_isbrokenwm) {
263
XCopyArea(display, src, window, gc, 0, 0, width, height, offset_w,
266
XCopyArea(display, src, icon_window, gc, 0, 0, width, height, offset_w,
273
dockapp_nextevent_or_timeout(XEvent *event, unsigned long miliseconds)
275
struct timeval timeout;
278
XSync(display, False);
279
if (XPending(display)) {
280
XNextEvent(display, event);
284
timeout.tv_sec = miliseconds / 1000;
285
timeout.tv_usec = (miliseconds % 1000) * 1000;
288
FD_SET(ConnectionNumber(display), &rset);
289
if (select(ConnectionNumber(display)+1, &rset, NULL, NULL, &timeout) > 0) {
290
XNextEvent(display, event);
291
if (event->type == ClientMessage) {
292
if (event->xclient.data.l[0] == delete_win) {
293
XDestroyWindow(display,event->xclient.window);
294
XCloseDisplay(display);
298
if (dockapp_iswindowed) {
299
event->xbutton.x -= offset_w;
300
event->xbutton.y -= offset_h;
310
dockapp_getcolor(char *color_name)
314
if (!XParseColor(display, DefaultColormap(display, DefaultScreen(display)),
316
fprintf(stderr, "can't parse color %s\n", color_name), exit(1);
318
if (!XAllocColor(display, DefaultColormap(display, DefaultScreen(display)),
320
fprintf(stderr, "can't allocate color %s. Using black\n", color_name);
321
return BlackPixel(display, DefaultScreen(display));
329
dockapp_blendedcolor(char *color_name, int r, int g, int b, float fac)
333
if ((r < -255 || r > 255)||(g < -255 || g > 255)||(b < -255 || b > 255)){
334
fprintf(stderr, "r:%d,g:%d,b:%d (r,g,b must be 0 to 255)", r, g, b);
342
if (!XParseColor(display, DefaultColormap(display, DefaultScreen(display)),
344
fprintf(stderr, "can't parse color %s\n", color_name), exit(1);
346
if (!XAllocColor(display, DefaultColormap(display, DefaultScreen(display)),
348
fprintf(stderr, "can't allocate color %s. Using black\n", color_name);
349
return BlackPixel(display, DefaultScreen(display));
352
if (DefaultDepth(display, DefaultScreen(display)) < 16)
356
if (color.red + r > 0xffff) {
358
} else if (color.red + r < 0) {
361
color.red = (unsigned short)(fac * color.red + r);
365
if (color.green + g > 0xffff) {
366
color.green = 0xffff;
367
} else if (color.green + g < 0) {
370
color.green = (unsigned short)(fac * color.green + g);
374
if (color.blue + b > 0xffff) {
376
} else if (color.blue + b < 0) {
379
color.blue = (unsigned short)(fac * color.blue + b);
382
color.flags = DoRed | DoGreen | DoBlue;
384
if (!XAllocColor(display, DefaultColormap(display, DefaultScreen(display)),
386
fprintf(stderr, "can't allocate color %s. Using black\n", color_name);
387
return BlackPixel(display, DefaultScreen(display));