diff options
| author | Dana Jansens <danakj@orodu.net> | 2003-03-16 21:11:39 +0000 |
|---|---|---|
| committer | Dana Jansens <danakj@orodu.net> | 2003-03-16 21:11:39 +0000 |
| commit | f8a47de5ec444c452093371e3db16857eb39a490 (patch) | |
| tree | 31db2567842d98232775f9980f7a8d2586c0ac71 /engines | |
| parent | 8ba0586bcbdc7fe9648f1063812126d71a041670 (diff) | |
merge the C branch into HEAD
Diffstat (limited to 'engines')
| -rw-r--r-- | engines/.cvsignore | 2 | ||||
| -rw-r--r-- | engines/Makefile.am | 1 | ||||
| -rw-r--r-- | engines/engineinterface.h | 53 | ||||
| -rw-r--r-- | engines/openbox/.cvsignore | 7 | ||||
| -rw-r--r-- | engines/openbox/Makefile.am | 20 | ||||
| -rw-r--r-- | engines/openbox/openbox.c | 829 | ||||
| -rw-r--r-- | engines/openbox/openbox.h | 43 | ||||
| -rw-r--r-- | engines/openbox/theme.c | 327 | ||||
| -rw-r--r-- | engines/openbox/theme.h | 8 |
9 files changed, 1290 insertions, 0 deletions
diff --git a/engines/.cvsignore b/engines/.cvsignore new file mode 100644 index 00000000..282522db --- /dev/null +++ b/engines/.cvsignore @@ -0,0 +1,2 @@ +Makefile +Makefile.in diff --git a/engines/Makefile.am b/engines/Makefile.am new file mode 100644 index 00000000..db3bc8d7 --- /dev/null +++ b/engines/Makefile.am @@ -0,0 +1 @@ +SUBDIRS = openbox diff --git a/engines/engineinterface.h b/engines/engineinterface.h new file mode 100644 index 00000000..96796edd --- /dev/null +++ b/engines/engineinterface.h @@ -0,0 +1,53 @@ +#ifndef __engineinterface_h +#define __engineinterface_h + +#include "../kernel/frame.h" +#include <glib.h> + +/* startup */ +typedef gboolean EngineStartup(); + +/* shutdown */ +typedef void EngineShutdown(); + +/* frame_new */ +typedef Frame *EngineFrameNew(); + +/* frame_grab_client */ +typedef void EngineFrameGrabClient(Frame *self, Client *client); +/* frame_release_client */ +typedef void EngineFrameReleaseClient(Frame *self, Client *client); + +/* frame_adjust_size */ +/*! Update the frame's size to match the client */ +typedef void EngineFrameAdjustSize(Frame *self); +/* frame_adjust_position */ +/*! Update the frame's position to match the client */ +typedef void EngineFrameAdjustPosition(Frame *self); +/* frame_adjust_shape */ +/*! Shape the frame window to the client window */ +typedef void EngineFrameAdjustShape(Frame *self); +/* frame_adjust_state */ +/*! Update the frame to match the client's new state (for things like toggle + buttons, focus, and the title) XXX break this up */ +typedef void EngineFrameAdjustState(Frame *self); +/* frame_adjust_focus */ +/*! Update the frame to match the client's focused state */ +typedef void EngineFrameAdjustFocus(Frame *self); +/* frame_adjust_title */ +/*! Update the frame to display the client's current title */ +typedef void EngineFrameAdjustTitle(Frame *self); +/* frame_adjust_icon */ +/*! Update the frame to display the client's current icon */ +typedef void EngineFrameAdjustIcon(Frame *self); + +/* frame_show */ +/*! Shows the frame */ +typedef void EngineFrameShow(Frame *self); +/*! Hides the frame */ +typedef void EngineFrameHide(Frame *self); + +/* get_context */ +typedef GQuark EngineGetContext(Client *client, Window win); + +#endif diff --git a/engines/openbox/.cvsignore b/engines/openbox/.cvsignore new file mode 100644 index 00000000..b15c7f61 --- /dev/null +++ b/engines/openbox/.cvsignore @@ -0,0 +1,7 @@ +Makefile +Makefile.in +.deps +.libs +openbox.la +openbox.lo +theme.lo diff --git a/engines/openbox/Makefile.am b/engines/openbox/Makefile.am new file mode 100644 index 00000000..64f9315d --- /dev/null +++ b/engines/openbox/Makefile.am @@ -0,0 +1,20 @@ +enginedir=$(libdir)/openbox/engines +themedir=$(datadir)/openbox/themes/openbox + +CPPFLAGS=$(XFT_CFLAGS) $(GLIB_CFLAGS) @CPPFLAGS@ \ +-DENGINEDIR=\"$(enginedir)\" \ +-DTHEMEDIR=\"$(themedir)\" \ +-DDEFAULT_THEME=\"nyz\" \ +-DG_LOG_DOMAIN=\"Openbox-Engine\" + +engine_LTLIBRARIES=openbox.la + +openbox_la_LDFLAGS=-module -avoid-version +openbox_la_SOURCES=openbox.c theme.c + +noinst_HEADERS= + +MAINTAINERCLEANFILES= Makefile.in + +distclean-local: + $(RM) *\~ *.orig *.rej .\#* diff --git a/engines/openbox/openbox.c b/engines/openbox/openbox.c new file mode 100644 index 00000000..fdab22a3 --- /dev/null +++ b/engines/openbox/openbox.c @@ -0,0 +1,829 @@ +#include "theme.h" +#include "../../kernel/openbox.h" +#include "../../kernel/screen.h" +#include "../../kernel/extensions.h" +#include "../../kernel/themerc.h" +#include "../../kernel/frame.h" +#include "../../render/render.h" +#include "../../render/color.h" + +#include <X11/Xlib.h> +#include <glib.h> + +#define TITLE_HEIGHT (s_font_height + s_bevel * 2) +#define LABEL_HEIGHT (s_font_height) +#define HANDLE_Y(f) (f->innersize.top + f->frame.client->area.height + \ + f->cbwidth) +#define BUTTON_SIZE (LABEL_HEIGHT - 2) +#define GRIP_WIDTH (BUTTON_SIZE * 2) +#define HANDLE_WIDTH(f) (f->width - (GRIP_WIDTH + f->bwidth) * 2) + +#define PLATE_EVENTMASK (SubstructureRedirectMask | ButtonPressMask) +#define FRAME_EVENTMASK (EnterWindowMask | LeaveWindowMask) + +/* style settings - geometry */ +int s_font_height; +int s_bevel; +int s_handle_height; +int s_bwidth; +int s_cbwidth; +/* style settings - colors */ +color_rgb *s_b_color; +color_rgb *s_cb_focused_color; +color_rgb *s_cb_unfocused_color; + +/* global appearances */ +Appearance *a_focused_unpressed_max; +Appearance *a_focused_pressed_max; +Appearance *a_unfocused_unpressed_max; +Appearance *a_unfocused_pressed_max; +Appearance *a_focused_unpressed_close; +Appearance *a_focused_pressed_close; +Appearance *a_unfocused_unpressed_close; +Appearance *a_unfocused_pressed_close; +Appearance *a_focused_unpressed_desk; +Appearance *a_focused_pressed_desk; +Appearance *a_unfocused_unpressed_desk; +Appearance *a_unfocused_pressed_desk; +Appearance *a_focused_unpressed_iconify; +Appearance *a_focused_pressed_iconify; +Appearance *a_unfocused_unpressed_iconify; +Appearance *a_unfocused_pressed_iconify; +Appearance *a_focused_grip; +Appearance *a_unfocused_grip; +Appearance *a_focused_title; +Appearance *a_unfocused_title; +Appearance *a_focused_label; +Appearance *a_unfocused_label; +Appearance *a_icon; /* always parentrelative, so no focused/unfocused */ +Appearance *a_focused_handle; +Appearance *a_unfocused_handle; + +typedef struct ObFrame { + Frame frame; + + Window title; + Window label; + Window max; + Window close; + Window desk; + Window icon; + Window iconify; + Window handle; + Window lgrip; + Window rgrip; + + Appearance *a_unfocused_title; + Appearance *a_focused_title; + Appearance *a_unfocused_label; + Appearance *a_focused_label; + Appearance *a_icon; + Appearance *a_unfocused_handle; + Appearance *a_focused_handle; + + Strut innersize; + + GSList *clients; + + int width; /* title and handle */ + int label_width; + int icon_x; /* x-position of the window icon button */ + int label_x; /* x-position of the window title */ + int iconify_x; /* x-position of the window iconify button */ + int desk_x; /* x-position of the window all-desktops button */ + int max_x; /* x-position of the window maximize button */ + int close_x; /* x-position of the window close button */ + int bwidth; /* border width */ + int cbwidth; /* client border width */ + + gboolean max_press; + gboolean close_press; + gboolean desk_press; + gboolean iconify_press; +} ObFrame; + +static void layout_title(ObFrame *self); +static void render(ObFrame *self); +static void render_label(ObFrame *self); +static void render_max(ObFrame *self); +static void render_icon(ObFrame *self); +static void render_iconify(ObFrame *self); +static void render_desk(ObFrame *self); +static void render_close(ObFrame *self); + +gboolean startup() +{ + g_quark_from_string("none"); + g_quark_from_string("root"); + g_quark_from_string("client"); + g_quark_from_string("titlebar"); + g_quark_from_string("handle"); + g_quark_from_string("frame"); + g_quark_from_string("blcorner"); + g_quark_from_string("brcorner"); + g_quark_from_string("maximize"); + g_quark_from_string("alldesktops"); + g_quark_from_string("iconify"); + g_quark_from_string("icon"); + g_quark_from_string("close"); + + s_b_color = s_cb_unfocused_color = s_cb_focused_color = NULL; + + a_focused_unpressed_max = appearance_new(Surface_Planar, 0);//1); + a_focused_pressed_max = appearance_new(Surface_Planar, 0);//1); + a_unfocused_unpressed_max = appearance_new(Surface_Planar, 0);//1); + a_unfocused_pressed_max = appearance_new(Surface_Planar, 0);//1); + a_focused_unpressed_close = NULL; + a_focused_pressed_close = NULL; + a_unfocused_unpressed_close = NULL; + a_unfocused_pressed_close = NULL; + a_focused_unpressed_desk = NULL; + a_focused_pressed_desk = NULL; + a_unfocused_unpressed_desk = NULL; + a_unfocused_pressed_desk = NULL; + a_focused_unpressed_iconify = NULL; + a_focused_pressed_iconify = NULL; + a_unfocused_unpressed_iconify = NULL; + a_unfocused_pressed_iconify = NULL; + a_focused_grip = appearance_new(Surface_Planar, 0); + a_unfocused_grip = appearance_new(Surface_Planar, 0); + a_focused_title = appearance_new(Surface_Planar, 0); + a_unfocused_title = appearance_new(Surface_Planar, 0); + a_focused_label = appearance_new(Surface_Planar, 0);//1); + a_unfocused_label = appearance_new(Surface_Planar, 0);//1); + a_icon = appearance_new(Surface_Planar, 0);//1); + a_focused_handle = appearance_new(Surface_Planar, 0); + a_unfocused_handle = appearance_new(Surface_Planar, 0); + + return load(); +} + +void shutdown() +{ + if (s_b_color != NULL) color_free(s_b_color); + if (s_cb_unfocused_color != NULL) color_free(s_cb_unfocused_color); + if (s_cb_focused_color != NULL) color_free(s_cb_focused_color); + + appearance_free(a_focused_unpressed_max); + appearance_free(a_focused_pressed_max); + appearance_free(a_unfocused_unpressed_max); + appearance_free(a_unfocused_pressed_max); + if (a_focused_unpressed_close != NULL) + appearance_free(a_focused_unpressed_close); + if (a_focused_pressed_close != NULL) + appearance_free(a_focused_pressed_close); + if (a_unfocused_unpressed_close != NULL) + appearance_free(a_unfocused_unpressed_close); + if (a_unfocused_pressed_close != NULL) + appearance_free(a_unfocused_pressed_close); + if (a_focused_unpressed_desk != NULL) + appearance_free(a_focused_unpressed_desk); + if (a_focused_pressed_desk != NULL) + appearance_free(a_focused_pressed_desk); + if (a_unfocused_unpressed_desk != NULL) + appearance_free(a_unfocused_unpressed_desk); + if (a_unfocused_pressed_desk != NULL) + appearance_free(a_unfocused_pressed_desk); + if (a_focused_unpressed_iconify != NULL) + appearance_free(a_focused_unpressed_iconify); + if (a_focused_pressed_iconify != NULL) + appearance_free(a_focused_pressed_iconify); + if (a_unfocused_unpressed_iconify != NULL) + appearance_free(a_unfocused_unpressed_iconify); + if (a_unfocused_pressed_iconify != NULL) + appearance_free(a_unfocused_pressed_iconify); + appearance_free(a_focused_grip); + appearance_free(a_unfocused_grip); + appearance_free(a_focused_title); + appearance_free(a_unfocused_title); + appearance_free(a_focused_label); + appearance_free(a_unfocused_label); + appearance_free(a_icon); + appearance_free(a_focused_handle); + appearance_free(a_unfocused_handle); +} + +static Window createWindow(Window parent, unsigned long mask, + XSetWindowAttributes *attrib) +{ + /* XXX DONT USE THE DEFAULT SHIT */ + return XCreateWindow(ob_display, parent, 0, 0, 1, 1, 0, + DefaultDepth(ob_display, ob_screen), InputOutput, + DefaultVisual(ob_display, ob_screen), + mask, attrib); + +} + +Frame *frame_new() +{ + XSetWindowAttributes attrib; + unsigned long mask; + ObFrame *self; + + self = g_new(ObFrame, 1); + + self->frame.visible = FALSE; + + /* create all of the decor windows */ + mask = CWOverrideRedirect | CWEventMask; + attrib.event_mask = FRAME_EVENTMASK; + attrib.override_redirect = TRUE; + self->frame.window = createWindow(ob_root, mask, &attrib); + + mask = 0; + self->frame.plate = createWindow(self->frame.window, mask, &attrib); + + mask = CWEventMask; + attrib.event_mask = (ButtonPressMask | ButtonReleaseMask | + ButtonMotionMask | ExposureMask); + self->title = createWindow(self->frame.window, mask, &attrib); + self->label = createWindow(self->title, mask, &attrib); + self->max = createWindow(self->title, mask, &attrib); + self->close = createWindow(self->title, mask, &attrib); + self->desk = createWindow(self->title, mask, &attrib); + self->icon = createWindow(self->title, mask, &attrib); + self->iconify = createWindow(self->title, mask, &attrib); + self->handle = createWindow(self->frame.window, mask, &attrib); + mask |= CWCursor; + attrib.cursor = ob_cursors.ll_angle; + self->lgrip = createWindow(self->handle, mask, &attrib); + attrib.cursor = ob_cursors.lr_angle; + self->rgrip = createWindow(self->handle, mask, &attrib); + + /* the other stuff is shown based on decor settings */ + XMapWindow(ob_display, self->frame.plate); + XMapWindow(ob_display, self->lgrip); + XMapWindow(ob_display, self->rgrip); + XMapWindow(ob_display, self->label); + + /* set colors/appearance/sizes for stuff that doesn't change */ + XSetWindowBorder(ob_display, self->frame.window, s_b_color->pixel); + XSetWindowBorder(ob_display, self->label, s_b_color->pixel); + XSetWindowBorder(ob_display, self->rgrip, s_b_color->pixel); + XSetWindowBorder(ob_display, self->lgrip, s_b_color->pixel); + + XResizeWindow(ob_display, self->max, BUTTON_SIZE, BUTTON_SIZE); + XResizeWindow(ob_display, self->iconify, BUTTON_SIZE, BUTTON_SIZE); + XResizeWindow(ob_display, self->icon, BUTTON_SIZE, BUTTON_SIZE); + XResizeWindow(ob_display, self->close, BUTTON_SIZE, BUTTON_SIZE); + XResizeWindow(ob_display, self->desk, BUTTON_SIZE, BUTTON_SIZE); + XResizeWindow(ob_display, self->lgrip, GRIP_WIDTH, s_handle_height); + XResizeWindow(ob_display, self->rgrip, GRIP_WIDTH, s_handle_height); + + /* set up the dynamic appearances */ + self->a_unfocused_title = appearance_copy(a_unfocused_title); + self->a_focused_title = appearance_copy(a_focused_title); + self->a_unfocused_label = appearance_copy(a_unfocused_label); + self->a_focused_label = appearance_copy(a_focused_label); + self->a_unfocused_handle = appearance_copy(a_unfocused_handle); + self->a_focused_handle = appearance_copy(a_focused_handle); + self->a_icon = appearance_copy(a_icon); + + self->max_press = self->close_press = self->desk_press = + self->iconify_press = FALSE; + + return (Frame*)self; +} + +static void frame_free(ObFrame *self) +{ + appearance_free(self->a_unfocused_title); + appearance_free(self->a_focused_title); + appearance_free(self->a_unfocused_label); + appearance_free(self->a_focused_label); + appearance_free(self->a_unfocused_handle); + appearance_free(self->a_focused_handle); + appearance_free(self->a_icon); + + XDestroyWindow(ob_display, self->frame.window); + + g_free(self); +} + +void frame_show(ObFrame *self) +{ + if (!self->frame.visible) { + self->frame.visible = TRUE; + XMapWindow(ob_display, self->frame.window); + } +} + +void frame_hide(ObFrame *self) +{ + if (self->frame.visible) { + self->frame.visible = FALSE; + self->frame.client->ignore_unmaps++; + XUnmapWindow(ob_display, self->frame.window); + } +} + +void frame_adjust_shape(ObFrame *self) +{ +#ifdef SHAPE + int num; + XRectangle xrect[2]; + + if (!self->frame.client->shaped) { + /* clear the shape on the frame window */ + XShapeCombineMask(ob_display, self->frame.window, ShapeBounding, + self->innersize.left, + self->innersize.top, + None, ShapeSet); + } else { + /* make the frame's shape match the clients */ + XShapeCombineShape(ob_display, self->frame.window, ShapeBounding, + self->innersize.left, + self->innersize.top, + self->frame.client->window, + ShapeBounding, ShapeSet); + + num = 0; + if (self->frame.client->decorations & Decor_Titlebar) { + xrect[0].x = -s_bevel; + xrect[0].y = -s_bevel; + xrect[0].width = self->width + self->bwidth * 2; + xrect[0].height = TITLE_HEIGHT + + self->bwidth * 2; + ++num; + } + + if (self->frame.client->decorations & Decor_Handle) { + xrect[1].x = -s_bevel; + xrect[1].y = HANDLE_Y(self); + xrect[1].width = self->width + self->bwidth * 2; + xrect[1].height = s_handle_height + + self->bwidth * 2; + ++num; + } + + XShapeCombineRectangles(ob_display, self->frame.window, + ShapeBounding, 0, 0, xrect, num, + ShapeUnion, Unsorted); + } +#endif +} + +void frame_adjust_size(ObFrame *self) +{ + if (self->frame.client->decorations & Decor_Border) { + self->bwidth = s_bwidth; + self->cbwidth = s_cbwidth; + } else { + self->bwidth = self->cbwidth = 0; + } + STRUT_SET(self->innersize, self->cbwidth, self->cbwidth, + self->cbwidth, self->cbwidth); + self->width = self->frame.client->area.width + self->cbwidth * 2; + g_assert(self->width > 0); + + /* set border widths */ + XSetWindowBorderWidth(ob_display, self->frame.plate, self->cbwidth); + XSetWindowBorderWidth(ob_display, self->frame.window, self->bwidth); + XSetWindowBorderWidth(ob_display, self->title, self->bwidth); + XSetWindowBorderWidth(ob_display, self->handle, self->bwidth); + XSetWindowBorderWidth(ob_display, self->lgrip, self->bwidth); + XSetWindowBorderWidth(ob_display, self->rgrip, self->bwidth); + + /* position/size and map/unmap all the windows */ + + if (self->frame.client->decorations & Decor_Titlebar) { + XMoveResizeWindow(ob_display, self->title, + -self->bwidth, -self->bwidth, + self->width, TITLE_HEIGHT); + self->innersize.top += TITLE_HEIGHT + self->bwidth; + XMapWindow(ob_display, self->title); + + /* layout the title bar elements */ + layout_title(self); + } else { + XUnmapWindow(ob_display, self->title); + /* make all the titlebar stuff not render */ + self->frame.client->decorations &= ~(Decor_Icon | Decor_Iconify | + Decor_Maximize | Decor_Close | + Decor_AllDesktops); + } + + if (self->frame.client->decorations & Decor_Handle) { + XMoveResizeWindow(ob_display, self->handle, + -self->bwidth, HANDLE_Y(self), + self->width, s_handle_height); + XMoveWindow(ob_display, self->lgrip, + -self->bwidth, -self->bwidth); + XMoveWindow(ob_display, self->rgrip, + -self->bwidth + self->width - + GRIP_WIDTH, -self->bwidth); + self->innersize.bottom += s_handle_height + + self->bwidth; + XMapWindow(ob_display, self->handle); + } else + XUnmapWindow(ob_display, self->handle); + + XResizeWindow(ob_display, self->frame.window, self->width, + (self->frame.client->shaded ? TITLE_HEIGHT : + self->innersize.top + self->innersize.bottom + + self->frame.client->area.height)); + + /* do this in two steps because clients whose gravity is set to + 'Static' don't end up getting moved at all with an XMoveResizeWindow */ + XMoveWindow(ob_display, self->frame.plate, + self->innersize.left - self->cbwidth, + self->innersize.top - self->cbwidth); + XResizeWindow(ob_display, self->frame.plate, + self->frame.client->area.width, + self->frame.client->area.height); + + STRUT_SET(self->frame.size, + self->innersize.left + self->bwidth, + self->innersize.top + self->bwidth, + self->innersize.right + self->bwidth, + self->innersize.bottom + self->bwidth); + + RECT_SET_SIZE(self->frame.area, + self->frame.client->area.width + + self->frame.size.left + self->frame.size.right, + self->frame.client->area.height + + self->frame.size.top + self->frame.size.bottom); + + render(self); + + frame_adjust_shape(self); +} + +void frame_adjust_position(ObFrame *self) +{ + self->frame.area.x = self->frame.client->area.x; + self->frame.area.y = self->frame.client->area.y; + frame_client_gravity((Frame*)self, + &self->frame.area.x, &self->frame.area.y); + XMoveWindow(ob_display, self->frame.window, + self->frame.area.x, self->frame.area.y); +} + +void frame_adjust_state(ObFrame *self) +{ + render_max(self); + render_desk(self); +} + +void frame_adjust_focus(ObFrame *self) +{ + render(self); +} + +void frame_adjust_title(ObFrame *self) +{ + render_label(self); +} + +void frame_adjust_icon(ObFrame *self) +{ + render_icon(self); +} + +void frame_grab_client(ObFrame *self, Client *client) +{ + self->frame.client = client; + + /* reparent the client to the frame */ + XReparentWindow(ob_display, client->window, self->frame.plate, 0, 0); + /* + When reparenting the client window, it is usually not mapped yet, since + this occurs from a MapRequest. However, in the case where Openbox is + starting up, the window is already mapped, so we'll see unmap events for + it. There are 2 unmap events generated that we see, one with the 'event' + member set the root window, and one set to the client, but both get + handled and need to be ignored. + */ + if (ob_state == State_Starting) + client->ignore_unmaps += 2; + + /* select the event mask on the client's parent (to receive config/map + req's) the ButtonPress is to catch clicks on the client border */ + XSelectInput(ob_display, self->frame.plate, PLATE_EVENTMASK); + + /* map the client so it maps when the frame does */ + XMapWindow(ob_display, client->window); + + frame_adjust_size(self); + frame_adjust_position(self); + + /* set all the windows for the frame in the client_map */ + g_hash_table_insert(client_map, (gpointer)self->frame.window, client); + g_hash_table_insert(client_map, (gpointer)self->frame.plate, client); + g_hash_table_insert(client_map, (gpointer)self->title, client); + g_hash_table_insert(client_map, (gpointer)self->label, client); + g_hash_table_insert(client_map, (gpointer)self->max, client); + g_hash_table_insert(client_map, (gpointer)self->close, client); + g_hash_table_insert(client_map, (gpointer)self->desk, client); + g_hash_table_insert(client_map, (gpointer)self->icon, client); + g_hash_table_insert(client_map, (gpointer)self->iconify, client); + g_hash_table_insert(client_map, (gpointer)self->handle, client); + g_hash_table_insert(client_map, (gpointer)self->lgrip, client); + g_hash_table_insert(client_map, (gpointer)self->rgrip, client); +} + +void frame_release_client(ObFrame *self, Client *client) +{ + XEvent ev; + + g_assert(self->frame.client == client); + + /* check if the app has already reparented its window away */ + if (XCheckTypedWindowEvent(ob_display, client->window, + ReparentNotify, &ev)) { + XPutBackEvent(ob_display, &ev); + /* re-map the window since the unmanaging process unmaps it */ + XMapWindow(ob_display, client->window); + } else { + /* according to the ICCCM - if the client doesn't reparent itself, + then we will reparent the window to root for them */ + XReparentWindow(ob_display, client->window, ob_root, + client->area.x, + client->area.y); + } + + /* remove all the windows for the frame from the client_map */ + g_hash_table_remove(client_map, (gpointer)self->frame.window); + g_hash_table_remove(client_map, (gpointer)self->frame.plate); + g_hash_table_remove(client_map, (gpointer)self->title); + g_hash_table_remove(client_map, (gpointer)self->label); + g_hash_table_remove(client_map, (gpointer)self->max); + g_hash_table_remove(client_map, (gpointer)self->close); + g_hash_table_remove(client_map, (gpointer)self->desk); + g_hash_table_remove(client_map, (gpointer)self->icon); + g_hash_table_remove(client_map, (gpointer)self->iconify); + g_hash_table_remove(client_map, (gpointer)self->handle); + g_hash_table_remove(client_map, (gpointer)self->lgrip); + g_hash_table_remove(client_map, (gpointer)self->rgrip); + + frame_free(self); +} + +static void layout_title(ObFrame *self) +{ + const char *lc; + int x; + gboolean n, d, i, l, m ,c; + n = d = i = l = m = c = FALSE; + + /* figure out whats being shown, and the width of the label */ + self->label_width = self->width - s_bevel * 2; + for (lc = themerc_titlebar_layout; *lc != '\0'; ++lc) { + switch (*lc) { + case 'N': + if (!(self->frame.client->decorations & Decor_Icon)) break; + n = TRUE; + self->label_width -= BUTTON_SIZE + s_bevel; + break; + case 'D': + if (!(self->frame.client->decorations & Decor_AllDesktops)) break; + d = TRUE; + self->label_width -= BUTTON_SIZE + s_bevel; + break; + case 'I': + if (!(self->frame.client->decorations & Decor_Iconify)) break; + i = TRUE; + self->label_width -= BUTTON_SIZE + s_bevel; + break; + case 'L': + l = TRUE; + break; + case 'M': + if (!(self->frame.client->decorations & Decor_Maximize)) break; + m = TRUE; + self->label_width -= BUTTON_SIZE + s_bevel; + break; + case 'C': + if (!(self->frame.client->decorations & Decor_Close)) break; + c = TRUE; + self->label_width -= BUTTON_SIZE + s_bevel; + break; + } + } + if (self->label_width < 1) self->label_width = 1; + + XResizeWindow(ob_display, self->label, self->label_width, s_font_height); + + if (!n) { + self->frame.client->decorations &= ~Decor_Icon; + XUnmapWindow(ob_display, self->icon); + self->icon_x = -1; + } + if (!d) { + self->frame.client->decorations &= ~Decor_AllDesktops; + XUnmapWindow(ob_display, self->desk); + self->desk_x = -1; + } + if (!i) { + self->frame.client->decorations &= ~Decor_Iconify; + XUnmapWindow(ob_display, self->iconify); + self->icon_x = -1; + } + if (!l) { + XUnmapWindow(ob_display, self->label); + self->label_x = -1; + } + if (!m) { + self->frame.client->decorations &= ~Decor_Maximize; + XUnmapWindow(ob_display, self->max); + self->max_x = -1; + } + if (!c) { + self->frame.client->decorations &= ~Decor_Close; + XUnmapWindow(ob_display, self->close); + self->close_x = -1; + } + + x = s_bevel; + for (lc = themerc_titlebar_layout; *lc != '\0'; ++lc) { + switch (*lc) { + case 'N': + if (!n) break; + self->icon_x = x; + XMapWindow(ob_display, self->icon); + XMoveWindow(ob_display, self->icon, x, s_bevel + 1); + x += BUTTON_SIZE + s_bevel; + break; + case 'D': + if (!d) break; + self->desk_x = x; + XMapWindow(ob_display, self->desk); + XMoveWindow(ob_display, self->desk, x, s_bevel + 1); + x += BUTTON_SIZE + s_bevel; + break; + case 'I': + if (!i) break; + self->iconify_x = x; + XMapWindow(ob_display, self->iconify); + XMoveWindow(ob_display, self->iconify, x, s_bevel + 1); + x += BUTTON_SIZE + s_bevel; + break; + case 'L': + if (!l) break; + self->label_x = x; + XMapWindow(ob_display, self->label); + XMoveWindow(ob_display, self->label, x, s_bevel); + x += self->label_width + s_bevel; + break; + case 'M': + if (!m) break; + self->max_x = x; + XMapWindow(ob_display, self->max); + XMoveWindow(ob_display, self->max, x, s_bevel + 1); + x += BUTTON_SIZE + s_bevel; + break; + case 'C': + if (!c) break; + self->close_x = x; + XMapWindow(ob_display, self->close); + XMoveWindow(ob_display, self->close, x, s_bevel + 1); + x += BUTTON_SIZE + s_bevel; + break; + } + } +} + +static void render(ObFrame *self) +{ + if (self->frame.client->focused) { + XSetWindowBorder(ob_display, self->frame.plate, + s_cb_focused_color->pixel); + } else { + XSetWindowBorder(ob_display, self->frame.plate, + s_cb_unfocused_color->pixel); + } + + if (self->frame.client->decorations & Decor_Titlebar) { + paint(self->title, (self->frame.client->focused ? + self->a_focused_title : + self->a_unfocused_title), + self->width, TITLE_HEIGHT); + render_label(self); + render_max(self); + render_icon(self); + render_iconify(self); + render_desk(self); + render_close(self); + } + + if (self->frame.client->decorations & Decor_Handle) { + paint(self->handle, (self->frame.client->focused ? + self->a_focused_handle : + self->a_unfocused_handle), + HANDLE_WIDTH(self), s_handle_height); + paint(self->lgrip, (self->frame.client->focused ? + a_focused_grip : + a_unfocused_grip), + GRIP_WIDTH, s_handle_height); + paint(self->rgrip, (self->frame.client->focused ? + a_focused_grip : + a_unfocused_grip), + GRIP_WIDTH, s_handle_height); + } +} + +static void render_label(ObFrame *self) +{ + if (self->label_x < 0) return; + + /* XXX set the texture's text! */ + paint(self->label, (self->frame.client->focused ? + self->a_focused_label : + self->a_unfocused_label), + self->label_width, LABEL_HEIGHT); +} + +static void render_icon(ObFrame *self) +{ + if (self->icon_x < 0) return; + + /* XXX set the texture's icon picture! */ + paint(self->icon, self->a_icon, BUTTON_SIZE, BUTTON_SIZE); +} + +static void render_max(ObFrame *self) +{ + gboolean press = self->max_press || + self->frame.client->max_vert || self->frame.client->max_horz; + + if (self->max_x < 0) return; + + paint(self->max, (self->frame.client->focused ? + (press ? + a_focused_pressed_max : + a_focused_unpressed_max) : + (press ? + a_unfocused_pressed_max : + a_unfocused_unpressed_max)), + BUTTON_SIZE, BUTTON_SIZE); +} + +static void render_iconify(ObFrame *self) +{ + if (self->iconify_x < 0) return; + + paint(self->iconify, (self->frame.client->focused ? + (self->iconify_press ? + a_focused_pressed_iconify : + a_focused_unpressed_iconify) : + (self->iconify_press ? + a_unfocused_pressed_iconify : + a_unfocused_unpressed_iconify)), + BUTTON_SIZE, BUTTON_SIZE); +} + +static void render_desk(ObFrame *self) +{ + gboolean press = self->desk_press || + self->frame.client->desktop == DESKTOP_ALL; + + if (self->desk_x < 0) return; + + paint(self->desk, (self->frame.client->focused ? + (press ? + a_focused_pressed_desk : + a_focused_unpressed_desk) : + (press ? + a_unfocused_pressed_desk : + a_unfocused_unpressed_desk)), + BUTTON_SIZE, BUTTON_SIZE); +} + +static void render_close(ObFrame *self) +{ + if (self->close_x < 0) return; + + paint(self->close, (self->frame.client->focused ? + (self->close_press ? + a_focused_pressed_close : + a_focused_unpressed_close) : + (self->close_press ? + a_unfocused_pressed_close : + a_unfocused_unpressed_close)), + BUTTON_SIZE, BUTTON_SIZE); +} + +GQuark get_context(Client *client, Window win) +{ + ObFrame *self; + + if (win == ob_root) return g_quark_try_string("root"); + if (client == NULL) return g_quark_try_string("none"); + if (win == client->window) return g_quark_try_string("client"); + + self = (ObFrame*) client->frame; + if (win == self->frame.window) return g_quark_try_string("frame"); + if (win == self->frame.plate) return g_quark_try_string("client"); + if (win == self->title) return g_quark_try_string("titlebar"); + if (win == self->label) return g_quark_try_string("titlebar"); + if (win == self->handle) return g_quark_try_string("handle"); + if (win == self->lgrip) return g_quark_try_string("blcorner"); + if (win == self->rgrip) return g_quark_try_string("brcorner"); + if (win == self->max) return g_quark_try_string("maximize"); + if (win == self->iconify) return g_quark_try_string("iconify"); + if (win == self->close) return g_quark_try_string("close"); + if (win == self->icon) return g_quark_try_string("icon"); + if (win == self->desk) return g_quark_try_string("alldesktops"); + + return g_quark_try_string("none"); +} diff --git a/engines/openbox/openbox.h b/engines/openbox/openbox.h new file mode 100644 index 00000000..f072a6e0 --- /dev/null +++ b/engines/openbox/openbox.h @@ -0,0 +1,43 @@ +#ifndef __engine_openbox_h +#define __engine_openbox_h + +#include "../../render/render.h" +#include "../../render/color.h" + +extern int s_font_height; +extern int s_bevel; +extern int s_handle_height; +extern int s_bwidth; +extern int s_cbwidth; + +extern color_rgb *s_b_color; +extern color_rgb *s_cb_focused_color; +extern color_rgb *s_cb_unfocused_color; + +extern Appearance *a_focused_unpressed_max; +extern Appearance *a_focused_pressed_max; +extern Appearance *a_unfocused_unpressed_max; +extern Appearance *a_unfocused_pressed_max; +extern Appearance *a_focused_unpressed_close; +extern Appearance *a_focused_pressed_close; +extern Appearance *a_unfocused_unpressed_close; +extern Appearance *a_unfocused_pressed_close; +extern Appearance *a_focused_unpressed_desk; +extern Appearance *a_focused_pressed_desk; +extern Appearance *a_unfocused_unpressed_desk; +extern Appearance *a_unfocused_pressed_desk; +extern Appearance *a_focused_unpressed_iconify; +extern Appearance *a_focused_pressed_iconify; +extern Appearance *a_unfocused_unpressed_iconify; +extern Appearance *a_unfocused_pressed_iconify; +extern Appearance *a_focused_grip; +extern Appearance *a_unfocused_grip; +extern Appearance *a_focused_title; +extern Appearance *a_unfocused_title; +extern Appearance *a_focused_label; +extern Appearance *a_unfocused_label; +extern Appearance *a_icon; +extern Appearance *a_focused_handle; +extern Appearance *a_unfocused_handle; + +#endif diff --git a/engines/openbox/theme.c b/engines/openbox/theme.c new file mode 100644 index 00000000..2c96224f --- /dev/null +++ b/engines/openbox/theme.c @@ -0,0 +1,327 @@ +#include "openbox.h" +#include "../../kernel/themerc.h" + +#include <glib.h> +#include <X11/Xlib.h> +#include <X11/Xresource.h> +#ifdef HAVE_STDLIB_H +# include <stdlib.h> +#endif +#ifdef HAVE_CTYPE_H +# include <ctype.h> +#endif +#ifdef HAVE_STRING_H +# include <string.h> +#endif + +static XrmDatabase loaddb(char *theme) +{ + XrmDatabase db; + + db = XrmGetFileDatabase(theme); + if (db == NULL) { + char *s = g_build_filename(g_get_home_dir(), ".openbox", "themes", + "openbox", theme, NULL); + db = XrmGetFileDatabase(s); + g_free(s); + } + if (db == NULL) { + char *s = g_build_filename(THEMEDIR, theme, NULL); + db = XrmGetFileDatabase(s); + g_free(s); + } + return db; +} + +static char *create_class_name(char *rname) +{ + char *rclass = g_strdup(rname); + char *p = rclass; + + while (TRUE) { + *p = toupper(*p); + p = strchr(p+1, '.'); + if (p == NULL) break; + ++p; + if (*p == '\0') break; + } + return rclass; +} + +gboolean read_bool(XrmDatabase db, char *rname, gboolean *value) +{ + gboolean ret = FALSE; + char *rclass = create_class_name(rname); + char *rettype; + XrmValue retvalue; + + if (XrmGetResource(db, rname, rclass, &rettype, &retvalue) && + retvalue.addr != NULL) { + if (!g_ascii_strcasecmp(retvalue.addr, "true")) + *value = TRUE; + else + *value = FALSE; + ret = TRUE; + } + + g_free(rclass); + return ret; +} + +gboolean read_int(XrmDatabase db, char *rname, int *value) +{ + gboolean ret = FALSE; + char *rclass = create_class_name(rname); + char *rettype, *end; + XrmValue retvalue; + + if (XrmGetResource(db, rname, rclass, &rettype, &retvalue) && + retvalue.addr != NULL) { + *value = (int)strtol(retvalue.addr, &end, 10); + if (end != retvalue.addr) + ret = TRUE; + } + + g_free(rclass); + return ret; +} + +gboolean read_string(XrmDatabase db, char *rname, char **value) +{ + gboolean ret = FALSE; + char *rclass = create_class_name(rname); + char *rettype; + XrmValue retvalue; + + if (XrmGetResource(db, rname, rclass, &rettype, &retvalue) && + retvalue.addr != NULL) { + *value = retvalue.addr; + ret = TRUE; + } + + g_free(rclass); + return ret; +} + +gboolean read_color(XrmDatabase db, char *rname, color_rgb **value) +{ + gboolean ret = FALSE; + char *rclass = create_class_name(rname); + char *rettype; + XrmValue retvalue; + + if (XrmGetResource(db, rname, rclass, &rettype, &retvalue) && + retvalue.addr != NULL) { + color_rgb *c = color_parse(retvalue.addr); + if (c != NULL) { + *value = c; + ret = TRUE; + } + } + + g_free(rclass); + return ret; +} + +static void parse_appearance(char *tex, SurfaceColorType *grad, + ReliefType *relief, BevelType *bevel, + gboolean *interlaced, gboolean *border) +{ + char *t; + + /* convert to all lowercase */ + for (t = tex; *t != '\0'; ++t) + *t = g_ascii_tolower(*t); + + if (strstr(tex, "parentrelative") != NULL) { + *grad = Background_ParentRelative; + } else { + if (strstr(tex, "gradient") != NULL) { + if (strstr(tex, "crossdiagonal") != NULL) + *grad = Background_CrossDiagonal; + else if (strstr(tex, "rectangle") != NULL) + *grad = Background_Rectangle; + else if (strstr(tex, "pyramid") != NULL) + *grad = Background_Pyramid; + else if (strstr(tex, "pipecross") != NULL) + *grad = Background_PipeCross; + else if (strstr(tex, "elliptic") != NULL) + *grad = Background_Elliptic; + else if (strstr(tex, "horizontal") != NULL) + *grad = Background_Horizontal; + else if (strstr(tex, "vertical") != NULL) + *grad = Background_Vertical; + else + *grad = Background_Diagonal; + } else { + *grad = Background_Solid; + } + + if (strstr(tex, "sunken") != NULL) + *relief = Sunken; + else if (strstr(tex, "flat") != NULL) + *relief = Flat; + else + *relief = Raised; + + *border = FALSE; + if (*relief == Flat) { + if (strstr(tex, "border") != NULL) + *border = TRUE; + } else { + if (strstr(tex, "bevel2") != NULL) + *bevel = Bevel2; + else + *bevel = Bevel1; + } + + if (strstr(tex, "interlaced") != NULL) + *interlaced = TRUE; + else + *interlaced = FALSE; + } +} + + +gboolean read_appearance(XrmDatabase db, char *rname, Appearance *value) +{ + gboolean ret = FALSE; + char *rclass = create_class_name(rname), *cname, *ctoname, *bcname; + char *rettype; + XrmValue retvalue; + + cname = g_strconcat(rname, ".color", NULL); + ctoname = g_strconcat(rname, ".colorTo", NULL); + bcname = g_strconcat(rname, ".borderColor", NULL); + + if (XrmGetResource(db, rname, rclass, &rettype, &retvalue) && + retvalue.addr != NULL) { + parse_appearance(retvalue.addr, + &value->surface.data.planar.grad, + &value->surface.data.planar.relief, + &value->surface.data.planar.bevel, + &value->surface.data.planar.interlaced, + &value->surface.data.planar.border); + if (!read_color(db, cname, &value->surface.data.planar.primary)) + value->surface.data.planar.primary = color_new(0, 0, 0); + if (!read_color(db, ctoname, &value->surface.data.planar.secondary)) + value->surface.data.planar.secondary = color_new(0, 0, 0); + if (value->surface.data.planar.border) + if (!read_color(db, bcname, + &value->surface.data.planar.border_color)) + value->surface.data.planar.border_color = color_new(0, 0, 0); + ret = TRUE; + } + + g_free(bcname); + g_free(ctoname); + g_free(cname); + g_free(rclass); + return ret; +} + +void set_default_appearance(Appearance *a) +{ + a->surface.data.planar.grad = Background_Solid; + a->surface.data.planar.relief = Flat; + a->surface.data.planar.bevel = Bevel1; + a->surface.data.planar.interlaced = FALSE; + a->surface.data.planar.border = FALSE; + a->surface.data.planar.primary = color_new(0, 0, 0); + a->surface.data.planar.secondary = color_new(0, 0, 0); +} + +gboolean load() +{ + XrmDatabase db = NULL; + + if (themerc_theme != NULL) { + db = loaddb(themerc_theme); + if (db == NULL) { + g_warning("Failed to load the theme '%s'", themerc_theme); + g_message("Falling back to the default: '%s'", DEFAULT_THEME); + } + } + if (db == NULL) { + db = loaddb(DEFAULT_THEME); + if (db == NULL) { + g_warning("Failed to load the theme '%s'.", DEFAULT_THEME); + return FALSE; + } + } + + /* XXX load the font, not from the theme file tho, its in themerc_font */ + s_font_height = 10; + + if (!read_int(db, "handleWidth", &s_handle_height) || + s_handle_height < 0 || s_handle_height > 100) s_handle_height = 6; + if (!read_int(db, "bevelWidth", &s_bevel) || + s_bevel <= 0 || s_bevel > 100) s_bevel = 3; + if (!read_int(db, "borderWidth", &s_bwidth) || + s_bwidth < 0 || s_bwidth > 100) s_bwidth = 1; + if (!read_int(db, "frameWidth", &s_cbwidth) || + s_cbwidth < 0 || s_cbwidth > 100) s_cbwidth = s_bevel; + + if (!read_color(db, "borderColor", &s_b_color)) + s_b_color = color_new(0, 0, 0); + if (!read_color(db, "window.frame.focusColor", &s_cb_focused_color)) + s_cb_focused_color = color_new(0xff, 0xff, 0xff); + if (!read_color(db, "window.frame.unfocusColor", &s_cb_unfocused_color)) + s_cb_unfocused_color = color_new(0xff, 0xff, 0xff); + + if (!read_appearance(db, "window.title.focus", a_focused_title)) + set_default_appearance(a_focused_title); + if (!read_appearance(db, "window.title.unfocus", a_unfocused_title)) + set_default_appearance(a_unfocused_title); + if (!read_appearance(db, "window.label.focus", a_focused_label)) + set_default_appearance(a_focused_label); + if (!read_appearance(db, "window.label.unfocus", a_unfocused_label)) + set_default_appearance(a_unfocused_label); + if (!read_appearance(db, "window.handle.focus", a_focused_handle)) + set_default_appearance(a_focused_handle); + if (!read_appearance(db, "window.handle.unfocus", a_unfocused_handle)) + set_default_appearance(a_unfocused_handle); + if (!read_appearance(db, "window.grip.focus", a_focused_grip)) + set_default_appearance(a_focused_grip); + if (!read_appearance(db, "window.grip.unfocus", a_unfocused_grip)) + set_default_appearance(a_unfocused_grip); + + if (!read_appearance(db, "window.button.pressed.focus", + a_focused_pressed_max)) + if (!read_appearance(db, "window.button.pressed", + a_focused_pressed_max)) + set_default_appearance(a_focused_pressed_max); + if (!read_appearance(db, "window.button.pressed.unfocus", + a_unfocused_pressed_max)) + if (!read_appearance(db, "window.button.pressed", + a_unfocused_pressed_max)) + set_default_appearance(a_unfocused_pressed_max); + if (!read_appearance(db, "window.button.focus", + a_focused_unpressed_max)) + set_default_appearance(a_focused_unpressed_max); + if (!read_appearance(db, "window.button.unfocus", + a_unfocused_unpressed_max)) + set_default_appearance(a_unfocused_unpressed_max); + + a_unfocused_unpressed_close = appearance_copy(a_unfocused_unpressed_max); + a_unfocused_pressed_close = appearance_copy(a_unfocused_pressed_max); + a_focused_unpressed_close = appearance_copy(a_focused_unpressed_max); + a_focused_pressed_close = appearance_copy(a_focused_pressed_max); + a_unfocused_unpressed_desk = appearance_copy(a_unfocused_unpressed_max); + a_unfocused_pressed_desk = appearance_copy(a_unfocused_pressed_max); + a_focused_unpressed_desk = appearance_copy(a_focused_unpressed_max); + a_focused_pressed_desk = appearance_copy(a_focused_pressed_max); + a_unfocused_unpressed_iconify = appearance_copy(a_unfocused_unpressed_max); + a_unfocused_pressed_iconify = appearance_copy(a_unfocused_pressed_max); + a_focused_unpressed_iconify = appearance_copy(a_focused_unpressed_max); + a_focused_pressed_iconify = appearance_copy(a_focused_pressed_max); + + a_icon->surface.data.planar.grad = Background_ParentRelative; + + /* XXX load the button masks */ + + XrmDestroyDatabase(db); + return TRUE; +} + + diff --git a/engines/openbox/theme.h b/engines/openbox/theme.h new file mode 100644 index 00000000..1b6011fe --- /dev/null +++ b/engines/openbox/theme.h @@ -0,0 +1,8 @@ +#ifndef __engine_theme_h +#define __engine_theme_h + +#include <glib.h> + +gboolean load(); + +#endif |
