2 * Window Maker window manager
4 * Copyright (c) 1997-2004 Alfredo K. Kojima
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.
28 #include "WindowMaker.h"
34 #include "switchpanel.h"
40 #include <X11/extensions/shape.h>
67 /* these values will be updated whenever the switch panel
68 * is created to to match the size defined in switch_panel_icon_size
69 * and the selected font size */
70 static short int icon_size
;
71 static short int border_space
;
72 static short int icon_tile_size
;
73 static short int label_height
;
75 #define SCREEN_BORDER_SPACING 2*20
77 #define ICON_SELECTED (1<<1)
78 #define ICON_DIM (1<<2)
80 static int canReceiveFocus(WWindow
*wwin
)
82 if (wwin
->frame
->workspace
!= wwin
->screen_ptr
->current_workspace
)
85 if (wPreferences
.cycle_active_head_only
&&
86 wGetHeadForWindow(wwin
) != wGetHeadForPointerLocation(wwin
->screen_ptr
))
89 if (WFLAGP(wwin
, no_focusable
))
92 if (!wwin
->flags
.mapped
) {
93 if (!wwin
->flags
.shaded
&& !wwin
->flags
.miniaturized
&& !wwin
->flags
.hidden
)
102 static Bool
sameWindowClass(WWindow
*wwin
, WWindow
*curwin
)
104 if (!wwin
->wm_class
|| !curwin
->wm_class
)
106 if (strcmp(wwin
->wm_class
, curwin
->wm_class
))
112 static void changeImage(WSwitchPanel
*panel
, int idecks
, int selected
, Bool dim
, Bool force
)
114 WMFrame
*icon
= NULL
;
115 RImage
*image
= NULL
;
119 /* This whole function is a no-op if we aren't drawing the panel */
120 if (!wPreferences
.swtileImage
)
123 icon
= WMGetFromArray(panel
->icons
, idecks
);
124 image
= WMGetFromArray(panel
->images
, idecks
);
125 flags
= (int) (uintptr_t) WMGetFromArray(panel
->flags
, idecks
);
128 desired
|= ICON_SELECTED
;
132 if (flags
== desired
&& !force
)
135 WMReplaceInArray(panel
->flags
, idecks
, (void *) (uintptr_t) desired
);
137 if (!panel
->bg
&& !panel
->tile
&& !selected
)
138 WMSetFrameRelief(icon
, WRFlat
);
142 int opaq
= (dim
) ? 75 : 255;
147 if (canReceiveFocus(WMGetFromArray(panel
->windows
, idecks
)) < 0)
150 pos
= WMGetViewPosition(WMWidgetView(icon
));
151 back
= panel
->tileTmp
;
153 RCopyArea(back
, panel
->bg
,
154 border_space
+ pos
.x
- panel
->firstVisible
* icon_tile_size
,
155 border_space
+ pos
.y
, back
->width
, back
->height
, 0, 0);
158 color
.red
= WMRedComponentOfColor(panel
->gray
) >> 8;
159 color
.green
= WMGreenComponentOfColor(panel
->gray
) >> 8;
160 color
.blue
= WMBlueComponentOfColor(panel
->gray
) >> 8;
161 RFillImage(back
, &color
);
166 RCombineArea(back
, tile
, 0, 0, tile
->width
, tile
->height
,
167 (back
->width
- tile
->width
) / 2, (back
->height
- tile
->height
) / 2);
170 RCombineAreaWithOpaqueness(back
, image
, 0, 0, image
->width
, image
->height
,
171 (back
->width
- image
->width
) / 2, (back
->height
- image
->height
) / 2,
174 RConvertImage(panel
->scr
->rcontext
, back
, &p
);
175 XSetWindowBackgroundPixmap(dpy
, WMWidgetXID(icon
), p
);
176 XClearWindow(dpy
, WMWidgetXID(icon
));
180 if (!panel
->bg
&& !panel
->tile
&& selected
)
181 WMSetFrameRelief(icon
, WRSimple
);
184 static void addIconForWindow(WSwitchPanel
*panel
, WMWidget
*parent
, WWindow
*wwin
, int x
, int y
)
186 WMFrame
*icon
= WMCreateFrame(parent
);
187 RImage
*image
= NULL
;
189 WMSetFrameRelief(icon
, WRFlat
);
190 WMResizeWidget(icon
, icon_tile_size
, icon_tile_size
);
191 WMMoveWidget(icon
, x
, y
);
193 if (!WFLAGP(wwin
, always_user_icon
) && wwin
->net_icon_image
)
194 image
= RRetainImage(wwin
->net_icon_image
);
196 /* get_icon_image() includes the default icon image */
198 image
= get_icon_image(panel
->scr
, wwin
->wm_instance
, wwin
->wm_class
, icon_tile_size
);
200 /* We must resize the icon size (~64) to the switch panel icon size (~48) */
201 image
= wIconValidateIconSize(image
, icon_size
);
203 WMAddToArray(panel
->images
, image
);
204 WMAddToArray(panel
->icons
, icon
);
207 static void scrollIcons(WSwitchPanel
*panel
, int delta
)
209 int nfirst
= panel
->firstVisible
+ delta
;
211 int count
= WMGetArrayItemCount(panel
->windows
);
214 if (count
<= panel
->visibleCount
)
219 else if (nfirst
>= count
- panel
->visibleCount
)
220 nfirst
= count
- panel
->visibleCount
;
222 if (nfirst
== panel
->firstVisible
)
225 WMMoveWidget(panel
->iconBox
, -nfirst
* icon_tile_size
, 0);
227 panel
->firstVisible
= nfirst
;
229 for (i
= panel
->firstVisible
; i
< panel
->firstVisible
+ panel
->visibleCount
; i
++) {
230 if (i
== panel
->current
)
232 dim
= ((int) (uintptr_t) WMGetFromArray(panel
->flags
, i
) & ICON_DIM
);
233 changeImage(panel
, i
, 0, dim
, True
);
242 static RImage
*assemblePuzzleImage(RImage
**images
, int width
, int height
)
249 tw
= width
- images
[0]->width
- images
[2]->width
;
250 th
= height
- images
[0]->height
- images
[6]->height
;
252 if (tw
<= 0 || th
<= 0)
255 img
= RCreateImage(width
, height
, 1);
263 RFillImage(img
, &color
);
266 tmp
= RSmoothScaleImage(images
[1], tw
, images
[1]->height
);
267 RCopyArea(img
, tmp
, 0, 0, tmp
->width
, tmp
->height
, images
[0]->width
, 0);
271 tmp
= RSmoothScaleImage(images
[7], tw
, images
[7]->height
);
272 RCopyArea(img
, tmp
, 0, 0, tmp
->width
, tmp
->height
, images
[6]->width
, height
- images
[6]->height
);
276 tmp
= RSmoothScaleImage(images
[3], images
[3]->width
, th
);
277 RCopyArea(img
, tmp
, 0, 0, tmp
->width
, tmp
->height
, 0, images
[0]->height
);
281 tmp
= RSmoothScaleImage(images
[5], images
[5]->width
, th
);
282 RCopyArea(img
, tmp
, 0, 0, tmp
->width
, tmp
->height
, width
- images
[5]->width
, images
[2]->height
);
286 tmp
= RSmoothScaleImage(images
[4], tw
, th
);
287 RCopyArea(img
, tmp
, 0, 0, tmp
->width
, tmp
->height
, images
[0]->width
, images
[0]->height
);
291 RCopyArea(img
, images
[0], 0, 0, images
[0]->width
, images
[0]->height
, 0, 0);
292 RCopyArea(img
, images
[2], 0, 0, images
[2]->width
, images
[2]->height
, width
- images
[2]->width
, 0);
293 RCopyArea(img
, images
[6], 0, 0, images
[6]->width
, images
[6]->height
, 0, height
- images
[6]->height
);
294 RCopyArea(img
, images
[8], 0, 0, images
[8]->width
, images
[8]->height
,
295 width
- images
[8]->width
, height
- images
[8]->height
);
300 static RImage
*createBackImage(int width
, int height
)
302 return assemblePuzzleImage(wPreferences
.swbackImage
, width
, height
);
305 static RImage
*getTile(void)
309 if (!wPreferences
.swtileImage
)
312 stile
= RScaleImage(wPreferences
.swtileImage
, icon_tile_size
, icon_tile_size
);
314 return wPreferences
.swtileImage
;
319 static void drawTitle(WSwitchPanel
*panel
, int idecks
, const char *title
)
322 int width
= WMWidgetWidth(panel
->win
);
326 ntitle
= ShrinkString(panel
->font
, title
, width
- 2 * border_space
);
332 if (strcmp(ntitle
, title
) != 0) {
335 int w
= WMWidthOfString(panel
->font
, ntitle
, strlen(ntitle
));
337 x
= border_space
+ (idecks
- panel
->firstVisible
) * icon_tile_size
+
338 icon_tile_size
/ 2 - w
/ 2;
339 if (x
< border_space
)
341 else if (x
+ w
> width
- border_space
)
342 x
= width
- border_space
- w
;
346 XClearWindow(dpy
, WMWidgetXID(panel
->win
));
348 WMDrawString(panel
->scr
->wmscreen
,
349 WMWidgetXID(panel
->win
),
350 panel
->white
, panel
->font
,
352 WMWidgetHeight(panel
->win
) - border_space
- label_height
+
353 WMFontHeight(panel
->font
) / 2, ntitle
, strlen(ntitle
));
356 WMSetLabelText(panel
->label
, ntitle
);
363 static WMArray
*makeWindowListArray(WScreen
*scr
, int include_unmapped
, Bool class_only
)
365 WMArray
*windows
= WMCreateArray(10);
366 WWindow
*wwin
= scr
->focused_window
;
369 if ((canReceiveFocus(wwin
) != 0) &&
370 (wwin
->flags
.mapped
|| wwin
->flags
.shaded
|| include_unmapped
)) {
372 if (!sameWindowClass(scr
->focused_window
, wwin
)) {
376 if (!WFLAGP(wwin
, skip_switchpanel
))
377 WMAddToArray(windows
, wwin
);
384 static WMArray
*makeWindowFlagsArray(int count
)
386 WMArray
*flags
= WMCreateArray(count
);
389 for (i
= 0; i
< count
; i
++)
390 WMAddToArray(flags
, (void *) 0);
395 WSwitchPanel
*wInitSwitchPanel(WScreen
*scr
, WWindow
*curwin
, Bool class_only
)
397 int wmScaleWidth
, wmScaleHeight
;
398 WMGetScaleBaseFromSystemFont(scr
->wmscreen
, &wmScaleWidth
, &wmScaleHeight
);
400 icon_size
= wPreferences
.switch_panel_icon_size
;
401 icon_tile_size
= (short int)(((float)icon_size
* 1.2F
) + 0.5F
);
402 border_space
= WMScaleY(10);
403 label_height
= WMScaleY(25);
406 WSwitchPanel
*panel
= wmalloc(sizeof(WSwitchPanel
));
408 int i
, width
, height
, iconsThatFitCount
, count
;
409 WMRect rect
= wGetRectForHead(scr
, wGetHeadForPointerLocation(scr
));
412 panel
->windows
= makeWindowListArray(scr
, wPreferences
.swtileImage
!= NULL
, class_only
);
413 count
= WMGetArrayItemCount(panel
->windows
);
415 panel
->flags
= makeWindowFlagsArray(count
);
418 WMFreeArray(panel
->windows
);
423 width
= icon_tile_size
* count
;
424 iconsThatFitCount
= count
;
426 if (width
> (int)rect
.size
.width
) {
427 iconsThatFitCount
= (rect
.size
.width
- SCREEN_BORDER_SPACING
) / icon_tile_size
;
428 width
= iconsThatFitCount
* icon_tile_size
;
431 panel
->visibleCount
= iconsThatFitCount
;
433 if (!wPreferences
.swtileImage
)
436 height
= label_height
+ icon_tile_size
;
438 panel
->tileTmp
= RCreateImage(icon_tile_size
, icon_tile_size
, 1);
439 panel
->tile
= getTile();
440 if (panel
->tile
&& wPreferences
.swbackImage
[8])
441 panel
->bg
= createBackImage(width
+ 2 * border_space
, height
+ 2 * border_space
);
443 if (!panel
->tileTmp
|| !panel
->tile
) {
445 RReleaseImage(panel
->bg
);
448 RReleaseImage(panel
->tile
);
451 RReleaseImage(panel
->tileTmp
);
452 panel
->tileTmp
= NULL
;
455 panel
->white
= WMWhiteColor(scr
->wmscreen
);
456 panel
->gray
= WMGrayColor(scr
->wmscreen
);
457 panel
->font
= WMBoldSystemFontOfSize(scr
->wmscreen
, WMScaleY(12));
458 panel
->icons
= WMCreateArray(count
);
459 panel
->images
= WMCreateArray(count
);
461 panel
->win
= WMCreateWindow(scr
->wmscreen
, "");
464 WMFrame
*frame
= WMCreateFrame(panel
->win
);
465 WMColor
*darkGray
= WMDarkGrayColor(scr
->wmscreen
);
466 WMSetFrameRelief(frame
, WRSimple
);
467 WMSetViewExpandsToParent(WMWidgetView(frame
), 0, 0, 0, 0);
469 panel
->label
= WMCreateLabel(panel
->win
);
470 WMResizeWidget(panel
->label
, width
, label_height
);
471 WMMoveWidget(panel
->label
, border_space
, border_space
+ icon_tile_size
+ 5);
472 WMSetLabelRelief(panel
->label
, WRSimple
);
473 WMSetWidgetBackgroundColor(panel
->label
, darkGray
);
474 WMSetLabelFont(panel
->label
, panel
->font
);
475 WMSetLabelTextColor(panel
->label
, panel
->white
);
477 WMReleaseColor(darkGray
);
481 WMResizeWidget(panel
->win
, width
+ 2 * border_space
, height
+ 2 * border_space
);
483 viewport
= WMCreateFrame(panel
->win
);
484 WMResizeWidget(viewport
, width
, icon_tile_size
);
485 WMMoveWidget(viewport
, border_space
, border_space
);
486 WMSetFrameRelief(viewport
, WRFlat
);
488 panel
->iconBox
= WMCreateFrame(viewport
);
489 WMMoveWidget(panel
->iconBox
, 0, 0);
490 WMResizeWidget(panel
->iconBox
, icon_tile_size
* count
, icon_tile_size
);
491 WMSetFrameRelief(panel
->iconBox
, WRFlat
);
493 WM_ITERATE_ARRAY(panel
->windows
, wwin
, i
) {
494 addIconForWindow(panel
, panel
->iconBox
, wwin
, i
* icon_tile_size
, 0);
497 WMMapSubwidgets(panel
->win
);
498 WMRealizeWidget(panel
->win
);
500 WM_ITERATE_ARRAY(panel
->windows
, wwin
, i
) {
501 changeImage(panel
, i
, 0, False
, True
);
507 RConvertImageMask(scr
->rcontext
, panel
->bg
, &pixmap
, &mask
, 250);
509 XSetWindowBackgroundPixmap(dpy
, WMWidgetXID(panel
->win
), pixmap
);
512 if (mask
&& w_global
.xext
.shape
.supported
)
513 XShapeCombineMask(dpy
, WMWidgetXID(panel
->win
), ShapeBounding
, 0, 0, mask
, ShapeSet
);
516 XFreePixmap(dpy
, pixmap
);
519 XFreePixmap(dpy
, mask
);
524 center
= wGetPointToCenterRectInHead(scr
, wGetHeadForPointerLocation(scr
),
525 width
+ 2 * border_space
, height
+ 2 * border_space
);
526 WMMoveWidget(panel
->win
, center
.x
, center
.y
);
529 panel
->current
= WMGetFirstInArray(panel
->windows
, curwin
);
530 if (panel
->current
>= 0)
531 changeImage(panel
, panel
->current
, 1, False
, False
);
533 WMMapWidget(panel
->win
);
538 void wSwitchPanelDestroy(WSwitchPanel
*panel
)
544 Window info_win
= panel
->scr
->info_window
;
546 ev
.xclient
.type
= ClientMessage
;
547 ev
.xclient
.message_type
= w_global
.atom
.wm
.ignore_focus_events
;
548 ev
.xclient
.format
= 32;
549 ev
.xclient
.data
.l
[0] = True
;
551 XSendEvent(dpy
, info_win
, True
, EnterWindowMask
, &ev
);
552 WMUnmapWidget(panel
->win
);
554 ev
.xclient
.data
.l
[0] = False
;
555 XSendEvent(dpy
, info_win
, True
, EnterWindowMask
, &ev
);
559 WM_ITERATE_ARRAY(panel
->images
, image
, i
) {
561 RReleaseImage(image
);
563 WMFreeArray(panel
->images
);
567 WMDestroyWidget(panel
->win
);
570 WMFreeArray(panel
->icons
);
573 WMFreeArray(panel
->flags
);
575 WMFreeArray(panel
->windows
);
578 RReleaseImage(panel
->tile
);
581 RReleaseImage(panel
->tileTmp
);
584 RReleaseImage(panel
->bg
);
587 WMReleaseFont(panel
->font
);
590 WMReleaseColor(panel
->white
);
593 WMReleaseColor(panel
->gray
);
598 WWindow
*wSwitchPanelSelectNext(WSwitchPanel
*panel
, int back
, int ignore_minimized
, Bool class_only
)
600 WWindow
*wwin
, *curwin
, *tmpwin
;
601 int count
= WMGetArrayItemCount(panel
->windows
);
602 int orig
= panel
->current
;
609 if (!wPreferences
.cycle_ignore_minimized
)
610 ignore_minimized
= False
;
612 if (ignore_minimized
&& canReceiveFocus(WMGetFromArray(panel
->windows
, (count
+ panel
->current
) % count
)) < 0)
613 ignore_minimized
= False
;
615 curwin
= WMGetFromArray(panel
->windows
, orig
);
623 panel
->current
= (count
+ panel
->current
) % count
;
624 wwin
= WMGetFromArray(panel
->windows
, panel
->current
);
628 if (panel
->current
== orig
)
630 } while (!sameWindowClass(wwin
, curwin
));
631 } while (ignore_minimized
&& panel
->current
!= orig
&& canReceiveFocus(wwin
) < 0);
633 WM_ITERATE_ARRAY(panel
->windows
, tmpwin
, i
) {
634 if (i
== panel
->current
)
636 if (!class_only
|| sameWindowClass(tmpwin
, curwin
))
637 changeImage(panel
, i
, 0, False
, False
);
641 changeImage(panel
, i
, 0, True
, False
);
646 if (panel
->current
< panel
->firstVisible
)
647 scrollIcons(panel
, panel
->current
- panel
->firstVisible
);
648 else if (panel
->current
- panel
->firstVisible
>= panel
->visibleCount
)
649 scrollIcons(panel
, panel
->current
- panel
->firstVisible
- panel
->visibleCount
+ 1);
652 drawTitle(panel
, panel
->current
, wwin
->frame
->title
);
653 if (panel
->current
!= orig
)
654 changeImage(panel
, orig
, 0, dim
, False
);
655 changeImage(panel
, panel
->current
, 1, False
, False
);
661 WWindow
*wSwitchPanelSelectFirst(WSwitchPanel
*panel
, int back
)
664 int count
= WMGetArrayItemCount(panel
->windows
);
672 panel
->current
= count
- 1;
673 scrollIcons(panel
, count
);
676 scrollIcons(panel
, -count
);
679 wwin
= WMGetFromArray(panel
->windows
, panel
->current
);
680 title
= wwin
->frame
->title
;
683 for (WMArrayFirst(panel
->windows
, &(i
)); (i
) != WANotFound
; WMArrayNext(panel
->windows
, &(i
)))
684 changeImage(panel
, i
, i
== panel
->current
, False
, False
);
686 drawTitle(panel
, panel
->current
, title
);
692 WWindow
*wSwitchPanelHandleEvent(WSwitchPanel
*panel
, XEvent
*event
)
701 if (event
->type
== MotionNotify
) {
702 WM_ITERATE_ARRAY(panel
->icons
, icon
, i
) {
703 if (WMWidgetXID(icon
) == event
->xmotion
.window
) {
710 if (focus
>= 0 && panel
->current
!= focus
) {
713 WM_ITERATE_ARRAY(panel
->windows
, wwin
, i
) {
714 changeImage(panel
, i
, i
== focus
, False
, False
);
716 panel
->current
= focus
;
718 wwin
= WMGetFromArray(panel
->windows
, focus
);
720 drawTitle(panel
, panel
->current
, wwin
->frame
->title
);
728 Window
wSwitchPanelGetWindow(WSwitchPanel
*swpanel
)
733 return WMWidgetXID(swpanel
->win
);