From 435afc43e431fbfb15bc231a2c7ebc53195eebeb Mon Sep 17 00:00:00 2001 From: Vadim Lopatin Date: Thu, 20 Mar 2014 13:27:11 +0400 Subject: [PATCH] GLX support fixes --- 3rdparty/X11/Xlib.d | 2 + src/dlangui/graphics/drawbuf.d | 12 +- src/dlangui/graphics/ftfonts.d | 6 +- src/dlangui/graphics/gldrawbuf.d | 1652 ++++++++++++++-------------- src/dlangui/graphics/images.d | 6 +- src/dlangui/platforms/x11/x11app.d | 170 ++- 6 files changed, 980 insertions(+), 868 deletions(-) diff --git a/3rdparty/X11/Xlib.d b/3rdparty/X11/Xlib.d index dd4ec8d4..3f969caa 100644 --- a/3rdparty/X11/Xlib.d +++ b/3rdparty/X11/Xlib.d @@ -1913,6 +1913,8 @@ extern Status XAllocColorPlanes( ulong* /* bmask_return */ ); + const AllocNone = 0; + extern Status XAllocNamedColor( Display* /* display */, Colormap /* colormap */, diff --git a/src/dlangui/graphics/drawbuf.d b/src/dlangui/graphics/drawbuf.d index 82f6790a..b886c52a 100644 --- a/src/dlangui/graphics/drawbuf.d +++ b/src/dlangui/graphics/drawbuf.d @@ -670,12 +670,12 @@ class ColorDrawBuf : ColorDrawBufBase { } class Drawable : RefCountedObject { - private static int _instanceCount; + //private static int _instanceCount; this() { - Log.d("Created drawable, count=", ++_instanceCount); + //Log.d("Created drawable, count=", ++_instanceCount); } ~this() { - Log.d("Destroyed drawable, count=", --_instanceCount); + //Log.d("Destroyed drawable, count=", --_instanceCount); } abstract void drawTo(DrawBuf buf, Rect rc, int tilex0 = 0, int tiley0 = 0); @property abstract int width(); @@ -699,17 +699,17 @@ class SolidFillDrawable : Drawable { class ImageDrawable : Drawable { protected DrawBufRef _image; protected bool _tiled; - private int _instanceCount; + //private int _instanceCount; this(ref DrawBufRef image, bool tiled = false, bool ninePatch = false) { _image = image; _tiled = tiled; if (ninePatch) _image.detectNinePatch(); - Log.d("Created ImageDrawable, count=", ++_instanceCount); + //Log.d("Created ImageDrawable, count=", ++_instanceCount); } ~this() { _image.clear(); - Log.d("Destroyed ImageDrawable, count=", --_instanceCount); + //Log.d("Destroyed ImageDrawable, count=", --_instanceCount); } @property override int width() { if (_image.isNull) diff --git a/src/dlangui/graphics/ftfonts.d b/src/dlangui/graphics/ftfonts.d index 54ebc461..2e8b1086 100644 --- a/src/dlangui/graphics/ftfonts.d +++ b/src/dlangui/graphics/ftfonts.d @@ -99,7 +99,7 @@ private class FreeTypeFontFile { @property int weight() { return _weight; } @property bool italic() { return _italic; } - private static int _instanceCount; + //private static int _instanceCount; this(FT_Library library, string filename) { _library = library; _filename = filename; @@ -107,12 +107,12 @@ private class FreeTypeFontFile { _matrix.yy = 0x10000; _matrix.xy = 0; _matrix.yx = 0; - Log.d("Created FreeTypeFontFile, count=", ++_instanceCount); + //Log.d("Created FreeTypeFontFile, count=", ++_instanceCount); } ~this() { clear(); - Log.d("Destroyed FreeTypeFontFile, count=", --_instanceCount); + //Log.d("Destroyed FreeTypeFontFile, count=", --_instanceCount); } private static string familyName(FT_Face face) diff --git a/src/dlangui/graphics/gldrawbuf.d b/src/dlangui/graphics/gldrawbuf.d index 199f324b..2a5a2a31 100644 --- a/src/dlangui/graphics/gldrawbuf.d +++ b/src/dlangui/graphics/gldrawbuf.d @@ -1,827 +1,827 @@ -module dlangui.graphics.gldrawbuf; - -version (USE_OPENGL) { - -import dlangui.graphics.drawbuf; -import dlangui.core.logger; -private import dlangui.graphics.glsupport; -private import std.algorithm; - -/// drawing buffer - image container which allows to perform some drawing operations -class GLDrawBuf : DrawBuf { - // width - protected int _dx; - // height - protected int _dy; - protected bool _framebuffer; // not yet supported - protected uint _framebufferId; // not yet supported - protected Scene _scene; - - /// get current scene (exists only between beforeDrawing() and afterDrawing() calls - @property Scene scene() { return _scene; } - - this(int dx, int dy, bool framebuffer = false) { - _dx = dx; - _dy = dy; - _framebuffer = framebuffer; - } - - /// returns current width - @property override int width() { return _dx; } - /// returns current height - @property override int height() { return _dy; } - - /// reserved for hardware-accelerated drawing - begins drawing batch - override void beforeDrawing() { - if (_scene !is null) { - _scene.reset(); - } - _scene = new Scene(); - } - - /// reserved for hardware-accelerated drawing - ends drawing batch - override void afterDrawing() { - setOrthoProjection(_dx, _dy); - _scene.draw(); - flushGL(); - destroy(_scene); - _scene = null; - } - - /// resize buffer - override void resize(int width, int height) { - _dx = width; - _dy = height; - } - - /// fill the whole buffer with solid color (no clipping applied) - override void fill(uint color) { - assert(_scene !is null); - _scene.add(new SolidRectSceneItem(Rect(0, 0, _dx, _dy), color)); - } - /// fill rectangle with solid color (clipping is applied) - override void fillRect(Rect rc, uint color) { - assert(_scene !is null); - _scene.add(new SolidRectSceneItem(rc, color)); - } - /// draw 8bit alpha image - usually font glyph using specified color (clipping is applied) - override void drawGlyph(int x, int y, Glyph * glyph, uint color) { - assert(_scene !is null); - Rect dstrect = Rect(x,y, x + glyph.blackBoxX, y + glyph.blackBoxY); - Rect srcrect = Rect(0, 0, glyph.blackBoxX, glyph.blackBoxY); - Log.v("GLDrawBuf.drawGlyph dst=", dstrect, " src=", srcrect, " color=", color); - if (applyClipping(dstrect, srcrect)) { - if (!glGlyphCache.get(glyph.id)) - glGlyphCache.put(glyph); - _scene.add(new GlyphSceneItem(glyph.id, dstrect, srcrect, color, null)); - } - } - /// draw source buffer rectangle contents to destination buffer - override void drawFragment(int x, int y, DrawBuf src, Rect srcrect) { - assert(_scene !is null); - Rect dstrect = Rect(x, y, x + srcrect.width, y + srcrect.height); - //Log.v("GLDrawBuf.frawFragment dst=", dstrect, " src=", srcrect); - if (applyClipping(dstrect, srcrect)) { - if (!glImageCache.get(src.id)) - glImageCache.put(src); - _scene.add(new TextureSceneItem(src.id, dstrect, srcrect, 0xFFFFFF, 0, null, 0)); - } - } - /// draw source buffer rectangle contents to destination buffer rectangle applying rescaling - override void drawRescaled(Rect dstrect, DrawBuf src, Rect srcrect) { - assert(_scene !is null); - //Log.v("GLDrawBuf.frawRescaled dst=", dstrect, " src=", srcrect); - if (applyClipping(dstrect, srcrect)) { - if (!glImageCache.get(src.id)) - glImageCache.put(src); - _scene.add(new TextureSceneItem(src.id, dstrect, srcrect, 0xFFFFFF, 0, null, 0)); - } - } - - /// cleanup resources - override void clear() { - if (_framebuffer) { - // TODO: delete framebuffer - } - } - ~this() { clear(); } -} - -/// base class for all drawing scene items. -class SceneItem { - abstract void draw(); -} - -/// Drawing scene (operations sheduled for drawing) -class Scene { - private SceneItem[] _items; - this() { - activeSceneCount++; - } - ~this() { - activeSceneCount--; - } - /// add new scene item to scene - void add(SceneItem item) { - _items ~= item; - } - /// draws all scene items and removes them from list - void draw() { - foreach(SceneItem item; _items) - item.draw(); - reset(); - } - /// resets scene for new drawing - deletes all items - void reset() { - foreach(ref SceneItem item; _items) { - destroy(item); - item = null; - } - _items.length = 0; - } -} - -private __gshared int activeSceneCount = 0; -bool hasActiveScene() { - return activeSceneCount > 0; -} - -immutable int MIN_TEX_SIZE = 64; -immutable int MAX_TEX_SIZE = 4096; -private int nearestPOT(int n) { - for (int i = MIN_TEX_SIZE; i <= MAX_TEX_SIZE; i *= 2) { - if (n <= i) - return i; - } - return MIN_TEX_SIZE; -} - -/// object deletion listener callback function type -void onObjectDestroyedCallback(uint pobject) { - glImageCache.onCachedObjectDeleted(pobject); -} - -/// object deletion listener callback function type -void onGlyphDestroyedCallback(uint pobject) { - glGlyphCache.onCachedObjectDeleted(pobject); -} - -private __gshared GLImageCache glImageCache; - -private __gshared GLGlyphCache glGlyphCache; - -shared static this() { - glImageCache = new GLImageCache(); - glGlyphCache = new GLGlyphCache(); -} - -void LVGLClearImageCache() { - glImageCache.clear(); - glGlyphCache.clear(); -} - -/// OpenGL texture cache for ColorDrawBuf objects -private class GLImageCache { - - static class GLImageCacheItem { - private GLImageCachePage _page; - - @property GLImageCachePage page() { return _page; } - - uint _objectId; - Rect _rc; - bool _deleted; - - this(GLImageCachePage page, uint objectId) { _page = page; _objectId = objectId; } - }; - - static class GLImageCachePage { - private GLImageCache _cache; - private int _tdx; - private int _tdy; - private ColorDrawBuf _drawbuf; - private int _currentLine; - private int _nextLine; - private int _x; - private bool _closed; - private bool _needUpdateTexture; - private uint _textureId; - private int _itemCount; - - this(GLImageCache cache, int dx, int dy) { - _cache = cache; - Log.v("created image cache page ", dx, "x", dy); - _tdx = nearestPOT(dx); - _tdy = nearestPOT(dy); - _itemCount = 0; - } - - ~this() { - if (_drawbuf) { - destroy(_drawbuf); - _drawbuf = null; - } - if (_textureId != 0) { - deleteTexture(_textureId); - _textureId = 0; - } - } - - void updateTexture() { - if (_drawbuf is null) - return; // no draw buffer!!! - if (_textureId == 0) { - //CRLog::debug("updateTexture - new texture"); - _textureId = genTexture(); - if (!_textureId) - return; - } - //CRLog::debug("updateTexture - setting image %dx%d", _drawbuf.width, _drawbuf.height); - uint * pixels = _drawbuf.scanLine(0); - if (!setTextureImage(_textureId, _drawbuf.width, _drawbuf.height, cast(ubyte*)pixels)) { - deleteTexture(_textureId); - _textureId = 0; - return; - } - _needUpdateTexture = false; - if (_closed) { - destroy(_drawbuf); - _drawbuf = null; - } - } - - void convertPixelFormat(GLImageCacheItem item) { - Rect rc = item._rc; - for (int y = rc.top - 1; y <= rc.bottom; y++) { - uint * row = _drawbuf.scanLine(y); - for (int x = rc.left - 1; x <= rc.right; x++) { - uint cl = row[x]; - // invert A - cl ^= 0xFF000000; - // swap R and B - uint r = (cl & 0x00FF0000) >> 16; - uint b = (cl & 0x000000FF) << 16; - row[x] = (cl & 0xFF00FF00) | r | b; - } - } - } - - GLImageCacheItem reserveSpace(uint objectId, int width, int height) { - GLImageCacheItem cacheItem = new GLImageCacheItem(this, objectId); - if (_closed) - return null; - - // next line if necessary - if (_x + width + 2 > _tdx) { - // move to next line - _currentLine = _nextLine; - _x = 0; - } - // check if no room left for glyph height - if (_currentLine + height + 2 > _tdy) { - _closed = true; - return null; - } - cacheItem._rc = Rect(_x + 1, _currentLine + 1, _x + width + 1, _currentLine + height + 1); - if (height && width) { - if (_nextLine < _currentLine + height + 2) - _nextLine = _currentLine + height + 2; - if (!_drawbuf) { - _drawbuf = new ColorDrawBuf(_tdx, _tdy); - //_drawbuf.SetBackgroundColor(0x000000); - //_drawbuf.SetTextColor(0xFFFFFF); - _drawbuf.fill(0xFF000000); - } - _x += width + 1; - _needUpdateTexture = true; - } - _itemCount++; - return cacheItem; - } - int deleteItem(GLImageCacheItem item) { - _itemCount--; - return _itemCount; - } - GLImageCacheItem addItem(DrawBuf buf) { - GLImageCacheItem cacheItem = reserveSpace(buf.id, buf.width, buf.height); - if (cacheItem is null) - return null; - buf.onDestroyCallback = &onObjectDestroyedCallback; - _drawbuf.drawImage(cacheItem._rc.left, cacheItem._rc.top, buf); - convertPixelFormat(cacheItem); - _needUpdateTexture = true; - return cacheItem; - } - void drawItem(GLImageCacheItem item, Rect dstrc, Rect srcrc, uint color, uint options, Rect * clip, int rotationAngle) { - //CRLog::trace("drawing item at %d,%d %dx%d <= %d,%d %dx%d ", x, y, dx, dy, srcx, srcy, srcdx, srcdy); - if (_needUpdateTexture) - updateTexture(); - if (_textureId != 0) { - if (!isTexture(_textureId)) { - Log.e("Invalid texture ", _textureId); - return; - } - //rotationAngle = 0; - int rx = dstrc.middlex; - int ry = dstrc.middley; - if (rotationAngle) { - //rotationAngle = 0; - //setRotation(rx, ry, rotationAngle); - } - // convert coordinates to cached texture - srcrc.offset(item._rc.left, item._rc.top); - if (clip) { - int srcw = srcrc.width(); - int srch = srcrc.height(); - int dstw = dstrc.width(); - int dsth = dstrc.height(); - if (dstw) { - srcrc.left += clip.left * srcw / dstw; - srcrc.right -= clip.right * srcw / dstw; - } - if (dsth) { - srcrc.top += clip.top * srch / dsth; - srcrc.bottom -= clip.bottom * srch / dsth; - } - dstrc.left += clip.left; - dstrc.right -= clip.right; - dstrc.top += clip.top; - dstrc.bottom -= clip.bottom; - } - if (!dstrc.empty) - drawColorAndTextureRect(_textureId, _tdx, _tdy, srcrc, dstrc, color, srcrc.width() != dstrc.width() || srcrc.height() != dstrc.height()); - //drawColorAndTextureRect(vertices, texcoords, color, _textureId); - - if (rotationAngle) { - // unset rotation - setRotation(rx, ry, 0); - // glMatrixMode(GL_PROJECTION); - // glPopMatrix(); - // checkError("pop matrix"); - } - - } - } - void close() { - _closed = true; - if (_needUpdateTexture) - updateTexture(); - } - } - - private GLImageCacheItem[uint] _map; - private GLImageCachePage[] _pages; - private GLImageCachePage _activePage; - private int tdx; - private int tdy; - - private void removePage(GLImageCachePage page) { - if (_activePage == page) - _activePage = null; - for (int i = 0; i < _pages.length; i++) - if (_pages[i] == page) { - _pages.remove(i); - break; - } - destroy(page); - } - - private void updateTextureSize() { - if (!tdx) { - // TODO - tdx = tdy = 1024; //getMaxTextureSize(); - if (tdx > 1024) - tdx = tdy = 1024; - } - } - - this() { - } - ~this() { - clear(); - } - /// returns true if object exists in cache - bool get(uint obj) { - if (obj in _map) - return true; - return false; - } - /// put new object to cache - void put(DrawBuf img) { - updateTextureSize(); - GLImageCacheItem res = null; - if (img.width <= tdx / 3 && img.height < tdy / 3) { - // trying to reuse common page for small images - if (_activePage is null) { - _activePage = new GLImageCachePage(this, tdx, tdy); - _pages ~= _activePage; - } - res = _activePage.addItem(img); - if (!res) { - _activePage = new GLImageCachePage(this, tdx, tdy); - _pages ~= _activePage; - res = _activePage.addItem(img); - } - } else { - // use separate page for big image - GLImageCachePage page = new GLImageCachePage(this, img.width, img.height); - _pages ~= page; - res = page.addItem(img); - page.close(); - } - _map[img.id] = res; - } - /// clears cache - void clear() { - for (int i = 0; i < _pages.length; i++) { - destroy(_pages[i]); - _pages[i] = null; - } - _pages.clear(); - _map.clear(); - } - /// draw cached item - void drawItem(uint objectId, Rect dstrc, Rect srcrc, uint color, int options, Rect * clip, int rotationAngle) { - if (objectId in _map) { - GLImageCacheItem item = _map[objectId]; - item.page.drawItem(item, dstrc, srcrc, color, options, clip, rotationAngle); - } - } - /// handle cached object deletion, mark as deleted - void onCachedObjectDeleted(uint objectId) { - if (objectId in _map) { - GLImageCacheItem item = _map[objectId]; - if (hasActiveScene()) { - item._deleted = true; - } else { - int itemsLeft = item.page.deleteItem(item); - //CRLog::trace("itemsLeft = %d", itemsLeft); - if (itemsLeft <= 0) { - //CRLog::trace("removing page"); - removePage(item.page); - } - _map.remove(objectId); - delete item; - } - } - } - /// remove deleted items - remove page if contains only deleted items - void removeDeletedItems() { - uint[] list; - foreach (GLImageCacheItem item; _map) { - if (item._deleted) - list ~= item._objectId; - } - for (int i = 0 ; i < list.length; i++) { - onCachedObjectDeleted(list[i]); - } - } -}; - - - -private class TextureSceneItem : SceneItem { - private uint objectId; - //CacheableObject * img; - private Rect dstrc; - private Rect srcrc; - private uint color; - private uint options; - private Rect * clip; - private int rotationAngle; - - override void draw() { - if (glImageCache) - glImageCache.drawItem(objectId, dstrc, srcrc, color, options, clip, rotationAngle); - } - - this(uint _objectId, Rect _dstrc, Rect _srcrc, uint _color, uint _options, Rect * _clip, int _rotationAngle) - { - objectId = _objectId; - dstrc = _dstrc; - srcrc = _srcrc; - color = _color; - options = _options; - clip = _clip; - rotationAngle = _rotationAngle; - } - - ~this() { - } -}; - - -/// by some reason ALPHA texture does not work as expected -private immutable USE_RGBA_TEXTURE_FOR_GLYPHS = true; - -private class GLGlyphCache { - - static class GLGlyphCacheItem { - GLGlyphCachePage _page; - public: - @property GLGlyphCachePage page() { return _page; } - uint _objectId; - // image size - Rect _rc; - bool _deleted; - this(GLGlyphCachePage page, uint objectId) { _page = page; _objectId = objectId; } - }; - - static class GLGlyphCachePage { - private GLGlyphCache _cache; - private int _tdx; - private int _tdy; - private GrayDrawBuf _drawbuf; - private int _currentLine; - private int _nextLine; - private int _x; - private bool _closed; - private bool _needUpdateTexture; - private uint _textureId; - private int _itemCount; - - this(GLGlyphCache cache, int dx, int dy) { - _cache = cache; - Log.v("created image cache page ", dx, "x", dy); - _tdx = nearestPOT(dx); - _tdy = nearestPOT(dy); - _itemCount = 0; - } - - ~this() { - if (_drawbuf) { - destroy(_drawbuf); - _drawbuf = null; - } - if (_textureId != 0) { - deleteTexture(_textureId); - _textureId = 0; - } - } - - static if (USE_RGBA_TEXTURE_FOR_GLYPHS) { - uint[] _rgbaBuffer; - } - void updateTexture() { - if (_drawbuf is null) - return; // no draw buffer!!! - if (_textureId == 0) { - //CRLog::debug("updateTexture - new texture"); - _textureId = genTexture(); - if (!_textureId) - return; - } - //CRLog::debug("updateTexture - setting image %dx%d", _drawbuf.width, _drawbuf.height); - ubyte * pixels = _drawbuf.scanLine(0); - static if (USE_RGBA_TEXTURE_FOR_GLYPHS) { - int len = _drawbuf.width * _drawbuf.height; - _rgbaBuffer.length = len; - for (int i = 0; i < len; i++) - _rgbaBuffer[i] = ((cast(uint)pixels[i]) << 24) | 0x00FFFFFF; - if (!setTextureImage(_textureId, _drawbuf.width, _drawbuf.height, cast(ubyte*)_rgbaBuffer.ptr)) { - deleteTexture(_textureId); - _textureId = 0; - return; - } - } else { - if (!setTextureImageAlpha(_textureId, _drawbuf.width, _drawbuf.height, pixels)) { - deleteTexture(_textureId); - _textureId = 0; - return; - } - } - _needUpdateTexture = false; - if (_closed) { - destroy(_drawbuf); - _drawbuf = null; - } - } - GLGlyphCacheItem reserveSpace(uint objectId, int width, int height) { - GLGlyphCacheItem cacheItem = new GLGlyphCacheItem(this, objectId); - if (_closed) - return null; - - // next line if necessary - if (_x + width + 2 > _tdx) { - // move to next line - _currentLine = _nextLine; - _x = 0; - } - // check if no room left for glyph height - if (_currentLine + height + 2 > _tdy) { - _closed = true; - return null; - } - cacheItem._rc = Rect(_x + 1, _currentLine + 1, _x + width + 1, _currentLine + height + 1); - if (height && width) { - if (_nextLine < _currentLine + height + 2) - _nextLine = _currentLine + height + 2; - if (!_drawbuf) { - _drawbuf = new GrayDrawBuf(_tdx, _tdy); - //_drawbuf.SetBackgroundColor(0x000000); - //_drawbuf.SetTextColor(0xFFFFFF); - _drawbuf.fill(0x00000000); - } - _x += width + 1; - _needUpdateTexture = true; - } - _itemCount++; - return cacheItem; - } - int deleteItem(GLGlyphCacheItem item) { - _itemCount--; - return _itemCount; - } - GLGlyphCacheItem addItem(Glyph * glyph) { - GLGlyphCacheItem cacheItem = reserveSpace(glyph.id, glyph.blackBoxX, glyph.blackBoxY); - if (cacheItem is null) - return null; - _drawbuf.drawGlyph(cacheItem._rc.left, cacheItem._rc.top, glyph, 0xFFFFFF); - _needUpdateTexture = true; - return cacheItem; - } - void drawItem(GLGlyphCacheItem item, Rect dstrc, Rect srcrc, uint color, Rect * clip) { - //CRLog::trace("drawing item at %d,%d %dx%d <= %d,%d %dx%d ", x, y, dx, dy, srcx, srcy, srcdx, srcdy); - if (_needUpdateTexture) - updateTexture(); - if (_textureId != 0) { - if (!isTexture(_textureId)) { - Log.e("Invalid texture ", _textureId); - return; - } - // convert coordinates to cached texture - srcrc.offset(item._rc.left, item._rc.top); - if (clip) { - int srcw = srcrc.width(); - int srch = srcrc.height(); - int dstw = dstrc.width(); - int dsth = dstrc.height(); - if (dstw) { - srcrc.left += clip.left * srcw / dstw; - srcrc.right -= clip.right * srcw / dstw; - } - if (dsth) { - srcrc.top += clip.top * srch / dsth; - srcrc.bottom -= clip.bottom * srch / dsth; - } - dstrc.left += clip.left; - dstrc.right -= clip.right; - dstrc.top += clip.top; - dstrc.bottom -= clip.bottom; - } - if (!dstrc.empty) { - Log.d("drawing glyph with color ", color); - drawColorAndTextureRect(_textureId, _tdx, _tdy, srcrc, dstrc, color, false); - } - - } - } - void close() { - _closed = true; - if (_needUpdateTexture) - updateTexture(); - static if (USE_RGBA_TEXTURE_FOR_GLYPHS) { - _rgbaBuffer = null; - } - } - } - - GLGlyphCacheItem[uint] _map; - GLGlyphCachePage[] _pages; - GLGlyphCachePage _activePage; - int tdx; - int tdy; - void removePage(GLGlyphCachePage page) { - if (_activePage == page) - _activePage = null; - for (int i = 0; i < _pages.length; i++) - if (_pages[i] == page) { - _pages.remove(i); - break; - } - destroy(page); - } - private void updateTextureSize() { - if (!tdx) { - // TODO - tdx = tdy = 1024; //getMaxTextureSize(); - if (tdx > 1024) - tdx = tdy = 1024; - } - } - - this() { - } - ~this() { - clear(); - } - /// check if item is in cache - bool get(uint obj) { - if (obj in _map) - return true; - return false; - } - /// put new item to cache - void put(Glyph * glyph) { - updateTextureSize(); - GLGlyphCacheItem res = null; - if (_activePage is null) { - _activePage = new GLGlyphCachePage(this, tdx, tdy); - _pages ~= _activePage; - } - res = _activePage.addItem(glyph); - if (!res) { - _activePage = new GLGlyphCachePage(this, tdx, tdy); - _pages ~= _activePage; - res = _activePage.addItem(glyph); - } - _map[glyph.id] = res; - } - void clear() { - for (int i = 0; i < _pages.length; i++) { - destroy(_pages[i]); - _pages[i] = null; - } - _pages.clear(); - _map.clear(); - } - /// draw cached item - void drawItem(uint objectId, Rect dstrc, Rect srcrc, uint color, Rect * clip) { - GLGlyphCacheItem * item = objectId in _map; - if (item) - item.page.drawItem(*item, dstrc, srcrc, color, clip); - } - /// handle cached object deletion, mark as deleted - void onCachedObjectDeleted(uint objectId) { - if (objectId in _map) { - GLGlyphCacheItem item = _map[objectId]; - if (hasActiveScene()) { - item._deleted = true; - } else { - int itemsLeft = item.page.deleteItem(item); - //CRLog::trace("itemsLeft = %d", itemsLeft); - if (itemsLeft <= 0) { - //CRLog::trace("removing page"); - removePage(item.page); - } - _map.remove(objectId); - delete item; - } - } - } - /// remove deleted items - remove page if contains only deleted items - void removeDeletedItems() { - uint[] list; - foreach (GLGlyphCacheItem item; _map) { - if (item._deleted) - list ~= item._objectId; - } - for (int i = 0 ; i < list.length; i++) { - onCachedObjectDeleted(list[i]); - } - } -}; - - - - - - - -class SolidRectSceneItem : SceneItem { - Rect _rc; - uint _color; - this(Rect rc, uint color) { - _rc = rc; - _color = color; - } - override void draw() { - drawSolidFillRect(_rc, _color, _color, _color, _color); - } -} - -private class GlyphSceneItem : SceneItem { - uint objectId; - Rect dstrc; - Rect srcrc; - uint color; - Rect * clip; -public: - override void draw() { - if (glGlyphCache) - glGlyphCache.drawItem(objectId, dstrc, srcrc, color, clip); - } - this(uint _objectId, Rect _dstrc, Rect _srcrc, uint _color, Rect * _clip) - { - objectId = _objectId; - dstrc = _dstrc; - srcrc = _srcrc; - color = _color; - clip = _clip; - } - ~this() { - } -} - - +module dlangui.graphics.gldrawbuf; + +version (USE_OPENGL) { + +import dlangui.graphics.drawbuf; +import dlangui.core.logger; +private import dlangui.graphics.glsupport; +private import std.algorithm; + +/// drawing buffer - image container which allows to perform some drawing operations +class GLDrawBuf : DrawBuf { + // width + protected int _dx; + // height + protected int _dy; + protected bool _framebuffer; // not yet supported + protected uint _framebufferId; // not yet supported + protected Scene _scene; + + /// get current scene (exists only between beforeDrawing() and afterDrawing() calls + @property Scene scene() { return _scene; } + + this(int dx, int dy, bool framebuffer = false) { + _dx = dx; + _dy = dy; + _framebuffer = framebuffer; + } + + /// returns current width + @property override int width() { return _dx; } + /// returns current height + @property override int height() { return _dy; } + + /// reserved for hardware-accelerated drawing - begins drawing batch + override void beforeDrawing() { + if (_scene !is null) { + _scene.reset(); + } + _scene = new Scene(); + } + + /// reserved for hardware-accelerated drawing - ends drawing batch + override void afterDrawing() { + setOrthoProjection(_dx, _dy); + _scene.draw(); + flushGL(); + destroy(_scene); + _scene = null; + } + + /// resize buffer + override void resize(int width, int height) { + _dx = width; + _dy = height; + } + + /// fill the whole buffer with solid color (no clipping applied) + override void fill(uint color) { + assert(_scene !is null); + _scene.add(new SolidRectSceneItem(Rect(0, 0, _dx, _dy), color)); + } + /// fill rectangle with solid color (clipping is applied) + override void fillRect(Rect rc, uint color) { + assert(_scene !is null); + _scene.add(new SolidRectSceneItem(rc, color)); + } + /// draw 8bit alpha image - usually font glyph using specified color (clipping is applied) + override void drawGlyph(int x, int y, Glyph * glyph, uint color) { + assert(_scene !is null); + Rect dstrect = Rect(x,y, x + glyph.blackBoxX, y + glyph.blackBoxY); + Rect srcrect = Rect(0, 0, glyph.blackBoxX, glyph.blackBoxY); + //Log.v("GLDrawBuf.drawGlyph dst=", dstrect, " src=", srcrect, " color=", color); + if (applyClipping(dstrect, srcrect)) { + if (!glGlyphCache.get(glyph.id)) + glGlyphCache.put(glyph); + _scene.add(new GlyphSceneItem(glyph.id, dstrect, srcrect, color, null)); + } + } + /// draw source buffer rectangle contents to destination buffer + override void drawFragment(int x, int y, DrawBuf src, Rect srcrect) { + assert(_scene !is null); + Rect dstrect = Rect(x, y, x + srcrect.width, y + srcrect.height); + //Log.v("GLDrawBuf.frawFragment dst=", dstrect, " src=", srcrect); + if (applyClipping(dstrect, srcrect)) { + if (!glImageCache.get(src.id)) + glImageCache.put(src); + _scene.add(new TextureSceneItem(src.id, dstrect, srcrect, 0xFFFFFF, 0, null, 0)); + } + } + /// draw source buffer rectangle contents to destination buffer rectangle applying rescaling + override void drawRescaled(Rect dstrect, DrawBuf src, Rect srcrect) { + assert(_scene !is null); + //Log.v("GLDrawBuf.frawRescaled dst=", dstrect, " src=", srcrect); + if (applyClipping(dstrect, srcrect)) { + if (!glImageCache.get(src.id)) + glImageCache.put(src); + _scene.add(new TextureSceneItem(src.id, dstrect, srcrect, 0xFFFFFF, 0, null, 0)); + } + } + + /// cleanup resources + override void clear() { + if (_framebuffer) { + // TODO: delete framebuffer + } + } + ~this() { clear(); } +} + +/// base class for all drawing scene items. +class SceneItem { + abstract void draw(); +} + +/// Drawing scene (operations sheduled for drawing) +class Scene { + private SceneItem[] _items; + this() { + activeSceneCount++; + } + ~this() { + activeSceneCount--; + } + /// add new scene item to scene + void add(SceneItem item) { + _items ~= item; + } + /// draws all scene items and removes them from list + void draw() { + foreach(SceneItem item; _items) + item.draw(); + reset(); + } + /// resets scene for new drawing - deletes all items + void reset() { + foreach(ref SceneItem item; _items) { + destroy(item); + item = null; + } + _items.length = 0; + } +} + +private __gshared int activeSceneCount = 0; +bool hasActiveScene() { + return activeSceneCount > 0; +} + +immutable int MIN_TEX_SIZE = 64; +immutable int MAX_TEX_SIZE = 4096; +private int nearestPOT(int n) { + for (int i = MIN_TEX_SIZE; i <= MAX_TEX_SIZE; i *= 2) { + if (n <= i) + return i; + } + return MIN_TEX_SIZE; +} + +/// object deletion listener callback function type +void onObjectDestroyedCallback(uint pobject) { + glImageCache.onCachedObjectDeleted(pobject); +} + +/// object deletion listener callback function type +void onGlyphDestroyedCallback(uint pobject) { + glGlyphCache.onCachedObjectDeleted(pobject); +} + +private __gshared GLImageCache glImageCache; + +private __gshared GLGlyphCache glGlyphCache; + +shared static this() { + glImageCache = new GLImageCache(); + glGlyphCache = new GLGlyphCache(); +} + +void LVGLClearImageCache() { + glImageCache.clear(); + glGlyphCache.clear(); +} + +/// OpenGL texture cache for ColorDrawBuf objects +private class GLImageCache { + + static class GLImageCacheItem { + private GLImageCachePage _page; + + @property GLImageCachePage page() { return _page; } + + uint _objectId; + Rect _rc; + bool _deleted; + + this(GLImageCachePage page, uint objectId) { _page = page; _objectId = objectId; } + }; + + static class GLImageCachePage { + private GLImageCache _cache; + private int _tdx; + private int _tdy; + private ColorDrawBuf _drawbuf; + private int _currentLine; + private int _nextLine; + private int _x; + private bool _closed; + private bool _needUpdateTexture; + private uint _textureId; + private int _itemCount; + + this(GLImageCache cache, int dx, int dy) { + _cache = cache; + Log.v("created image cache page ", dx, "x", dy); + _tdx = nearestPOT(dx); + _tdy = nearestPOT(dy); + _itemCount = 0; + } + + ~this() { + if (_drawbuf) { + destroy(_drawbuf); + _drawbuf = null; + } + if (_textureId != 0) { + deleteTexture(_textureId); + _textureId = 0; + } + } + + void updateTexture() { + if (_drawbuf is null) + return; // no draw buffer!!! + if (_textureId == 0) { + //CRLog::debug("updateTexture - new texture"); + _textureId = genTexture(); + if (!_textureId) + return; + } + //CRLog::debug("updateTexture - setting image %dx%d", _drawbuf.width, _drawbuf.height); + uint * pixels = _drawbuf.scanLine(0); + if (!setTextureImage(_textureId, _drawbuf.width, _drawbuf.height, cast(ubyte*)pixels)) { + deleteTexture(_textureId); + _textureId = 0; + return; + } + _needUpdateTexture = false; + if (_closed) { + destroy(_drawbuf); + _drawbuf = null; + } + } + + void convertPixelFormat(GLImageCacheItem item) { + Rect rc = item._rc; + for (int y = rc.top - 1; y <= rc.bottom; y++) { + uint * row = _drawbuf.scanLine(y); + for (int x = rc.left - 1; x <= rc.right; x++) { + uint cl = row[x]; + // invert A + cl ^= 0xFF000000; + // swap R and B + uint r = (cl & 0x00FF0000) >> 16; + uint b = (cl & 0x000000FF) << 16; + row[x] = (cl & 0xFF00FF00) | r | b; + } + } + } + + GLImageCacheItem reserveSpace(uint objectId, int width, int height) { + GLImageCacheItem cacheItem = new GLImageCacheItem(this, objectId); + if (_closed) + return null; + + // next line if necessary + if (_x + width + 2 > _tdx) { + // move to next line + _currentLine = _nextLine; + _x = 0; + } + // check if no room left for glyph height + if (_currentLine + height + 2 > _tdy) { + _closed = true; + return null; + } + cacheItem._rc = Rect(_x + 1, _currentLine + 1, _x + width + 1, _currentLine + height + 1); + if (height && width) { + if (_nextLine < _currentLine + height + 2) + _nextLine = _currentLine + height + 2; + if (!_drawbuf) { + _drawbuf = new ColorDrawBuf(_tdx, _tdy); + //_drawbuf.SetBackgroundColor(0x000000); + //_drawbuf.SetTextColor(0xFFFFFF); + _drawbuf.fill(0xFF000000); + } + _x += width + 1; + _needUpdateTexture = true; + } + _itemCount++; + return cacheItem; + } + int deleteItem(GLImageCacheItem item) { + _itemCount--; + return _itemCount; + } + GLImageCacheItem addItem(DrawBuf buf) { + GLImageCacheItem cacheItem = reserveSpace(buf.id, buf.width, buf.height); + if (cacheItem is null) + return null; + buf.onDestroyCallback = &onObjectDestroyedCallback; + _drawbuf.drawImage(cacheItem._rc.left, cacheItem._rc.top, buf); + convertPixelFormat(cacheItem); + _needUpdateTexture = true; + return cacheItem; + } + void drawItem(GLImageCacheItem item, Rect dstrc, Rect srcrc, uint color, uint options, Rect * clip, int rotationAngle) { + //CRLog::trace("drawing item at %d,%d %dx%d <= %d,%d %dx%d ", x, y, dx, dy, srcx, srcy, srcdx, srcdy); + if (_needUpdateTexture) + updateTexture(); + if (_textureId != 0) { + if (!isTexture(_textureId)) { + Log.e("Invalid texture ", _textureId); + return; + } + //rotationAngle = 0; + int rx = dstrc.middlex; + int ry = dstrc.middley; + if (rotationAngle) { + //rotationAngle = 0; + //setRotation(rx, ry, rotationAngle); + } + // convert coordinates to cached texture + srcrc.offset(item._rc.left, item._rc.top); + if (clip) { + int srcw = srcrc.width(); + int srch = srcrc.height(); + int dstw = dstrc.width(); + int dsth = dstrc.height(); + if (dstw) { + srcrc.left += clip.left * srcw / dstw; + srcrc.right -= clip.right * srcw / dstw; + } + if (dsth) { + srcrc.top += clip.top * srch / dsth; + srcrc.bottom -= clip.bottom * srch / dsth; + } + dstrc.left += clip.left; + dstrc.right -= clip.right; + dstrc.top += clip.top; + dstrc.bottom -= clip.bottom; + } + if (!dstrc.empty) + drawColorAndTextureRect(_textureId, _tdx, _tdy, srcrc, dstrc, color, srcrc.width() != dstrc.width() || srcrc.height() != dstrc.height()); + //drawColorAndTextureRect(vertices, texcoords, color, _textureId); + + if (rotationAngle) { + // unset rotation + setRotation(rx, ry, 0); + // glMatrixMode(GL_PROJECTION); + // glPopMatrix(); + // checkError("pop matrix"); + } + + } + } + void close() { + _closed = true; + if (_needUpdateTexture) + updateTexture(); + } + } + + private GLImageCacheItem[uint] _map; + private GLImageCachePage[] _pages; + private GLImageCachePage _activePage; + private int tdx; + private int tdy; + + private void removePage(GLImageCachePage page) { + if (_activePage == page) + _activePage = null; + for (int i = 0; i < _pages.length; i++) + if (_pages[i] == page) { + _pages.remove(i); + break; + } + destroy(page); + } + + private void updateTextureSize() { + if (!tdx) { + // TODO + tdx = tdy = 1024; //getMaxTextureSize(); + if (tdx > 1024) + tdx = tdy = 1024; + } + } + + this() { + } + ~this() { + clear(); + } + /// returns true if object exists in cache + bool get(uint obj) { + if (obj in _map) + return true; + return false; + } + /// put new object to cache + void put(DrawBuf img) { + updateTextureSize(); + GLImageCacheItem res = null; + if (img.width <= tdx / 3 && img.height < tdy / 3) { + // trying to reuse common page for small images + if (_activePage is null) { + _activePage = new GLImageCachePage(this, tdx, tdy); + _pages ~= _activePage; + } + res = _activePage.addItem(img); + if (!res) { + _activePage = new GLImageCachePage(this, tdx, tdy); + _pages ~= _activePage; + res = _activePage.addItem(img); + } + } else { + // use separate page for big image + GLImageCachePage page = new GLImageCachePage(this, img.width, img.height); + _pages ~= page; + res = page.addItem(img); + page.close(); + } + _map[img.id] = res; + } + /// clears cache + void clear() { + for (int i = 0; i < _pages.length; i++) { + destroy(_pages[i]); + _pages[i] = null; + } + _pages.clear(); + _map.clear(); + } + /// draw cached item + void drawItem(uint objectId, Rect dstrc, Rect srcrc, uint color, int options, Rect * clip, int rotationAngle) { + if (objectId in _map) { + GLImageCacheItem item = _map[objectId]; + item.page.drawItem(item, dstrc, srcrc, color, options, clip, rotationAngle); + } + } + /// handle cached object deletion, mark as deleted + void onCachedObjectDeleted(uint objectId) { + if (objectId in _map) { + GLImageCacheItem item = _map[objectId]; + if (hasActiveScene()) { + item._deleted = true; + } else { + int itemsLeft = item.page.deleteItem(item); + //CRLog::trace("itemsLeft = %d", itemsLeft); + if (itemsLeft <= 0) { + //CRLog::trace("removing page"); + removePage(item.page); + } + _map.remove(objectId); + delete item; + } + } + } + /// remove deleted items - remove page if contains only deleted items + void removeDeletedItems() { + uint[] list; + foreach (GLImageCacheItem item; _map) { + if (item._deleted) + list ~= item._objectId; + } + for (int i = 0 ; i < list.length; i++) { + onCachedObjectDeleted(list[i]); + } + } +}; + + + +private class TextureSceneItem : SceneItem { + private uint objectId; + //CacheableObject * img; + private Rect dstrc; + private Rect srcrc; + private uint color; + private uint options; + private Rect * clip; + private int rotationAngle; + + override void draw() { + if (glImageCache) + glImageCache.drawItem(objectId, dstrc, srcrc, color, options, clip, rotationAngle); + } + + this(uint _objectId, Rect _dstrc, Rect _srcrc, uint _color, uint _options, Rect * _clip, int _rotationAngle) + { + objectId = _objectId; + dstrc = _dstrc; + srcrc = _srcrc; + color = _color; + options = _options; + clip = _clip; + rotationAngle = _rotationAngle; + } + + ~this() { + } +}; + + +/// by some reason ALPHA texture does not work as expected +private immutable USE_RGBA_TEXTURE_FOR_GLYPHS = true; + +private class GLGlyphCache { + + static class GLGlyphCacheItem { + GLGlyphCachePage _page; + public: + @property GLGlyphCachePage page() { return _page; } + uint _objectId; + // image size + Rect _rc; + bool _deleted; + this(GLGlyphCachePage page, uint objectId) { _page = page; _objectId = objectId; } + }; + + static class GLGlyphCachePage { + private GLGlyphCache _cache; + private int _tdx; + private int _tdy; + private GrayDrawBuf _drawbuf; + private int _currentLine; + private int _nextLine; + private int _x; + private bool _closed; + private bool _needUpdateTexture; + private uint _textureId; + private int _itemCount; + + this(GLGlyphCache cache, int dx, int dy) { + _cache = cache; + Log.v("created image cache page ", dx, "x", dy); + _tdx = nearestPOT(dx); + _tdy = nearestPOT(dy); + _itemCount = 0; + } + + ~this() { + if (_drawbuf) { + destroy(_drawbuf); + _drawbuf = null; + } + if (_textureId != 0) { + deleteTexture(_textureId); + _textureId = 0; + } + } + + static if (USE_RGBA_TEXTURE_FOR_GLYPHS) { + uint[] _rgbaBuffer; + } + void updateTexture() { + if (_drawbuf is null) + return; // no draw buffer!!! + if (_textureId == 0) { + //CRLog::debug("updateTexture - new texture"); + _textureId = genTexture(); + if (!_textureId) + return; + } + //CRLog::debug("updateTexture - setting image %dx%d", _drawbuf.width, _drawbuf.height); + ubyte * pixels = _drawbuf.scanLine(0); + static if (USE_RGBA_TEXTURE_FOR_GLYPHS) { + int len = _drawbuf.width * _drawbuf.height; + _rgbaBuffer.length = len; + for (int i = 0; i < len; i++) + _rgbaBuffer[i] = ((cast(uint)pixels[i]) << 24) | 0x00FFFFFF; + if (!setTextureImage(_textureId, _drawbuf.width, _drawbuf.height, cast(ubyte*)_rgbaBuffer.ptr)) { + deleteTexture(_textureId); + _textureId = 0; + return; + } + } else { + if (!setTextureImageAlpha(_textureId, _drawbuf.width, _drawbuf.height, pixels)) { + deleteTexture(_textureId); + _textureId = 0; + return; + } + } + _needUpdateTexture = false; + if (_closed) { + destroy(_drawbuf); + _drawbuf = null; + } + } + GLGlyphCacheItem reserveSpace(uint objectId, int width, int height) { + GLGlyphCacheItem cacheItem = new GLGlyphCacheItem(this, objectId); + if (_closed) + return null; + + // next line if necessary + if (_x + width + 2 > _tdx) { + // move to next line + _currentLine = _nextLine; + _x = 0; + } + // check if no room left for glyph height + if (_currentLine + height + 2 > _tdy) { + _closed = true; + return null; + } + cacheItem._rc = Rect(_x + 1, _currentLine + 1, _x + width + 1, _currentLine + height + 1); + if (height && width) { + if (_nextLine < _currentLine + height + 2) + _nextLine = _currentLine + height + 2; + if (!_drawbuf) { + _drawbuf = new GrayDrawBuf(_tdx, _tdy); + //_drawbuf.SetBackgroundColor(0x000000); + //_drawbuf.SetTextColor(0xFFFFFF); + _drawbuf.fill(0x00000000); + } + _x += width + 1; + _needUpdateTexture = true; + } + _itemCount++; + return cacheItem; + } + int deleteItem(GLGlyphCacheItem item) { + _itemCount--; + return _itemCount; + } + GLGlyphCacheItem addItem(Glyph * glyph) { + GLGlyphCacheItem cacheItem = reserveSpace(glyph.id, glyph.blackBoxX, glyph.blackBoxY); + if (cacheItem is null) + return null; + _drawbuf.drawGlyph(cacheItem._rc.left, cacheItem._rc.top, glyph, 0xFFFFFF); + _needUpdateTexture = true; + return cacheItem; + } + void drawItem(GLGlyphCacheItem item, Rect dstrc, Rect srcrc, uint color, Rect * clip) { + //CRLog::trace("drawing item at %d,%d %dx%d <= %d,%d %dx%d ", x, y, dx, dy, srcx, srcy, srcdx, srcdy); + if (_needUpdateTexture) + updateTexture(); + if (_textureId != 0) { + if (!isTexture(_textureId)) { + Log.e("Invalid texture ", _textureId); + return; + } + // convert coordinates to cached texture + srcrc.offset(item._rc.left, item._rc.top); + if (clip) { + int srcw = srcrc.width(); + int srch = srcrc.height(); + int dstw = dstrc.width(); + int dsth = dstrc.height(); + if (dstw) { + srcrc.left += clip.left * srcw / dstw; + srcrc.right -= clip.right * srcw / dstw; + } + if (dsth) { + srcrc.top += clip.top * srch / dsth; + srcrc.bottom -= clip.bottom * srch / dsth; + } + dstrc.left += clip.left; + dstrc.right -= clip.right; + dstrc.top += clip.top; + dstrc.bottom -= clip.bottom; + } + if (!dstrc.empty) { + //Log.d("drawing glyph with color ", color); + drawColorAndTextureRect(_textureId, _tdx, _tdy, srcrc, dstrc, color, false); + } + + } + } + void close() { + _closed = true; + if (_needUpdateTexture) + updateTexture(); + static if (USE_RGBA_TEXTURE_FOR_GLYPHS) { + _rgbaBuffer = null; + } + } + } + + GLGlyphCacheItem[uint] _map; + GLGlyphCachePage[] _pages; + GLGlyphCachePage _activePage; + int tdx; + int tdy; + void removePage(GLGlyphCachePage page) { + if (_activePage == page) + _activePage = null; + for (int i = 0; i < _pages.length; i++) + if (_pages[i] == page) { + _pages.remove(i); + break; + } + destroy(page); + } + private void updateTextureSize() { + if (!tdx) { + // TODO + tdx = tdy = 1024; //getMaxTextureSize(); + if (tdx > 1024) + tdx = tdy = 1024; + } + } + + this() { + } + ~this() { + clear(); + } + /// check if item is in cache + bool get(uint obj) { + if (obj in _map) + return true; + return false; + } + /// put new item to cache + void put(Glyph * glyph) { + updateTextureSize(); + GLGlyphCacheItem res = null; + if (_activePage is null) { + _activePage = new GLGlyphCachePage(this, tdx, tdy); + _pages ~= _activePage; + } + res = _activePage.addItem(glyph); + if (!res) { + _activePage = new GLGlyphCachePage(this, tdx, tdy); + _pages ~= _activePage; + res = _activePage.addItem(glyph); + } + _map[glyph.id] = res; + } + void clear() { + for (int i = 0; i < _pages.length; i++) { + destroy(_pages[i]); + _pages[i] = null; + } + _pages.clear(); + _map.clear(); + } + /// draw cached item + void drawItem(uint objectId, Rect dstrc, Rect srcrc, uint color, Rect * clip) { + GLGlyphCacheItem * item = objectId in _map; + if (item) + item.page.drawItem(*item, dstrc, srcrc, color, clip); + } + /// handle cached object deletion, mark as deleted + void onCachedObjectDeleted(uint objectId) { + if (objectId in _map) { + GLGlyphCacheItem item = _map[objectId]; + if (hasActiveScene()) { + item._deleted = true; + } else { + int itemsLeft = item.page.deleteItem(item); + //CRLog::trace("itemsLeft = %d", itemsLeft); + if (itemsLeft <= 0) { + //CRLog::trace("removing page"); + removePage(item.page); + } + _map.remove(objectId); + delete item; + } + } + } + /// remove deleted items - remove page if contains only deleted items + void removeDeletedItems() { + uint[] list; + foreach (GLGlyphCacheItem item; _map) { + if (item._deleted) + list ~= item._objectId; + } + for (int i = 0 ; i < list.length; i++) { + onCachedObjectDeleted(list[i]); + } + } +}; + + + + + + + +class SolidRectSceneItem : SceneItem { + Rect _rc; + uint _color; + this(Rect rc, uint color) { + _rc = rc; + _color = color; + } + override void draw() { + drawSolidFillRect(_rc, _color, _color, _color, _color); + } +} + +private class GlyphSceneItem : SceneItem { + uint objectId; + Rect dstrc; + Rect srcrc; + uint color; + Rect * clip; +public: + override void draw() { + if (glGlyphCache) + glGlyphCache.drawItem(objectId, dstrc, srcrc, color, clip); + } + this(uint _objectId, Rect _dstrc, Rect _srcrc, uint _color, Rect * _clip) + { + objectId = _objectId; + dstrc = _dstrc; + srcrc = _srcrc; + color = _color; + clip = _clip; + } + ~this() { + } +} + + } \ No newline at end of file diff --git a/src/dlangui/graphics/images.d b/src/dlangui/graphics/images.d index ee0002b0..7d6bc536 100644 --- a/src/dlangui/graphics/images.d +++ b/src/dlangui/graphics/images.d @@ -113,17 +113,17 @@ class DrawableCache { bool _error; bool _used; DrawableRef _drawable; - private int _instanceCount; + //private int _instanceCount; this(string id, string filename, bool tiled) { _id = id; _filename = filename; _tiled = tiled; _error = filename is null; - Log.d("Created DrawableCacheItem, count=", ++_instanceCount); + //Log.d("Created DrawableCacheItem, count=", ++_instanceCount); } ~this() { _drawable.clear(); - Log.d("Destroyed DrawableCacheItem, count=", --_instanceCount); + //Log.d("Destroyed DrawableCacheItem, count=", --_instanceCount); } /// remove from memory, will cause reload on next access void compact() { diff --git a/src/dlangui/platforms/x11/x11app.d b/src/dlangui/platforms/x11/x11app.d index b97cf80e..4e062a3d 100644 --- a/src/dlangui/platforms/x11/x11app.d +++ b/src/dlangui/platforms/x11/x11app.d @@ -54,6 +54,8 @@ version(linux) { } bool create() { + import std.c.linux.X11.Xlib; + uint mask; uint values[3]; @@ -68,11 +70,51 @@ version(linux) { xcb_create_gc(_xcbconnection, _g, _w, mask, &values[0]); } + ubyte depth = _xcbscreen.root_depth; /* create window */ _w = xcb_generate_id(_xcbconnection); Log.d("window=", _w, " gc=", _g); + if (_enableOpengl) { + int visual_attribs[] = [ + GLX_RENDER_TYPE, GLX_RGBA_BIT, + GLX_DRAWABLE_TYPE, GLX_WINDOW_BIT, + GLX_DOUBLEBUFFER, 1, + GLX_RED_SIZE, 8, + GLX_GREEN_SIZE, 8, + GLX_BLUE_SIZE, 8, + std.c.linux.X11.Xlib.None]; + + Log.d("Getting framebuffer config"); + int fbcount; + GLXFBConfig *fbc = glXChooseFBConfig(_display, DefaultScreen(_display), visual_attribs.ptr, &fbcount); + if (!fbc) + { + Log.d("Failed to retrieve a framebuffer config"); + //return 1; + } + + Log.d("Getting XVisualInfo"); + _fb_config = fbc[0]; + auto vi = glXGetVisualFromFBConfig(_display, _fb_config); + + //auto vi = glXChooseVisual(_display, std.c.linux.X11.Xlib.DefaultScreen(_display), attributeList.ptr); + _visualID = vi.visualid; + //swa.colormap = std.c.linux.X11.Xlib.XCreateColormap(_display, std.c.linux.X11.Xlib.RootWindow(_display, vi.screen), vi.visual, 0); // AllocNone + + Log.d("Creating color map"); + _colormap = xcb_generate_id(_xcbconnection); + /* Create colormap */ + xcb_create_colormap( + _xcbconnection, + XCB_COLORMAP_ALLOC_NONE, + _colormap, + _xcbscreen.root, + _visualID + ); + depth = cast(ubyte)vi.depth; + } //mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK; //values[0] = _xcbscreen.white_pixel; @@ -84,12 +126,12 @@ version(linux) { | XCB_EVENT_MASK_KEY_PRESS | XCB_EVENT_MASK_KEY_RELEASE | XCB_EVENT_MASK_STRUCTURE_NOTIFY | XCB_EVENT_MASK_VISIBILITY_CHANGE; if (_enableOpengl) { - mask = XCB_CW_EVENT_MASK | XCB_CW_COLORMAP; - //values[0] = _xcbscreen.white_pixel; - values[0] = eventmask; - values[1] = _colormap; - visualId = _xcbscreen.root_visual; - //visualId = _visualID; + mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK | XCB_CW_COLORMAP; + values[0] = _xcbscreen.white_pixel; + values[1] = eventmask; + values[2] = _colormap; + //visualId = _xcbscreen.root_visual; + visualId = _visualID; } else { mask = XCB_CW_BACK_PIXEL | XCB_CW_EVENT_MASK; values[0] = _xcbscreen.white_pixel; @@ -98,7 +140,7 @@ version(linux) { } Log.d("xcb_create_window - window=", _w, " VisualID=", _visualID); auto res = xcb_create_window(_xcbconnection, - _xcbscreen.root_depth, + depth, //_xcbscreen.root_depth, //XCB_COPY_FROM_PARENT,//_xcbscreen.root_depth, _w, _xcbscreen.root, @@ -193,14 +235,15 @@ version(linux) { cast(short)rc.left, cast(short)rc.top, cast(short)rc.left, cast(short)rc.top, cast(ushort)rc.width(), cast(ushort)rc.height(), 0); xcb_flush(_xcbconnection); } - + + bool _derelictgl3Reloaded; override void show() { Log.d("XCBWindow.show()"); /* map (show) the window */ xcb_map_window(_xcbconnection, _w); xcb_flush(_xcbconnection); //_enableOpengl = false; // test - if (_enableOpengl) { + if (_enableOpengl && !_glxwindow) { Log.d("Calling glXCreateWindow display=", _display, " fbconfig=", _fb_config, " window=", _w); _glxwindow = glXCreateWindow( _display, @@ -211,14 +254,77 @@ version(linux) { Log.e("Failed to create GLX window: disabling OpenGL"); _enableOpengl = false; } else { + import derelict.opengl3.glxext; + import std.c.linux.X11.Xlib; + _drawable = _glxwindow; + + if (!_derelictgl3Reloaded) { + Log.e("Reloading DerelictGL3"); + _derelictgl3Reloaded = true; + _context = glXCreateNewContext(_display, _fb_config, GLX_RGBA_TYPE, null, true); + if (_context is null) { + Log.e("Cannot create temporary context"); + } + glXMakeContextCurrent(_display, _drawable, _drawable, _context); + //glXMakeCurrent(_display, _w, _context); + DerelictGL3.reload(); + Log.e("Reloaded DerelictGL3 - removing temporary context"); + glXMakeCurrent(_display, 0, null); + Log.e("Destroying context"); + glXDestroyContext(_display, _context); + Log.e("DerelictGL3 initialized"); + _context = null; + } + + + // Get the default screen's GLX extension list + const char *glxExts = glXQueryExtensionsString( _display, + DefaultScreen( _display ) ); + Log.d("Extensions: ", fromStringz(glxExts)); + + + +// da_glXCreateContextAttribsARB glXCreateContextAttribsARB; +// glXCreateContextAttribsARB = cast(da_glXCreateContextAttribsARB) +// glXGetProcAddressARB( cast(const GLubyte *)("glXCreateContextAttribsARB".toStringz)); + + int context_attribs[] = + [ + GLX_CONTEXT_MAJOR_VERSION_ARB, 3, + GLX_CONTEXT_MINOR_VERSION_ARB, 0, + None + ]; + + Log.d("Creating context"); + _context = glXCreateNewContext(_display, _fb_config, GLX_RGBA_TYPE, null, true); + //if (glXCreateContextAttribsARB is null) { + // Log.e("glXCreateContextAttribsARB function is not found"); + // _context = glXCreateNewContext(_display, _fb_config, GLX_RGBA_TYPE, null, true); + //} else { + // Log.e("calling glXCreateContextAttribsARB"); + // _context = glXCreateContextAttribsARB(_display, _fb_config, null, true, context_attribs.ptr); + //} + Log.d("Created context: ", _context); + + /* Create OpenGL context */ + //auto context = glXCreateNewContext(_display, _fb_config, GLX_RGBA_TYPE, null, true); + if (!_context) { + _enableOpengl = false; + Log.e("Failed to create OpenGL context"); + } else { + + } + /* make OpenGL context current */ - if(!glXMakeContextCurrent(_display, _drawable, _drawable, _context)) { + if(!glXMakeContextCurrent(_display, _drawable, _drawable, _context) || !initShaders()) { Log.e("Failed to make GL context current"); _enableOpengl = false; glXDestroyContext(_display, _context); _context = null; - } + } else { + + } } } } @@ -242,9 +348,26 @@ version(linux) { void redraw() { if (_enableOpengl) { + Log.d("Drawing using opengl ", _dx, "x", _dy, " context=", _context); glXMakeContextCurrent(_display, _drawable, _drawable, _context); - glClearColor(0.2, 0.4, 0.9, 1.0); + glEnable(GL_BLEND); + glDisable(GL_CULL_FACE); + glDisable(GL_DEPTH_TEST); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + glViewport(0, 0, _dx, _dy); + Log.d("glClearColor"); + glClearColor(0.2f, 0.4f, 0.9f, 1.0f); + Log.d("glClear"); glClear(GL_COLOR_BUFFER_BIT); + + import dlangui.graphics.gldrawbuf; + GLDrawBuf buf = new GLDrawBuf(_dx, _dy); + buf.beforeDrawing(); + onDraw(buf); + buf.fillRect(Rect(0, 0, 100, 100), 0x805010); + buf.afterDrawing(); + destroy(buf); + Log.d("Calling glx swap buffers"); glXSwapBuffers(_display, _drawable); //glXMakeContextCurrent(_display, _drawable, _drawable, null); } else { @@ -514,7 +637,11 @@ version(linux) { GLX_BLUE_SIZE, 8, None]; fbc = glXChooseFBConfig(_display, DefaultScreen(_display), attrib_list.ptr, &nelements); - auto vi = glXGetVisualFromFBConfig(_display, fbc[0]); + + int attributeList[] = [GLX_RGBA, GLX_DOUBLEBUFFER, GLX_RED_SIZE, 1, GLX_GREEN_SIZE, 1, GLX_BLUE_SIZE, 1, None]; + auto vi = glXChooseVisual(_display, DefaultScreen(_display), attributeList.ptr); + //_visualID = vi.visual; + //auto vi = glXGetVisualFromFBConfig(_display, fbc[0]); _fb_config = fbc[0]; /* Select first framebuffer config and query visualID */ @@ -526,23 +653,6 @@ version(linux) { } Log.d("Selected fbconfig=", _fb_config, " visualId=", _visualID); - /* Create OpenGL context */ - _context = glXCreateNewContext(_display, _fb_config, GLX_RGBA_TYPE, null, true); - if (!_context) { - _enableOpengl = false; - Log.e("Failed to create OpenGL context"); - } else { - Log.d("Creating color map"); - xcb_colormap_t colormap = xcb_generate_id(_xcbconnection); - /* Create colormap */ - xcb_create_colormap( - _xcbconnection, - XCB_COLORMAP_ALLOC_NONE, - _colormap, - _xcbscreen.root, - _visualID - ); - } } }