+ src = _title_sur->pixelData() + w * (geom.bevel + 1) + geom.icon_x;
+
+ // get the background under the icon button
+ for (int y = 0; y < geom.button_size; ++y, src += w - geom.button_size)
+ for (int x = 0; x < geom.button_size; ++x, ++dest, ++src)
+ *dest = *src;
+ // draw the icon over it
+ const Icon *icon = _client->icon(otk::Size(geom.button_size,
+ geom.button_size));
+ control->drawImage(*s, icon->w, icon->h, icon->data);
+ if (!icon->data) {
+ Pixmap p = _client->pixmapIcon(), m = _client->pixmapIconMask();
+ if (p != None)
+ control->drawImage(*s, p, m);
+ }
+
+ XSetWindowBackgroundPixmap(**otk::display, _icon, s->pixmap());
+ XClearWindow(**otk::display, _icon);
+ if (_icon_sur) delete _icon_sur;
+ _icon_sur = s;
+}
+
+void Frame::layoutTitle()
+{
+ // figure out whats being shown, and the width of the label
+ geom.label_width = geom.width - geom.bevel * 2;
+ bool n, d, i, t, m ,c;
+ n = d = i = t = m = c = false;
+ for (const char *l = _layout.c_str(); *l; ++l) {
+ switch (*l) {
+ case 'n':
+ case 'N':
+ if (!(_decorations & Client::Decor_Icon)) break;
+ n = true;
+ geom.label_width -= geom.button_size + geom.bevel;
+ break;
+ case 'd':
+ case 'D':
+ if (!(_decorations & Client::Decor_AllDesktops)) break;
+ d = true;
+ geom.label_width -= geom.button_size + geom.bevel;
+ break;
+ case 'i':
+ case 'I':
+ if (!(_decorations & Client::Decor_Iconify)) break;
+ i = true;
+ geom.label_width -= geom.button_size + geom.bevel;
+ break;
+ case 't':
+ case 'T':
+ t = true;
+ break;
+ case 'm':
+ case 'M':
+ if (!(_decorations & Client::Decor_Maximize)) break;
+ m = true;
+ geom.label_width -= geom.button_size + geom.bevel;
+ break;
+ case 'c':
+ case 'C':
+ if (!(_decorations & Client::Decor_Close)) break;
+ c = true;
+ geom.label_width -= geom.button_size + geom.bevel;
+ break;
+ }
+ }
+ if (geom.label_width < 1) geom.label_width = 1;
+
+ XResizeWindow(**otk::display, _label, geom.label_width, geom.font_height);
+
+ if (!n) {
+ _decorations &= ~Client::Decor_Icon;
+ XUnmapWindow(**otk::display, _icon);
+ }
+ if (!d) {
+ _decorations &= ~Client::Decor_AllDesktops;
+ XUnmapWindow(**otk::display, _desk);
+ }
+ if (!i) {
+ _decorations &= ~Client::Decor_Iconify;
+ XUnmapWindow(**otk::display, _iconify);
+ }
+ if (!t)
+ XUnmapWindow(**otk::display, _label);
+ if (!m) {
+ _decorations &= ~Client::Decor_Maximize;
+ XUnmapWindow(**otk::display, _max);
+ }
+ if (!c) {
+ _decorations &= ~Client::Decor_Close;
+ XUnmapWindow(**otk::display, _close);
+ }
+
+ int x = geom.bevel;
+ for (const char *lc = _layout.c_str(); *lc; ++lc) {
+ switch (*lc) {
+ case 'n':
+ case 'N':
+ if (!n) break;
+ geom.icon_x = x;
+ XMapWindow(**otk::display, _icon);
+ XMoveWindow(**otk::display, _icon, x, geom.bevel + 1);
+ x += geom.button_size + geom.bevel;
+ break;
+ case 'd':
+ case 'D':
+ if (!d) break;
+ geom.desktop_x = x;
+ XMapWindow(**otk::display, _desk);
+ XMoveWindow(**otk::display, _desk, x, geom.bevel + 1);
+ x += geom.button_size + geom.bevel;
+ break;
+ case 'i':
+ case 'I':
+ if (!i) break;
+ geom.iconify_x = x;
+ XMapWindow(**otk::display, _iconify);
+ XMoveWindow(**otk::display, _iconify, x, geom.bevel + 1);
+ x += geom.button_size + geom.bevel;
+ break;
+ case 't':
+ case 'T':
+ if (!t) break;
+ geom.title_x = x;
+ XMapWindow(**otk::display, _label);
+ XMoveWindow(**otk::display, _label, x, geom.bevel);
+ x += geom.label_width + geom.bevel;
+ break;
+ case 'm':
+ case 'M':
+ if (!m) break;
+ geom.max_x = x;
+ XMapWindow(**otk::display, _max);
+ XMoveWindow(**otk::display, _max, x, geom.bevel + 1);
+ x += geom.button_size + geom.bevel;
+ break;
+ case 'c':
+ case 'C':
+ if (!c) break;
+ geom.close_x = x;
+ XMapWindow(**otk::display, _close);
+ XMoveWindow(**otk::display, _close, x, geom.bevel + 1);
+ x += geom.button_size + geom.bevel;
+ break;
+ }
+ }
+}
+
+void Frame::adjustPosition()
+{
+ int x, y;
+ x = _client->area().x();
+ y = _client->area().y();
+ clientGravity(x, y);
+ XMoveWindow(**otk::display, _frame, x, y);
+ _area = otk::Rect(otk::Point(x, y), _area.size());
+}
+
+
+void Frame::adjustShape()
+{
+#ifdef SHAPE
+ if (!_client->shaped()) {
+ // clear the shape on the frame window
+ XShapeCombineMask(**otk::display, _frame, ShapeBounding,
+ _innersize.left,
+ _innersize.top,
+ None, ShapeSet);
+ } else {
+ // make the frame's shape match the clients
+ XShapeCombineShape(**otk::display, _frame, ShapeBounding,
+ _innersize.left,
+ _innersize.top,
+ _client->window(), ShapeBounding, ShapeSet);
+
+ int num = 0;
+ XRectangle xrect[2];
+
+ if (_decorations & Client::Decor_Titlebar) {
+ xrect[0].x = -geom.bevel;
+ xrect[0].y = -geom.bevel;
+ xrect[0].width = geom.width + geom.bwidth * 2;
+ xrect[0].height = geom.title_height() + geom.bwidth * 2;
+ ++num;
+ }
+
+ if (_decorations & Client::Decor_Handle) {
+ xrect[1].x = -geom.bevel;
+ xrect[1].y = geom.handle_y;
+ xrect[1].width = geom.width + geom.bwidth * 2;
+ xrect[1].height = geom.handle_height + geom.bwidth * 2;
+ ++num;
+ }
+
+ XShapeCombineRectangles(**otk::display, _frame,
+ ShapeBounding, 0, 0, xrect, num,
+ ShapeUnion, Unsorted);
+ }
+#endif // SHAPE
+}
+
+void Frame::adjustState()
+{
+ renderDesk();
+ renderMax();
+}