mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-15 14:16:07 -06:00
481 lines
13 KiB
C++
481 lines
13 KiB
C++
#include "CtrlCore.h"
|
|
|
|
#ifdef GUI_X11
|
|
|
|
namespace Upp {
|
|
|
|
#define LLOG(x) // DLOG(x)
|
|
|
|
static void sInitXImage(XImage& ximg, Size sz)
|
|
{
|
|
Zero(ximg);
|
|
ximg.width = sz.cx;
|
|
ximg.height = sz.cy;
|
|
ximg.xoffset = 0;
|
|
ximg.format = ZPixmap;
|
|
ximg.bitmap_bit_order = MSBFirst;
|
|
#ifdef CPU_LITTLE_ENDIAN
|
|
ximg.byte_order = LSBFirst;
|
|
#else
|
|
ximg.byte_order = MSBFirst;
|
|
#endif
|
|
}
|
|
|
|
void SetSurface(SystemDraw& w, const Rect& dest, const RGBA *pixels, Size srcsz, Point srcoff)
|
|
{
|
|
GuiLock __;
|
|
XImage ximg;
|
|
sInitXImage(ximg, srcsz);
|
|
ximg.bitmap_pad = 32;
|
|
ximg.bytes_per_line = sizeof(RGBA) * srcsz.cx;
|
|
ximg.bits_per_pixel = 32;
|
|
ximg.blue_mask = 0x00ff0000;
|
|
ximg.green_mask = 0x0000ff00;
|
|
ximg.red_mask = 0x000000ff;
|
|
ximg.bitmap_unit = 32;
|
|
ximg.depth = 24;
|
|
ximg.data = (char *)pixels;
|
|
XInitImage(&ximg);
|
|
Drawable dw = w.GetDrawable();
|
|
GC gc = XCreateGC(Xdisplay, dw, 0, 0);
|
|
Point p = dest.TopLeft() + w.GetOffset();
|
|
XPutImage(Xdisplay, dw, gc, &ximg, srcoff.x, srcoff.y,
|
|
p.x, p.y, dest.GetWidth(), dest.GetHeight());
|
|
XFreeGC(Xdisplay, gc);
|
|
}
|
|
|
|
void SetSurface(SystemDraw& w, int x, int y, int cx, int cy, const RGBA *pixels)
|
|
{
|
|
GuiLock __;
|
|
Pixmap pixmap = XCreatePixmap(Xdisplay, Xroot, cx, cy, 24);
|
|
XPicture picture = XRenderCreatePicture(
|
|
Xdisplay, pixmap,
|
|
XRenderFindStandardFormat(Xdisplay, PictStandardRGB24),
|
|
0, 0
|
|
);
|
|
XImage ximg;
|
|
sInitXImage(ximg, Size(cx, cy));
|
|
ximg.bitmap_pad = 32;
|
|
ximg.bytes_per_line = 4 * cx;
|
|
ximg.bits_per_pixel = 32;
|
|
ximg.blue_mask = 0x00ff0000;
|
|
ximg.green_mask = 0x0000ff00;
|
|
ximg.red_mask = 0x000000ff;
|
|
ximg.bitmap_unit = 32;
|
|
ximg.depth = 24;
|
|
ximg.data = (char *)pixels;
|
|
XInitImage(&ximg);
|
|
GC gc = XCreateGC(Xdisplay, pixmap, 0, 0);
|
|
XPutImage(Xdisplay, pixmap, gc, &ximg, 0, 0, 0, 0, cx, cy);
|
|
XFreeGC(Xdisplay, gc);
|
|
XFreePixmap(Xdisplay, pixmap);
|
|
XRenderComposite(Xdisplay, PictOpOver,
|
|
picture, 0, XftDrawPicture(w.GetXftDraw()),
|
|
0, 0, 0, 0, x, y, cx, cy);
|
|
XRenderFreePicture(Xdisplay, picture);
|
|
}
|
|
|
|
struct ImageSysData {
|
|
Image img;
|
|
XPicture picture;
|
|
XPicture picture8;
|
|
int paintcount;
|
|
|
|
void Init(const Image& m);
|
|
void Paint(SystemDraw& w, int x, int y, const Rect& src, Color c);
|
|
~ImageSysData();
|
|
};
|
|
|
|
void ImageSysData::Init(const Image& m)
|
|
{
|
|
ASSERT(~m);
|
|
img = m;
|
|
picture = 0;
|
|
picture8 = 0;
|
|
paintcount = 0;
|
|
}
|
|
|
|
ImageSysData::~ImageSysData()
|
|
{
|
|
SysImageReleased(img);
|
|
if(Xdisplay) {
|
|
if(picture)
|
|
XRenderFreePicture(Xdisplay, picture);
|
|
if(picture8)
|
|
XRenderFreePicture(Xdisplay, picture8);
|
|
}
|
|
picture = 0;
|
|
picture8 = 0;
|
|
}
|
|
|
|
struct XRSolidFill {
|
|
Color color;
|
|
XPicture picture;
|
|
Pixmap pixmap;
|
|
};
|
|
|
|
enum { XRSolidFillCount = 67 };
|
|
|
|
static XRSolidFill sFill[XRSolidFillCount];
|
|
|
|
inline int s255d16(int x)
|
|
{
|
|
return (x << 8)|x;
|
|
}
|
|
|
|
static XPicture sGetSolidFill(Color c)
|
|
{
|
|
int q = FoldHash(GetHashValue(c)) % (int)XRSolidFillCount;
|
|
XRSolidFill& f = sFill[q];
|
|
if(f.color == c && f.picture)
|
|
return f.picture;
|
|
if(f.picture)
|
|
XRenderFreePicture(Xdisplay, f.picture);
|
|
if(f.pixmap)
|
|
XFreePixmap(Xdisplay, f.pixmap);
|
|
f.pixmap = XCreatePixmap(Xdisplay, Xroot, 1, 1, 32);
|
|
XRenderPictureAttributes attr;
|
|
attr.repeat = XTrue;
|
|
f.picture = XRenderCreatePicture(Xdisplay, f.pixmap,
|
|
XRenderFindStandardFormat(Xdisplay, PictStandardARGB32),
|
|
CPRepeat, &attr);
|
|
f.color = c;
|
|
XRenderColor xc;
|
|
xc.red = s255d16(c.GetR());
|
|
xc.green = s255d16(c.GetG());
|
|
xc.blue = s255d16(c.GetB());
|
|
xc.alpha = 65535;
|
|
XRenderFillRectangle(Xdisplay, PictOpSrc, f.picture, &xc, 0, 0, 1, 1);
|
|
return f.picture;
|
|
}
|
|
|
|
void ImageSysData::Paint(SystemDraw& w, int x, int y, const Rect& src, Color c)
|
|
{
|
|
GuiLock __;
|
|
x += w.GetOffset().x;
|
|
y += w.GetOffset().y;
|
|
Size sz = img.GetSize();
|
|
int len = sz.cx * sz.cy;
|
|
Rect sr = src & sz;
|
|
Size ssz = sr.Size();
|
|
if(sr.IsEmpty())
|
|
return;
|
|
int kind = img.GetKind();
|
|
if(kind == IMAGE_EMPTY)
|
|
return;
|
|
if(kind == IMAGE_OPAQUE && !IsNull(c)) {
|
|
w.DrawRect(x, y, sz.cx, sz.cy, c);
|
|
return;
|
|
}
|
|
if(kind == IMAGE_OPAQUE && paintcount == 0 && sr == Rect(sz)) {
|
|
SetSurface(w, x, y, sz.cx, sz.cy, ~img);
|
|
paintcount++;
|
|
return;
|
|
}
|
|
if(IsNull(c)) {
|
|
if(!picture) {
|
|
bool opaque = kind == IMAGE_OPAQUE;
|
|
Pixmap pixmap = XCreatePixmap(Xdisplay, Xroot, sz.cx, sz.cy, opaque ? 24 : 32);
|
|
picture = XRenderCreatePicture(
|
|
Xdisplay, pixmap,
|
|
XRenderFindStandardFormat(Xdisplay, opaque ? PictStandardRGB24
|
|
: PictStandardARGB32),
|
|
0, 0
|
|
);
|
|
XImage ximg;
|
|
sInitXImage(ximg, sz);
|
|
ximg.bitmap_pad = 32;
|
|
ximg.bytes_per_line = 4 * sz.cx;
|
|
ximg.bits_per_pixel = 32;
|
|
ximg.blue_mask = 0x00ff0000;
|
|
ximg.green_mask = 0x0000ff00;
|
|
ximg.red_mask = 0x000000ff;
|
|
ximg.bitmap_unit = 32;
|
|
ximg.data = (char *)~img;
|
|
ximg.depth = opaque ? 24 : 32;
|
|
XInitImage(&ximg);
|
|
GC gc = XCreateGC(Xdisplay, pixmap, 0, 0);
|
|
XPutImage(Xdisplay, pixmap, gc, &ximg, 0, 0, 0, 0, sz.cx, sz.cy);
|
|
XFreeGC(Xdisplay, gc);
|
|
XFreePixmap(Xdisplay, pixmap);
|
|
SysImageRealized(img);
|
|
}
|
|
XRenderComposite(Xdisplay, PictOpOver,
|
|
picture, 0, XftDrawPicture(w.GetXftDraw()),
|
|
sr.left, sr.top, 0, 0, x, y, ssz.cx, ssz.cy);
|
|
}
|
|
else {
|
|
if(!picture8) {
|
|
Pixmap pixmap = XCreatePixmap(Xdisplay, Xroot, sz.cx, sz.cy, 8);
|
|
picture8 = XRenderCreatePicture(Xdisplay, pixmap,
|
|
XRenderFindStandardFormat(Xdisplay, PictStandardA8),
|
|
0, 0);
|
|
Buffer<byte> ab(len);
|
|
byte *t = ab;
|
|
const RGBA *s = ~img;
|
|
const RGBA *e = s + len;
|
|
while(s < e)
|
|
*t++ = (s++)->a;
|
|
XImage ximg;
|
|
sInitXImage(ximg, sz);
|
|
ximg.data = (char *)~ab;
|
|
ximg.bitmap_unit = 8;
|
|
ximg.bitmap_pad = 8;
|
|
ximg.depth = 8;
|
|
ximg.bytes_per_line = sz.cx;
|
|
ximg.bits_per_pixel = 8;
|
|
XInitImage(&ximg);
|
|
GC gc = XCreateGC(Xdisplay, pixmap, 0, 0);
|
|
XPutImage(Xdisplay, pixmap, gc, &ximg, 0, 0, 0, 0, sz.cx, sz.cy);
|
|
XFreeGC(Xdisplay, gc);
|
|
XFreePixmap(Xdisplay, pixmap);
|
|
}
|
|
XRenderComposite(Xdisplay, PictOpOver,
|
|
sGetSolidFill(c), picture8, XftDrawPicture(w.GetXftDraw()),
|
|
sr.left, sr.top, 0, 0, x, y, ssz.cx, ssz.cy);
|
|
}
|
|
}
|
|
|
|
struct ImageSysDataMaker : LRUCache<ImageSysData, int64>::Maker {
|
|
Image img;
|
|
|
|
virtual int64 Key() const { return img.GetSerialId(); }
|
|
virtual int Make(ImageSysData& object) const { object.Init(img); return img.GetLength(); }
|
|
};
|
|
|
|
void SystemDraw::SysDrawImageOp(int x, int y, const Image& img, const Rect& src, Color color)
|
|
{
|
|
GuiLock __;
|
|
if(img.GetLength() == 0)
|
|
return;
|
|
LLOG("SysDrawImageOp " << img.GetSerialId() << ' ' << img.GetSize());
|
|
ImageSysDataMaker m;
|
|
static LRUCache<ImageSysData, int64> cache;
|
|
static int Rsz;
|
|
Rsz += img.GetLength();
|
|
if(Rsz > 200 * 200) { // we do not want to do this for each small image painted...
|
|
Rsz = 0;
|
|
cache.Remove([](const ImageSysData& object) { return object.img.GetRefCount() == 1; });
|
|
}
|
|
LLOG("SysImage cache pixels " << cache.GetSize() << ", count " << cache.GetCount());
|
|
m.img = img;
|
|
cache.Get(m).Paint(*this, x, y, src, color);
|
|
Size sz = Ctrl::GetPrimaryScreenArea().GetSize();
|
|
cache.Shrink(4 * sz.cx * sz.cy, 1000); // Cache must be after Paint because of PaintOnly!
|
|
}
|
|
|
|
void ImageDraw::Init()
|
|
{
|
|
GuiLock __;
|
|
dw = XCreatePixmap(Xdisplay, Xroot, max(size.cx, 1), max(size.cy, 1), Xdepth);
|
|
gc = XCreateGC(Xdisplay, dw, 0, 0);
|
|
xftdraw = XftDrawCreate(Xdisplay, (Drawable) dw, DefaultVisual(Xdisplay, Xscreenno), Xcolormap);
|
|
|
|
alpha.dw = XCreatePixmap(Xdisplay, Xroot, max(size.cx, 1), max(size.cy, 1), Xdepth);
|
|
alpha.gc = XCreateGC(Xdisplay, alpha.dw, 0, 0);
|
|
alpha.xftdraw = XftDrawCreate(Xdisplay, (Drawable) alpha.dw, DefaultVisual(Xdisplay, Xscreenno), Xcolormap);
|
|
|
|
Vector<Rect> clip;
|
|
clip.Add(RectC(0, 0, size.cx, size.cy));
|
|
SystemDraw::Init(clip, Point(0, 0));
|
|
alpha.Init(clip, Point(0, 0));
|
|
|
|
has_alpha = false;
|
|
}
|
|
|
|
ImageDraw::operator Image() const
|
|
{
|
|
GuiLock __;
|
|
XImage *xim = XGetImage(Xdisplay, dw, 0, 0, max(size.cx, 1), max(size.cy, 1), AllPlanes, ZPixmap);
|
|
if(!xim)
|
|
return Null;
|
|
Visual *v = DefaultVisual(Xdisplay, Xscreenno);
|
|
RasterFormat fmt;
|
|
|
|
RGBA palette[256];
|
|
|
|
switch(xim->depth) {
|
|
case 15:
|
|
case 16:
|
|
if(xim->byte_order == LSBFirst)
|
|
fmt.Set16le(v->red_mask, v->green_mask, v->blue_mask);
|
|
else
|
|
fmt.Set16be(v->red_mask, v->green_mask, v->blue_mask);
|
|
break;
|
|
case 8: {
|
|
int n = min(v->map_entries, 256);
|
|
XColor colors[256];
|
|
for(int i = 0; i < 256; i++) {
|
|
colors[i].pixel = i;
|
|
colors[i].flags = DoRed|DoGreen|DoBlue;
|
|
}
|
|
XQueryColors(Xdisplay, Xcolormap, colors, n);
|
|
XColor *s = colors;
|
|
XColor *e = s + n;
|
|
while(s < e) {
|
|
RGBA& t = palette[s->pixel];
|
|
t.r = s->red >> 8;
|
|
t.g = s->green >> 8;
|
|
t.b = s->blue >> 8;
|
|
t.a = 255;
|
|
s++;
|
|
}
|
|
fmt.Set8();
|
|
break;
|
|
}
|
|
default:
|
|
if(xim->bits_per_pixel == 32)
|
|
if(xim->byte_order == LSBFirst)
|
|
fmt.Set32le(v->red_mask, v->green_mask, v->blue_mask);
|
|
else
|
|
fmt.Set32be(v->red_mask, v->green_mask, v->blue_mask);
|
|
else
|
|
if(xim->byte_order == LSBFirst)
|
|
fmt.Set24le(v->red_mask, v->green_mask, v->blue_mask);
|
|
else
|
|
fmt.Set24be(v->red_mask, v->green_mask, v->blue_mask);
|
|
break;
|
|
}
|
|
|
|
ImageBuffer ib(size);
|
|
const byte *s = (const byte *)xim->data;
|
|
RGBA *t = ib;
|
|
for(int y = 0; y < size.cy; y++) {
|
|
fmt.Read(t, s, size.cx, palette);
|
|
s += xim->bytes_per_line;
|
|
t += size.cx;
|
|
}
|
|
XDestroyImage(xim);
|
|
if(has_alpha) {
|
|
xim = XGetImage(Xdisplay, alpha.dw, 0, 0, max(size.cx, 1), max(size.cy, 1), AllPlanes, ZPixmap);
|
|
if(xim) {
|
|
const byte *s = (const byte *)xim->data;
|
|
t = ib;
|
|
Buffer<RGBA> line(size.cx);
|
|
for(int y = 0; y < size.cy; y++) {
|
|
fmt.Read(line, s, size.cx, palette);
|
|
for(int x = 0; x < size.cx; x++)
|
|
(t++)->a = line[x].r;
|
|
s += xim->bytes_per_line;
|
|
}
|
|
XDestroyImage(xim);
|
|
}
|
|
}
|
|
Premultiply(ib);
|
|
return Image(ib);
|
|
}
|
|
|
|
ImageDraw::ImageDraw(Size sz)
|
|
{
|
|
size = sz;
|
|
Init();
|
|
}
|
|
|
|
ImageDraw::ImageDraw(int cx, int cy)
|
|
{
|
|
size = Size(cx, cy);
|
|
Init();
|
|
}
|
|
|
|
ImageDraw::~ImageDraw()
|
|
{
|
|
GuiLock __;
|
|
XftDrawDestroy(xftdraw);
|
|
XFreePixmap(Xdisplay, dw);
|
|
XFreeGC(Xdisplay, gc);
|
|
XftDrawDestroy(alpha.xftdraw);
|
|
XFreePixmap(Xdisplay, alpha.dw);
|
|
XFreeGC(Xdisplay, alpha.gc);
|
|
}
|
|
|
|
Draw& ImageDraw::Alpha()
|
|
{
|
|
if(!has_alpha) {
|
|
alpha.DrawRect(size, GrayColor(0));
|
|
has_alpha = true;
|
|
}
|
|
return alpha;
|
|
}
|
|
|
|
Image SystemDraw::X11Cursor(int c)
|
|
{
|
|
ImageBuffer b(32, 32);
|
|
Image m(b);
|
|
m.SetAuxData(c + 1);
|
|
return m;
|
|
}
|
|
|
|
Image X11Cursor(int c)
|
|
{
|
|
return SystemDraw::X11Cursor(c);
|
|
}
|
|
|
|
#include <X11/cursorfont.h>
|
|
|
|
#define FCURSOR_(x) { Image h; INTERLOCKED { static Image m = X11Cursor(x); h = m; } return h; }
|
|
|
|
Image Image::Arrow() FCURSOR_(XC_left_ptr)
|
|
Image Image::Wait() FCURSOR_(XC_watch)
|
|
Image Image::IBeam() FCURSOR_(XC_xterm)
|
|
Image Image::No() FCURSOR_(XC_circle)
|
|
Image Image::SizeAll() FCURSOR_(XC_fleur)
|
|
Image Image::SizeHorz() FCURSOR_(XC_sb_h_double_arrow)
|
|
Image Image::SizeVert() FCURSOR_(XC_sb_v_double_arrow)
|
|
Image Image::SizeTopLeft() FCURSOR_(XC_top_left_corner)
|
|
Image Image::SizeTop() FCURSOR_(XC_top_side)
|
|
Image Image::SizeTopRight() FCURSOR_(XC_top_right_corner)
|
|
Image Image::SizeLeft() FCURSOR_(XC_left_side)
|
|
Image Image::SizeRight() FCURSOR_(XC_right_side)
|
|
Image Image::SizeBottomLeft() FCURSOR_(XC_bottom_left_corner)
|
|
Image Image::SizeBottom() FCURSOR_(XC_bottom_side)
|
|
Image Image::SizeBottomRight() FCURSOR_(XC_bottom_right_corner)
|
|
Image Image::Cross() FCURSOR_(XC_crosshair)
|
|
Image Image::Hand() FCURSOR_(XC_hand2)
|
|
|
|
void *CursorX11(const Image& img)
|
|
{
|
|
return SystemDraw::CursorX11(img);
|
|
}
|
|
|
|
void *SystemDraw::CursorX11(const Image& img)
|
|
{
|
|
GuiLock __;
|
|
int q = (int64)img.GetAuxData() - 1;
|
|
if(q >= 0)
|
|
return (void *)XCreateFontCursor(Xdisplay, q);
|
|
int len = img.GetLength();
|
|
Size sz = img.GetSize();
|
|
Pixmap pixmap = XCreatePixmap(Xdisplay, Xroot, sz.cx, sz.cy, 32);
|
|
XPicture picture = XRenderCreatePicture(
|
|
Xdisplay, pixmap,
|
|
XRenderFindStandardFormat(Xdisplay, PictStandardARGB32),
|
|
0, 0
|
|
);
|
|
XImage ximg;
|
|
sInitXImage(ximg, sz);
|
|
ximg.bitmap_pad = 32;
|
|
ximg.bytes_per_line = 4 * sz.cx;
|
|
ximg.bits_per_pixel = 32;
|
|
ximg.blue_mask = 0x00ff0000;
|
|
ximg.green_mask = 0x0000ff00;
|
|
ximg.red_mask = 0x000000ff;
|
|
Buffer<RGBA> pma;
|
|
pma.Alloc(len);
|
|
memcpy(pma, ~img, len * sizeof(RGBA));
|
|
ximg.bitmap_unit = 32;
|
|
ximg.depth = 32;
|
|
ximg.data = (char *)~pma;
|
|
XInitImage(&ximg);
|
|
GC gc = XCreateGC(Xdisplay, pixmap, 0, 0);
|
|
XPutImage(Xdisplay, pixmap, gc, &ximg, 0, 0, 0, 0, sz.cx, sz.cy);
|
|
XFreeGC(Xdisplay, gc);
|
|
XFreePixmap(Xdisplay, pixmap);
|
|
Point p = img.GetHotSpot();
|
|
Cursor c = XRenderCreateCursor(Xdisplay, picture, p.x, p.y);
|
|
XRenderFreePicture(Xdisplay, picture);
|
|
return (void *)c;
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|