2003-01-18 00:33:48 +00:00
|
|
|
// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 2; -*-
|
|
|
|
|
2003-02-09 23:07:54 +00:00
|
|
|
#include "config.h"
|
2003-01-18 00:33:48 +00:00
|
|
|
|
|
|
|
#include "rendercontrol.hh"
|
|
|
|
#include "truerendercontrol.hh"
|
2003-02-03 07:06:45 +00:00
|
|
|
#include "pseudorendercontrol.hh"
|
2003-01-18 04:31:16 +00:00
|
|
|
#include "rendertexture.hh"
|
2003-01-22 22:46:16 +00:00
|
|
|
#include "rendercolor.hh"
|
2003-02-08 07:33:48 +00:00
|
|
|
#include "renderstyle.hh"
|
2003-01-18 00:33:48 +00:00
|
|
|
#include "display.hh"
|
|
|
|
#include "screeninfo.hh"
|
2003-01-18 07:23:28 +00:00
|
|
|
#include "surface.hh"
|
|
|
|
#include "font.hh"
|
|
|
|
#include "ustring.hh"
|
2003-01-18 00:33:48 +00:00
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
#ifdef HAVE_STDLIB_H
|
|
|
|
# include <stdlib.h>
|
|
|
|
#endif // HAVE_STDLIB_H
|
|
|
|
|
2003-01-28 03:24:45 +00:00
|
|
|
#include "../src/gettext.h"
|
2003-01-18 00:33:48 +00:00
|
|
|
#define _(str) gettext(str)
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace otk {
|
|
|
|
|
|
|
|
RenderControl *RenderControl::getRenderControl(int screen)
|
|
|
|
{
|
|
|
|
// get the visual on the screen and return the correct type of RenderControl
|
2003-01-18 07:23:28 +00:00
|
|
|
int vclass = display->screenInfo(screen)->visual()->c_class;
|
2003-01-18 00:33:48 +00:00
|
|
|
switch (vclass) {
|
|
|
|
case TrueColor:
|
2003-01-18 07:23:28 +00:00
|
|
|
return new TrueRenderControl(screen);
|
2003-01-18 00:33:48 +00:00
|
|
|
case PseudoColor:
|
|
|
|
case StaticColor:
|
2003-02-03 07:06:45 +00:00
|
|
|
return new PseudoRenderControl(screen);
|
2003-01-18 00:33:48 +00:00
|
|
|
case GrayScale:
|
|
|
|
case StaticGray:
|
2003-02-03 07:41:25 +00:00
|
|
|
return new PseudoRenderControl(screen);
|
2003-01-18 00:33:48 +00:00
|
|
|
default:
|
|
|
|
printf(_("RenderControl: Unsupported visual %d specified. Aborting.\n"),
|
|
|
|
vclass);
|
|
|
|
::exit(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-01-18 07:23:28 +00:00
|
|
|
RenderControl::RenderControl(int screen)
|
2003-01-18 00:33:48 +00:00
|
|
|
: _screen(screen)
|
|
|
|
{
|
|
|
|
printf("Initializing RenderControl\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
RenderControl::~RenderControl()
|
|
|
|
{
|
|
|
|
printf("Destroying RenderControl\n");
|
|
|
|
}
|
|
|
|
|
2003-01-23 07:26:32 +00:00
|
|
|
void RenderControl::drawRoot(const RenderColor &color) const
|
|
|
|
{
|
|
|
|
Window root = display->screenInfo(_screen)->rootWindow();
|
|
|
|
XSetWindowBackground(**display, root, color.pixel());
|
|
|
|
XClearWindow(**display, root);
|
|
|
|
}
|
|
|
|
|
2003-01-20 06:11:12 +00:00
|
|
|
void RenderControl::drawString(Surface& sf, const Font &font, int x, int y,
|
2003-01-22 22:46:16 +00:00
|
|
|
const RenderColor &color,
|
|
|
|
const ustring &string) const
|
2003-01-18 07:23:28 +00:00
|
|
|
{
|
2003-01-20 06:11:12 +00:00
|
|
|
assert(sf._screen == _screen);
|
|
|
|
XftDraw *d = sf._xftdraw;
|
|
|
|
assert(d); // this means that the background hasn't been rendered yet!
|
2003-01-18 07:23:28 +00:00
|
|
|
|
|
|
|
if (font._shadow) {
|
|
|
|
XftColor c;
|
|
|
|
c.color.red = 0;
|
|
|
|
c.color.green = 0;
|
|
|
|
c.color.blue = 0;
|
|
|
|
c.color.alpha = font._tint | font._tint << 8; // transparent shadow
|
|
|
|
c.pixel = BlackPixel(**display, _screen);
|
|
|
|
|
|
|
|
if (string.utf8())
|
|
|
|
XftDrawStringUtf8(d, &c, font._xftfont, x + font._offset,
|
|
|
|
font._xftfont->ascent + y + font._offset,
|
|
|
|
(FcChar8*)string.c_str(), string.bytes());
|
|
|
|
else
|
|
|
|
XftDrawString8(d, &c, font._xftfont, x + font._offset,
|
|
|
|
font._xftfont->ascent + y + font._offset,
|
|
|
|
(FcChar8*)string.c_str(), string.bytes());
|
|
|
|
}
|
|
|
|
|
|
|
|
XftColor c;
|
|
|
|
c.color.red = color.red() | color.red() << 8;
|
|
|
|
c.color.green = color.green() | color.green() << 8;
|
|
|
|
c.color.blue = color.blue() | color.blue() << 8;
|
|
|
|
c.pixel = color.pixel();
|
|
|
|
c.color.alpha = 0xff | 0xff << 8; // no transparency in Color yet
|
|
|
|
|
|
|
|
if (string.utf8())
|
|
|
|
XftDrawStringUtf8(d, &c, font._xftfont, x, font._xftfont->ascent + y,
|
|
|
|
(FcChar8*)string.c_str(), string.bytes());
|
|
|
|
else
|
|
|
|
XftDrawString8(d, &c, font._xftfont, x, font._xftfont->ascent + y,
|
|
|
|
(FcChar8*)string.c_str(), string.bytes());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-01-20 20:20:06 +00:00
|
|
|
void RenderControl::drawSolidBackground(Surface& sf,
|
|
|
|
const RenderTexture& texture) const
|
|
|
|
{
|
|
|
|
assert(_screen == sf._screen);
|
|
|
|
assert(_screen == texture.color().screen());
|
|
|
|
|
|
|
|
if (texture.parentRelative()) return;
|
|
|
|
|
|
|
|
sf.setPixmap(texture.color());
|
|
|
|
|
2003-02-08 07:33:48 +00:00
|
|
|
int width = sf.size().width(), height = sf.size().height();
|
2003-01-20 20:20:06 +00:00
|
|
|
int left = 0, top = 0, right = width - 1, bottom = height - 1;
|
|
|
|
|
|
|
|
if (texture.interlaced())
|
|
|
|
for (int i = 0; i < height; i += 2)
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.interlaceColor().gc(),
|
|
|
|
0, i, width, i);
|
|
|
|
|
|
|
|
switch (texture.relief()) {
|
|
|
|
case RenderTexture::Raised:
|
|
|
|
switch (texture.bevel()) {
|
|
|
|
case RenderTexture::Bevel1:
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
left, bottom, right, bottom);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
right, bottom, right, top);
|
|
|
|
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
left, top, right, top);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
left, bottom, left, top);
|
|
|
|
break;
|
|
|
|
case RenderTexture::Bevel2:
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
left + 1, bottom - 2, right - 2, bottom - 2);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
right - 2, bottom - 2, right - 2, top + 1);
|
|
|
|
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
left + 1, top + 1, right - 2, top + 1);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
left + 1, bottom - 2, left + 1, top + 1);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(false); // unhandled RenderTexture::BevelType
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case RenderTexture::Sunken:
|
|
|
|
switch (texture.bevel()) {
|
|
|
|
case RenderTexture::Bevel1:
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
left, bottom, right, bottom);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
right, bottom, right, top);
|
|
|
|
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
left, top, right, top);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
left, bottom, left, top);
|
|
|
|
break;
|
|
|
|
case RenderTexture::Bevel2:
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
left + 1, bottom - 2, right - 2, bottom - 2);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelLightColor().gc(),
|
|
|
|
right - 2, bottom - 2, right - 2, top + 1);
|
|
|
|
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
left + 1, top + 1, right - 2, top + 1);
|
|
|
|
XDrawLine(**display, sf.pixmap(), texture.bevelDarkColor().gc(),
|
|
|
|
left + 1, bottom - 2, left + 1, top + 1);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(false); // unhandled RenderTexture::BevelType
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case RenderTexture::Flat:
|
|
|
|
if (texture.border())
|
|
|
|
XDrawRectangle(**display, sf.pixmap(), texture.borderColor().gc(),
|
|
|
|
left, top, right, bottom);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
assert(false); // unhandled RenderTexture::ReliefType
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-11 22:10:50 +00:00
|
|
|
void RenderControl::drawMask(Surface &sf, const RenderColor &color,
|
|
|
|
const PixmapMask &mask) const
|
|
|
|
{
|
|
|
|
assert(_screen == sf._screen);
|
|
|
|
assert(_screen == color.screen());
|
|
|
|
|
|
|
|
if (mask.mask == None) return; // no mask given
|
|
|
|
|
|
|
|
int width = sf.size().width(), height = sf.size().height();
|
|
|
|
|
|
|
|
// set the clip region
|
|
|
|
int x = (width - mask.w) / 2, y = (height - mask.h) / 2;
|
|
|
|
XSetClipMask(**display, color.gc(), mask.mask);
|
|
|
|
XSetClipOrigin(**display, color.gc(), x, y);
|
|
|
|
|
|
|
|
// fill in the clipped region
|
|
|
|
XFillRectangle(**display, sf.pixmap(), color.gc(), x, y,
|
|
|
|
x + mask.w, y + mask.h);
|
|
|
|
|
|
|
|
// unset the clip region
|
|
|
|
XSetClipMask(**display, color.gc(), None);
|
|
|
|
XSetClipOrigin(**display, color.gc(), 0, 0);
|
|
|
|
}
|
|
|
|
|
2003-01-18 00:33:48 +00:00
|
|
|
}
|