From bf57f2f26f33418350fec345b8af8aba11be7616 Mon Sep 17 00:00:00 2001 From: Vadim Lopatin Date: Fri, 14 Mar 2014 17:35:53 +0400 Subject: [PATCH] app --- 3rdparty/X11/xcb/image.d | 369 +++++++++++++++++++++++ dlanguilib.dproj | 1 + examples/example1/main.d | 7 +- src/dlangui/platforms/x11/x11app.d | 457 ++++------------------------- 4 files changed, 437 insertions(+), 397 deletions(-) create mode 100644 3rdparty/X11/xcb/image.d diff --git a/3rdparty/X11/xcb/image.d b/3rdparty/X11/xcb/image.d new file mode 100644 index 00000000..995a348e --- /dev/null +++ b/3rdparty/X11/xcb/image.d @@ -0,0 +1,369 @@ +// Manually created +module std.c.linux.X11.xcb.image; + +import std.c.linux.X11.xcb.xcb; +import std.c.linux.X11.xcb.shm; +import std.c.linux.X11.xcb.xproto; +import std.c.stdlib; +import std.conv; + +struct xcb_image_t +{ + ushort width; + ushort height; + xcb_image_format_t format; + ubyte scanline_pad; + ubyte depth; + ubyte bpp; + ubyte unit; + uint plane_mask; + xcb_image_order_t byte_order; + xcb_image_order_t bit_order; + uint stride; + uint size; + void * base; + ubyte * data; +} + +xcb_format_t * +find_format_by_depth (xcb_setup_t *setup, ubyte depth) +{ + xcb_format_t *fmt = xcb_setup_pixmap_formats(setup); + xcb_format_t *fmtend = fmt + xcb_setup_pixmap_formats_length(setup); + for(; fmt != fmtend; ++fmt) + if(fmt.depth == depth) + return fmt; + return null; +} + + +xcb_image_format_t +effective_format(xcb_image_format_t format, ubyte bpp) +{ + if (format == XCB_IMAGE_FORMAT_Z_PIXMAP && bpp != 1) + return format; + return XCB_IMAGE_FORMAT_XY_PIXMAP; +} + + +int +format_valid (ubyte depth, ubyte bpp, ubyte unit, + xcb_image_format_t format, ubyte xpad) +{ + xcb_image_format_t ef = effective_format(format, bpp); + if (depth > bpp) + return 0; + switch(ef) { + case XCB_IMAGE_FORMAT_XY_PIXMAP: + switch(unit) { + case 8: + case 16: + case 32: + break; + default: + return 0; + } + if (xpad < bpp) + return 0; + switch (xpad) { + case 8: + case 16: + case 32: + break; + default: + return 0; + } + break; + case XCB_IMAGE_FORMAT_Z_PIXMAP: + switch (bpp) { + case 4: + if (unit != 8) + return 0; + break; + case 8: + case 16: + case 24: + case 32: + if (unit != bpp) + return 0; + break; + default: + return 0; + } + break; + default: + return 0; + } + return 1; +} + + +int +image_format_valid (xcb_image_t *image) { + return format_valid(image.depth, + image.bpp, + image.unit, + image.format, + image.scanline_pad); +} + +uint xcb_roundup(uint base, + uint pad +) +{ + uint b = base + pad - 1; + /* faster if pad is a power of two */ + if (((pad - 1) & pad) == 0) + return b & -pad; + return b - b % pad; +} + +void +xcb_image_annotate (xcb_image_t *image) +{ + xcb_image_format_t ef = effective_format(image.format, image.bpp); + switch (ef) { + case XCB_IMAGE_FORMAT_XY_PIXMAP: + image.stride = xcb_roundup(image.width, image.scanline_pad) >> 3; + image.size = image.height * image.stride * image.depth; + break; + case XCB_IMAGE_FORMAT_Z_PIXMAP: + image.stride = xcb_roundup(cast(uint)image.width * + cast(uint)image.bpp, + image.scanline_pad) >> 3; + image.size = image.height * image.stride; + break; + default: + assert(0); + } +} + +xcb_image_t * +xcb_image_create_native (xcb_connection_t * c, + ushort width, + ushort height, + xcb_image_format_t format, + ubyte depth, + void * base, + uint bytes, + ubyte * data) +{ + xcb_setup_t * setup = xcb_get_setup(c); + xcb_format_t * fmt; + xcb_image_format_t ef = format; + + if (ef == XCB_IMAGE_FORMAT_Z_PIXMAP && depth == 1) + ef = XCB_IMAGE_FORMAT_XY_PIXMAP; + switch (ef) { + case XCB_IMAGE_FORMAT_XY_BITMAP: + if (depth != 1) + return null; + /* fall through */ + case XCB_IMAGE_FORMAT_XY_PIXMAP: + if (depth > 1) { + fmt = find_format_by_depth(setup, depth); + if (!fmt) + return null; + } + return xcb_image_create(width, height, format, + setup.bitmap_format_scanline_pad, + depth, depth, setup.bitmap_format_scanline_unit, + setup.image_byte_order, + setup.bitmap_format_bit_order, + base, bytes, data); + case XCB_IMAGE_FORMAT_Z_PIXMAP: + fmt = find_format_by_depth(setup, depth); + if (!fmt) + return null; + return xcb_image_create(width, height, format, + fmt.scanline_pad, + fmt.depth, fmt.bits_per_pixel, 0, + setup.image_byte_order, + XCB_IMAGE_ORDER_MSB_FIRST, + base, bytes, data); + default: + assert(0); + } + assert(0); +} + +uint xcb_mask(uint n) +{ + return n == 32 ? ~0 : (1 << n) - 1; +} + +xcb_image_t * +xcb_image_create (ushort width, + ushort height, + xcb_image_format_t format, + ubyte xpad, + ubyte depth, + ubyte bpp, + ubyte unit, + xcb_image_order_t byte_order, + xcb_image_order_t bit_order, + void * base, + uint bytes, + ubyte * data) +{ + xcb_image_t * image; + + if (unit == 0) { + switch (format) { + case XCB_IMAGE_FORMAT_XY_BITMAP: + case XCB_IMAGE_FORMAT_XY_PIXMAP: + unit = 32; + break; + case XCB_IMAGE_FORMAT_Z_PIXMAP: + if (bpp == 1) { + unit = 32; + break; + } + if (bpp < 8) { + unit = 8; + break; + } + unit = bpp; + break; + default: + break; + + } + } + if (!format_valid(depth, bpp, unit, format, xpad)) + return null; + import std.c.stdlib; + image = cast(xcb_image_t*)malloc(xcb_image_t.sizeof); + if (image is null) + return null; + image.width = width; + image.height = height; + image.format = format; + image.scanline_pad = xpad; + image.depth = depth; + image.bpp = bpp; + image.unit = unit; + image.plane_mask = xcb_mask(depth); + image.byte_order = byte_order; + image.bit_order = bit_order; + xcb_image_annotate(image); + + /* + * Ways this function can be called: + * * with data: we fail if bytes isn't + * large enough, else leave well enough alone. + * * with base and !data: if bytes is zero, we + * default; otherwise we fail if bytes isn't + * large enough, else fill in data + * * with !base and !data: we malloc storage + * for the data, save that address as the base, + * and fail if malloc does. + * + * When successful, we establish the invariant that data + * points at sufficient storage that may have been + * supplied, and base is set iff it should be + * auto-freed when the image is destroyed. + * + * Except as a special case when base = 0 && data == 0 && + * bytes == ~0 we just return the image structure and let + * the caller deal with getting the allocation right. + */ + if (!base && !data && bytes == ~0) { + image.base = null; + image.data = null; + return image; + } + if (!base && data && bytes == 0) + bytes = image.size; + image.base = base; + image.data = data; + if (!image.data) { + if (image.base) { + image.data = cast(ubyte*)image.base; + } else { + bytes = image.size; + image.base = malloc(bytes); + image.data = cast(ubyte*)image.base; + } + } + if (!image.data || bytes < image.size) { + free(image); + return null; + } + return image; +} + + +void +xcb_image_destroy (xcb_image_t *image) +{ + if (image.base) + free (image.base); + free (image); +} + +ubyte +xcb_aux_get_depth(xcb_connection_t *c, + xcb_screen_t *screen) +{ + xcb_drawable_t drawable; + xcb_get_geometry_reply_t *geom; + ubyte depth; + + drawable = screen.root; + geom = xcb_get_geometry_reply (c, xcb_get_geometry(c, drawable), null); + + if (!geom) { + //Log.e("GetGeometry(root) failed"); + exit (0); + } + + depth = geom.depth; + free (geom); + + return depth; +} + +extern (C) int xcb_image_shm_get(xcb_connection_t * conn, + xcb_drawable_t draw, + xcb_image_t * image, + xcb_shm_segment_info_t shminfo, + ushort x, + ushort y, + uint plane_mask); +const XCB_ALL_PLANES = ~0; + +extern (C) xcb_image_t * +xcb_image_shm_put (xcb_connection_t * conn, + xcb_drawable_t draw, + xcb_gcontext_t gc, + xcb_image_t * image, + xcb_shm_segment_info_t shminfo, + short src_x, + short src_y, + short dest_x, + short dest_y, + ushort src_width, + ushort src_height, + ubyte send_event); + +/** + * @struct xcb_shm_segment_info_t + * A structure that stores the informations needed by the MIT Shm + * Extension. + */ +struct xcb_shm_segment_info_t +{ + xcb_shm_seg_t shmseg; + uint shmid; + ubyte *shmaddr; +} + +alias int key_t; +extern (C) int shmget(key_t key, size_t size, int shmflg); +extern (C) int getpagesize(); +extern (C) ubyte *shmat(int shmid, ubyte *shmaddr, int shmflg); +extern (C) int shmctl(int shmid, int cmd, void *buf); +const IPC_CREAT = octal!1000; +const IPC_PRIVATE = (cast(key_t) 0); +const IPC_RMID = 0; diff --git a/dlanguilib.dproj b/dlanguilib.dproj index 940698f8..584b02d1 100644 --- a/dlanguilib.dproj +++ b/dlanguilib.dproj @@ -160,6 +160,7 @@ + diff --git a/examples/example1/main.d b/examples/example1/main.d index 2688016a..fdaf047a 100644 --- a/examples/example1/main.d +++ b/examples/example1/main.d @@ -9,6 +9,7 @@ version (linux) { pragma(lib, "xcb-shm"); pragma(lib, "xcb-image"); pragma(lib, "X11"); + pragma(lib, "dl"); } /// workaround for link issue when WinMain is located in library @@ -27,7 +28,11 @@ version(Windows) { /// entry point for dlangui based application extern (C) int UIAppMain(string[] args) { // setup resource dir - string resourceDir = exePath() ~ "..\\res\\"; + version (Windows) { + string resourceDir = exePath() ~ "..\\res\\"; + } else { + string resourceDir = exePath() ~ "../../res/"; + } string[] imageDirs = [ resourceDir ]; diff --git a/src/dlangui/platforms/x11/x11app.d b/src/dlangui/platforms/x11/x11app.d index 1c9f785f..b2618933 100644 --- a/src/dlangui/platforms/x11/x11app.d +++ b/src/dlangui/platforms/x11/x11app.d @@ -6,6 +6,7 @@ import std.string; import std.c.linux.X11.xcb.xcb; import std.c.linux.X11.xcb.shm; import std.c.linux.X11.xcb.xproto; +import std.c.linux.X11.xcb.image; import std.c.linux.X11.keysymdef; import std.c.linux.linux; import std.c.stdlib; @@ -13,367 +14,9 @@ import std.conv; import dlangui.core.logger; import dlangui.graphics.drawbuf; +import dlangui.graphics.fonts; +import dlangui.graphics.ftfonts; import dlangui.platforms.common.platform; -struct xcb_image_t -{ - ushort width; - ushort height; - xcb_image_format_t format; - ubyte scanline_pad; - ubyte depth; - ubyte bpp; - ubyte unit; - uint plane_mask; - xcb_image_order_t byte_order; - xcb_image_order_t bit_order; - uint stride; - uint size; - void * base; - ubyte * data; -} - -xcb_format_t * -find_format_by_depth (xcb_setup_t *setup, ubyte depth) -{ - xcb_format_t *fmt = xcb_setup_pixmap_formats(setup); - xcb_format_t *fmtend = fmt + xcb_setup_pixmap_formats_length(setup); - for(; fmt != fmtend; ++fmt) - if(fmt.depth == depth) - return fmt; - return null; -} - - -xcb_image_format_t -effective_format(xcb_image_format_t format, ubyte bpp) -{ - if (format == XCB_IMAGE_FORMAT_Z_PIXMAP && bpp != 1) - return format; - return XCB_IMAGE_FORMAT_XY_PIXMAP; -} - - -int -format_valid (ubyte depth, ubyte bpp, ubyte unit, - xcb_image_format_t format, ubyte xpad) -{ - xcb_image_format_t ef = effective_format(format, bpp); - if (depth > bpp) - return 0; - switch(ef) { - case XCB_IMAGE_FORMAT_XY_PIXMAP: - switch(unit) { - case 8: - case 16: - case 32: - break; - default: - return 0; - } - if (xpad < bpp) - return 0; - switch (xpad) { - case 8: - case 16: - case 32: - break; - default: - return 0; - } - break; - case XCB_IMAGE_FORMAT_Z_PIXMAP: - switch (bpp) { - case 4: - if (unit != 8) - return 0; - break; - case 8: - case 16: - case 24: - case 32: - if (unit != bpp) - return 0; - break; - default: - return 0; - } - break; - default: - return 0; - } - return 1; -} - - -int -image_format_valid (xcb_image_t *image) { - return format_valid(image.depth, - image.bpp, - image.unit, - image.format, - image.scanline_pad); -} - -uint xcb_roundup(uint base, - uint pad -) -{ - uint b = base + pad - 1; - /* faster if pad is a power of two */ - if (((pad - 1) & pad) == 0) - return b & -pad; - return b - b % pad; -} - -void -xcb_image_annotate (xcb_image_t *image) -{ - xcb_image_format_t ef = effective_format(image.format, image.bpp); - switch (ef) { - case XCB_IMAGE_FORMAT_XY_PIXMAP: - image.stride = xcb_roundup(image.width, image.scanline_pad) >> 3; - image.size = image.height * image.stride * image.depth; - break; - case XCB_IMAGE_FORMAT_Z_PIXMAP: - image.stride = xcb_roundup(cast(uint)image.width * - cast(uint)image.bpp, - image.scanline_pad) >> 3; - image.size = image.height * image.stride; - break; - default: - assert(0); - } -} - -xcb_image_t * -xcb_image_create_native (xcb_connection_t * c, - ushort width, - ushort height, - xcb_image_format_t format, - ubyte depth, - void * base, - uint bytes, - ubyte * data) -{ - xcb_setup_t * setup = xcb_get_setup(c); - xcb_format_t * fmt; - xcb_image_format_t ef = format; - - if (ef == XCB_IMAGE_FORMAT_Z_PIXMAP && depth == 1) - ef = XCB_IMAGE_FORMAT_XY_PIXMAP; - switch (ef) { - case XCB_IMAGE_FORMAT_XY_BITMAP: - if (depth != 1) - return null; - /* fall through */ - case XCB_IMAGE_FORMAT_XY_PIXMAP: - if (depth > 1) { - fmt = find_format_by_depth(setup, depth); - if (!fmt) - return null; - } - return xcb_image_create(width, height, format, - setup.bitmap_format_scanline_pad, - depth, depth, setup.bitmap_format_scanline_unit, - setup.image_byte_order, - setup.bitmap_format_bit_order, - base, bytes, data); - case XCB_IMAGE_FORMAT_Z_PIXMAP: - fmt = find_format_by_depth(setup, depth); - if (!fmt) - return null; - return xcb_image_create(width, height, format, - fmt.scanline_pad, - fmt.depth, fmt.bits_per_pixel, 0, - setup.image_byte_order, - XCB_IMAGE_ORDER_MSB_FIRST, - base, bytes, data); - default: - assert(0); - } - assert(0); -} - -uint xcb_mask(uint n) -{ - return n == 32 ? ~0 : (1 << n) - 1; -} - -xcb_image_t * -xcb_image_create (ushort width, - ushort height, - xcb_image_format_t format, - ubyte xpad, - ubyte depth, - ubyte bpp, - ubyte unit, - xcb_image_order_t byte_order, - xcb_image_order_t bit_order, - void * base, - uint bytes, - ubyte * data) -{ - xcb_image_t * image; - - if (unit == 0) { - switch (format) { - case XCB_IMAGE_FORMAT_XY_BITMAP: - case XCB_IMAGE_FORMAT_XY_PIXMAP: - unit = 32; - break; - case XCB_IMAGE_FORMAT_Z_PIXMAP: - if (bpp == 1) { - unit = 32; - break; - } - if (bpp < 8) { - unit = 8; - break; - } - unit = bpp; - break; - default: - break; - - } - } - if (!format_valid(depth, bpp, unit, format, xpad)) - return null; - import std.c.stdlib; - image = cast(xcb_image_t*)malloc(xcb_image_t.sizeof); - if (image is null) - return null; - image.width = width; - image.height = height; - image.format = format; - image.scanline_pad = xpad; - image.depth = depth; - image.bpp = bpp; - image.unit = unit; - image.plane_mask = xcb_mask(depth); - image.byte_order = byte_order; - image.bit_order = bit_order; - xcb_image_annotate(image); - - /* - * Ways this function can be called: - * * with data: we fail if bytes isn't - * large enough, else leave well enough alone. - * * with base and !data: if bytes is zero, we - * default; otherwise we fail if bytes isn't - * large enough, else fill in data - * * with !base and !data: we malloc storage - * for the data, save that address as the base, - * and fail if malloc does. - * - * When successful, we establish the invariant that data - * points at sufficient storage that may have been - * supplied, and base is set iff it should be - * auto-freed when the image is destroyed. - * - * Except as a special case when base = 0 && data == 0 && - * bytes == ~0 we just return the image structure and let - * the caller deal with getting the allocation right. - */ - if (!base && !data && bytes == ~0) { - image.base = null; - image.data = null; - return image; - } - if (!base && data && bytes == 0) - bytes = image.size; - image.base = base; - image.data = data; - if (!image.data) { - if (image.base) { - image.data = cast(ubyte*)image.base; - } else { - bytes = image.size; - image.base = malloc(bytes); - image.data = cast(ubyte*)image.base; - } - } - if (!image.data || bytes < image.size) { - free(image); - return null; - } - return image; -} - - -void -xcb_image_destroy (xcb_image_t *image) -{ - if (image.base) - free (image.base); - free (image); -} - -ubyte -xcb_aux_get_depth(xcb_connection_t *c, - xcb_screen_t *screen) -{ - xcb_drawable_t drawable; - xcb_get_geometry_reply_t *geom; - ubyte depth; - - drawable = screen.root; - geom = xcb_get_geometry_reply (c, xcb_get_geometry(c, drawable), null); - - if (!geom) { - Log.e("GetGeometry(root) failed"); - exit (0); - } - - depth = geom.depth; - free (geom); - - return depth; -} - -extern (C) int xcb_image_shm_get(xcb_connection_t * conn, - xcb_drawable_t draw, - xcb_image_t * image, - xcb_shm_segment_info_t shminfo, - ushort x, - ushort y, - uint plane_mask); -const XCB_ALL_PLANES = ~0; - -extern (C) xcb_image_t * -xcb_image_shm_put (xcb_connection_t * conn, - xcb_drawable_t draw, - xcb_gcontext_t gc, - xcb_image_t * image, - xcb_shm_segment_info_t shminfo, - short src_x, - short src_y, - short dest_x, - short dest_y, - ushort src_width, - ushort src_height, - ubyte send_event); - -/** - * @struct xcb_shm_segment_info_t - * A structure that stores the informations needed by the MIT Shm - * Extension. - */ -struct xcb_shm_segment_info_t -{ - xcb_shm_seg_t shmseg; - uint shmid; - ubyte *shmaddr; -} - -alias int key_t; -extern (C) int shmget(key_t key, size_t size, int shmflg); -extern (C) int getpagesize(); -extern (C) ubyte *shmat(int shmid, ubyte *shmaddr, int shmflg); -extern (C) int shmctl(int shmid, int cmd, void *buf); -const IPC_CREAT = octal!1000; -const IPC_PRIVATE = (cast(key_t) 0); -const IPC_RMID = 0; class XCBWindow : Window { xcb_window_t _w; @@ -467,41 +110,41 @@ class XCBWindow : Window { } } - void draw(ColorDrawBuf buf) { - int i; - i = xcb_image_shm_get(_xcbconnection, _w, - _image, shminfo, - 0, 0, - XCB_ALL_PLANES); - if (!i) { - Log.e("cannot get shm image"); - return; - } - Rect rc; - rc.right = buf.width; - rc.bottom = buf.height; - switch ( _image.bpp ) { - case 32: - { - for (int y = rc.top; y