refactoring

This commit is contained in:
Vadim Lopatin 2014-04-11 12:09:14 +04:00
parent f3c0197837
commit 9e562548ba
8 changed files with 591 additions and 535 deletions

View File

@ -312,6 +312,7 @@
<File path="src\dlangui\graphics\gldrawbuf.d" />
<File path="src\dlangui\graphics\glsupport.d" />
<File path="src\dlangui\graphics\images.d" />
<File path="src\dlangui\graphics\resources.d" />
</Folder>
<Folder name="platforms">
<Folder name="common">

View File

@ -2,7 +2,7 @@
<selector xmlns:android="http://schemas.android.com/apk/res/android"
android:constantSize="true"
android:dither="false"
android:variablePadding="false"
android:variablePadding="false" >
<item
android:drawable="tab_btn_up_hover"
android:state_hovered="true" />

View File

@ -193,3 +193,20 @@ wstring fromWStringz(const(wchar[]) s) {
return cast(wstring)(s[0..i].dup);
}
/// widget state flags - bits
enum State : uint {
/// state not specified / normal
Normal = 0,
Pressed = 1,
Focused = 2,
Enabled = 4,
Hovered = 8, // mouse pointer is over control, buttons not pressed
Selected = 16,
Checkable = 32,
Checked = 64,
Activated = 128,
WindowFocused = 256,
Parent = 0x10000, // use parent's state
}

View File

@ -679,292 +679,4 @@ class ColorDrawBuf : ColorDrawBufBase {
}
}
class Drawable : RefCountedObject {
//private static int _instanceCount;
this() {
//Log.d("Created drawable, count=", ++_instanceCount);
}
~this() {
//Log.d("Destroyed drawable, count=", --_instanceCount);
}
abstract void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0);
@property abstract int width();
@property abstract int height();
@property Rect padding() { return Rect(0,0,0,0); }
}
class EmptyDrawable : Drawable {
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
}
@property override int width() { return 0; }
@property override int height() { return 0; }
}
class SolidFillDrawable : Drawable {
protected uint _color;
this(uint color) {
_color = color;
}
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
if ((_color >> 24) != 0xFF) // not fully transparent
buf.fillRect(rc, _color);
}
@property override int width() { return 1; }
@property override int height() { return 1; }
}
class ImageDrawable : Drawable {
protected DrawBufRef _image;
protected bool _tiled;
//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);
}
~this() {
_image.clear();
//Log.d("Destroyed ImageDrawable, count=", --_instanceCount);
}
@property override int width() {
if (_image.isNull)
return 0;
if (_image.hasNinePatch)
return _image.width - 2;
return _image.width;
}
@property override int height() {
if (_image.isNull)
return 0;
if (_image.hasNinePatch)
return _image.height - 2;
return _image.height;
}
@property override Rect padding() {
if (!_image.isNull && _image.hasNinePatch)
return _image.ninePatch.padding;
return Rect(0,0,0,0);
}
private static void correctFrameBounds(ref int n1, ref int n2, ref int n3, ref int n4) {
if (n1 > n2) {
//assert(n2 - n1 == n4 - n3);
int middledist = (n1 + n2) / 2 - n1;
n1 = n2 = n1 + middledist;
n3 = n4 = n3 + middledist;
}
}
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
if (_image.isNull)
return;
if (_image.hasNinePatch) {
// draw nine patch
const NinePatch * p = _image.ninePatch;
//Log.d("drawing nine patch image with frame ", p.frame, " padding ", p.padding);
int w = width;
int h = height;
Rect dstrect = rc;
Rect srcrect = Rect(1, 1, w + 1, h + 1);
if (true) { //buf.applyClipping(dstrect, srcrect)) {
int x0 = srcrect.left;
int x1 = srcrect.left + p.frame.left;
int x2 = srcrect.right - p.frame.right;
int x3 = srcrect.right;
int y0 = srcrect.top;
int y1 = srcrect.top + p.frame.top;
int y2 = srcrect.bottom - p.frame.bottom;
int y3 = srcrect.bottom;
int dstx0 = rc.left;
int dstx1 = rc.left + p.frame.left;
int dstx2 = rc.right - p.frame.right;
int dstx3 = rc.right;
int dsty0 = rc.top;
int dsty1 = rc.top + p.frame.top;
int dsty2 = rc.bottom - p.frame.bottom;
int dsty3 = rc.bottom;
//Log.d("x bounds: ", x0, ", ", x1, ", ", x2, ", ", x3, " dst ", dstx0, ", ", dstx1, ", ", dstx2, ", ", dstx3);
//Log.d("y bounds: ", y0, ", ", y1, ", ", y2, ", ", y3, " dst ", dsty0, ", ", dsty1, ", ", dsty2, ", ", dsty3);
correctFrameBounds(x1, x2, dstx1, dstx2);
correctFrameBounds(y1, y2, dsty1, dsty2);
//correctFrameBounds(x1, x2);
//correctFrameBounds(y1, y2);
//correctFrameBounds(dstx1, dstx2);
//correctFrameBounds(dsty1, dsty2);
if (y0 < y1 && dsty0 < dsty1) {
// top row
if (x0 < x1 && dstx0 < dstx1)
buf.drawFragment(dstx0, dsty0, _image.get, Rect(x0, y0, x1, y1)); // top left
if (x1 < x2 && dstx1 < dstx2)
buf.drawRescaled(Rect(dstx1, dsty0, dstx2, dsty1), _image.get, Rect(x1, y0, x2, y1)); // top center
if (x2 < x3 && dstx2 < dstx3)
buf.drawFragment(dstx2, dsty0, _image.get, Rect(x2, y0, x3, y1)); // top right
}
if (y1 < y2 && dsty1 < dsty2) {
// middle row
if (x0 < x1 && dstx0 < dstx1)
buf.drawRescaled(Rect(dstx0, dsty1, dstx1, dsty2), _image.get, Rect(x0, y1, x1, y2)); // middle center
if (x1 < x2 && dstx1 < dstx2)
buf.drawRescaled(Rect(dstx1, dsty1, dstx2, dsty2), _image.get, Rect(x1, y1, x2, y2)); // center
if (x2 < x3 && dstx2 < dstx3)
buf.drawRescaled(Rect(dstx2, dsty1, dstx3, dsty2), _image.get, Rect(x2, y1, x3, y2)); // middle center
}
if (y2 < y3 && dsty2 < dsty3) {
// bottom row
if (x0 < x1 && dstx0 < dstx1)
buf.drawFragment(dstx0, dsty2, _image.get, Rect(x0, y2, x1, y3)); // bottom left
if (x1 < x2 && dstx1 < dstx2)
buf.drawRescaled(Rect(dstx1, dsty2, dstx2, dsty3), _image.get, Rect(x1, y2, x2, y3)); // bottom center
if (x2 < x3 && dstx2 < dstx3)
buf.drawFragment(dstx2, dsty2, _image.get, Rect(x2, y2, x3, y3)); // bottom right
}
}
} else if (_tiled) {
// tiled
} else {
// rescaled or normal
if (rc.width != _image.width || rc.height != _image.height)
buf.drawRescaled(rc, _image.get, Rect(0, 0, _image.width, _image.height));
else
buf.drawImage(rc.left, rc.top, _image);
}
}
}
import std.xml;
import std.algorithm;
import dlangui.widgets.styles;
void extractStateFlag(ref string[string] attr, string attrName, State state, ref uint stateMask, ref uint stateValue) {
if (attrName in attr) {
string value = attr[attrName];
if (value.equal("true"))
stateValue |= state;
stateMask |= state;
}
}
/// converts XML attribute name to State (see http://developer.android.com/guide/topics/resources/drawable-resource.html#StateList)
void extractStateFlags(ref string[string] attr, ref uint stateMask, ref uint stateValue) {
extractStateFlag(attr, "state_pressed", State.Pressed, stateMask, stateValue);
extractStateFlag(attr, "state_focused", State.Focused, stateMask, stateValue);
extractStateFlag(attr, "state_hovered", State.Hovered, stateMask, stateValue);
extractStateFlag(attr, "state_selected", State.Selected, stateMask, stateValue);
extractStateFlag(attr, "state_checkable", State.Checkable, stateMask, stateValue);
extractStateFlag(attr, "state_checked", State.Checked, stateMask, stateValue);
extractStateFlag(attr, "state_enabled", State.Enabled, stateMask, stateValue);
extractStateFlag(attr, "state_activated", State.Activated, stateMask, stateValue);
extractStateFlag(attr, "state_window_focused", State.WindowFocused, stateMask, stateValue);
}
/*
sample:
(prefix android: is optional)
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android"
android:constantSize=["true" | "false"]
android:dither=["true" | "false"]
android:variablePadding=["true" | "false"] >
<item
android:drawable="@[package:]drawable/drawable_resource"
android:state_pressed=["true" | "false"]
android:state_focused=["true" | "false"]
android:state_hovered=["true" | "false"]
android:state_selected=["true" | "false"]
android:state_checkable=["true" | "false"]
android:state_checked=["true" | "false"]
android:state_enabled=["true" | "false"]
android:state_activated=["true" | "false"]
android:state_window_focused=["true" | "false"] />
</selector>
*/
/// Drawable which is drawn depending on state (see http://developer.android.com/guide/topics/resources/drawable-resource.html#StateList)
class StateDrawable : Drawable {
static struct StateItem {
uint stateMask;
uint stateValue;
DrawableRef drawable;
@property bool matchState(uint state) {
return (stateMask & state) == stateValue;
}
}
protected StateItem[] _stateList;
void addState(uint stateMask, uint stateValue, string resourceId) {
StateItem item;
item.stateMask = stateMask;
item.stateValue = stateValue;
item.drawable = dlangui.graphics.images.drawableCache.get(resourceId);
_stateList ~= item;
}
void addState(uint stateMask, uint stateValue, DrawableRef drawable) {
StateItem item;
item.stateMask = stateMask;
item.stateValue = stateValue;
item.drawable = drawable;
_stateList ~= item;
}
bool load(Element element) {
foreach(item; element.elements) {
if (item.tag.name.equal("item")) {
if ("drawable" in item.tag.attr) {
string drawableId = item.tag.attr["drawable"];
uint stateMask, stateValue;
extractStateFlags(item.tag.attr, stateMask, stateValue);
if (drawableId !is null) {
addState(stateMask, stateValue, drawableId);
}
}
}
}
return _stateList.length > 0;
}
/// load from XML file
bool load(string filename) {
import std.file;
import std.string;
try {
string s = cast(string)std.file.read(filename);
// Check for well-formedness
check(s);
// Make a DOM tree
auto doc = new Document(s);
return load(doc);
} catch (Exception e) {
Log.e("Cannot read drawable resource from file ", filename);
return false;
}
}
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
foreach(ref item; _stateList)
if (item.matchState(state)) {
item.drawable.drawTo(buf, rc, state, tilex0, tiley0);
return;
}
}
@property override int width() {
return (_stateList.length > 0) ? _stateList[0].drawable.width : 0;
}
@property override int height() {
return (_stateList.length > 0) ? _stateList[0].drawable.height : 0;
}
@property override Rect padding() {
return (_stateList.length > 0) ? _stateList[0].drawable.padding : Rect(0,0,0,0);
}
}
alias DrawableRef = Ref!Drawable;

View File

@ -4,231 +4,8 @@ import dlangui.core.logger;
import dlangui.core.types;
import dlangui.graphics.drawbuf;
import std.stream;
import std.file;
import std.algorithm;
import libpng.png;
/// decoded image cache
class ImageCache {
static class ImageCacheItem {
string _filename;
DrawBufRef _drawbuf;
bool _error; // flag to avoid loading of file if it has been failed once
bool _used;
this(string filename) {
_filename = filename;
}
@property ref DrawBufRef get() {
if (!_drawbuf.isNull || _error) {
_used = true;
return _drawbuf;
}
_drawbuf = loadImage(_filename);
_used = true;
if (_drawbuf.isNull)
_error = true;
return _drawbuf;
}
/// remove from memory, will cause reload on next access
void compact() {
if (!_drawbuf.isNull)
_drawbuf.clear();
}
/// mark as not used
void checkpoint() {
_used = false;
}
/// cleanup if unused since last checkpoint
void cleanup() {
if (!_used)
compact();
}
}
ImageCacheItem[string] _map;
/// get and cache image
ref DrawBufRef get(string filename) {
if (filename in _map) {
return _map[filename].get;
}
ImageCacheItem item = new ImageCacheItem(filename);
_map[filename] = item;
return item.get;
}
// clear usage flags for all entries
void checkpoint() {
foreach (item; _map)
item.checkpoint();
}
// removes entries not used after last call of checkpoint() or cleanup()
void cleanup() {
foreach (item; _map)
item.cleanup();
}
this() {
Log.i("Creating ImageCache");
}
~this() {
Log.i("Destroying ImageCache");
foreach (ref item; _map) {
destroy(item);
item = null;
}
_map.clear();
}
}
__gshared ImageCache _imageCache;
/// image cache singleton
@property ImageCache imageCache() { return _imageCache; }
/// image cache singleton
@property void imageCache(ImageCache cache) {
if (_imageCache !is null)
destroy(_imageCache);
_imageCache = cache;
}
__gshared DrawableCache _drawableCache;
/// drawable cache singleton
@property DrawableCache drawableCache() { return _drawableCache; }
/// drawable cache singleton
@property void drawableCache(DrawableCache cache) {
if (_drawableCache !is null)
destroy(_drawableCache);
_drawableCache = cache;
}
shared static this() {
_imageCache = new ImageCache();
_drawableCache = new DrawableCache();
}
class DrawableCache {
static class DrawableCacheItem {
string _id;
string _filename;
bool _tiled;
bool _error;
bool _used;
DrawableRef _drawable;
//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);
}
~this() {
_drawable.clear();
//Log.d("Destroyed DrawableCacheItem, count=", --_instanceCount);
}
/// remove from memory, will cause reload on next access
void compact() {
if (!_drawable.isNull)
_drawable.clear();
}
/// mark as not used
void checkpoint() {
_used = false;
}
/// cleanup if unused since last checkpoint
void cleanup() {
if (!_used)
compact();
}
@property ref DrawableRef drawable() {
_used = true;
if (!_drawable.isNull || _error)
return _drawable;
if (_filename !is null) {
// reload from file
DrawBufRef image = imageCache.get(_filename);
if (!image.isNull) {
bool ninePatch = _filename.endsWith(".9.png");
_drawable = new ImageDrawable(image, _tiled, ninePatch);
} else
_error = true;
}
return _drawable;
}
}
void clear() {
Log.d("DrawableCache.clear()");
_idToFileMap.clear();
foreach(DrawableCacheItem item; _idToDrawableMap)
item.drawable.clear();
_idToDrawableMap.clear();
}
// clear usage flags for all entries
void checkpoint() {
foreach (item; _idToDrawableMap)
item.checkpoint();
}
// removes entries not used after last call of checkpoint() or cleanup()
void cleanup() {
foreach (item; _idToDrawableMap)
item.cleanup();
}
string[] _resourcePaths;
string[string] _idToFileMap;
DrawableCacheItem[string] _idToDrawableMap;
ref DrawableRef get(string id) {
if (id in _idToDrawableMap)
return _idToDrawableMap[id].drawable;
string resourceId = id;
bool tiled = false;
if (id.endsWith(".tiled")) {
resourceId = id[0..$-6]; // remove .tiled
tiled = true;
}
string filename = findResource(resourceId);
DrawableCacheItem item = new DrawableCacheItem(id, filename, tiled);
_idToDrawableMap[id] = item;
return item.drawable;
}
@property string[] resourcePaths() {
return _resourcePaths;
}
@property void resourcePaths(string[] paths) {
_resourcePaths = paths;
clear();
}
string findResource(string id) {
if (id in _idToFileMap)
return _idToFileMap[id];
foreach(string path; _resourcePaths) {
char[] name = path.dup;
name ~= id;
name ~= ".png";
if (!exists(name)) {
name = path.dup;
name ~= id;
name ~= ".9.png";
}
if (exists(name) && isFile(name)) {
string filename = name.dup;
_idToFileMap[id] = filename;
return filename;
}
}
return null;
}
this() {
Log.i("Creating DrawableCache");
}
~this() {
Log.i("Destroying DrawableCache");
foreach (ref item; _idToDrawableMap) {
destroy(item);
item = null;
}
_idToDrawableMap.clear();
}
}
/// load and decode image from file to ColorDrawBuf, returns null if loading or decoding is failed
ColorDrawBuf loadImage(string filename) {
Log.d("Loading image from file " ~ filename);

View File

@ -0,0 +1,562 @@
module dlangui.graphics.resources;
import dlangui.graphics.images;
import dlangui.graphics.drawbuf;
import dlangui.core.logger;
import std.file;
import std.algorithm;
import std.xml;
import std.algorithm;
class Drawable : RefCountedObject {
//private static int _instanceCount;
this() {
//Log.d("Created drawable, count=", ++_instanceCount);
}
~this() {
//Log.d("Destroyed drawable, count=", --_instanceCount);
}
abstract void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0);
@property abstract int width();
@property abstract int height();
@property Rect padding() { return Rect(0,0,0,0); }
}
class EmptyDrawable : Drawable {
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
}
@property override int width() { return 0; }
@property override int height() { return 0; }
}
class SolidFillDrawable : Drawable {
protected uint _color;
this(uint color) {
_color = color;
}
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
if ((_color >> 24) != 0xFF) // not fully transparent
buf.fillRect(rc, _color);
}
@property override int width() { return 1; }
@property override int height() { return 1; }
}
class ImageDrawable : Drawable {
protected DrawBufRef _image;
protected bool _tiled;
//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);
}
~this() {
_image.clear();
//Log.d("Destroyed ImageDrawable, count=", --_instanceCount);
}
@property override int width() {
if (_image.isNull)
return 0;
if (_image.hasNinePatch)
return _image.width - 2;
return _image.width;
}
@property override int height() {
if (_image.isNull)
return 0;
if (_image.hasNinePatch)
return _image.height - 2;
return _image.height;
}
@property override Rect padding() {
if (!_image.isNull && _image.hasNinePatch)
return _image.ninePatch.padding;
return Rect(0,0,0,0);
}
private static void correctFrameBounds(ref int n1, ref int n2, ref int n3, ref int n4) {
if (n1 > n2) {
//assert(n2 - n1 == n4 - n3);
int middledist = (n1 + n2) / 2 - n1;
n1 = n2 = n1 + middledist;
n3 = n4 = n3 + middledist;
}
}
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
if (_image.isNull)
return;
if (_image.hasNinePatch) {
// draw nine patch
const NinePatch * p = _image.ninePatch;
//Log.d("drawing nine patch image with frame ", p.frame, " padding ", p.padding);
int w = width;
int h = height;
Rect dstrect = rc;
Rect srcrect = Rect(1, 1, w + 1, h + 1);
if (true) { //buf.applyClipping(dstrect, srcrect)) {
int x0 = srcrect.left;
int x1 = srcrect.left + p.frame.left;
int x2 = srcrect.right - p.frame.right;
int x3 = srcrect.right;
int y0 = srcrect.top;
int y1 = srcrect.top + p.frame.top;
int y2 = srcrect.bottom - p.frame.bottom;
int y3 = srcrect.bottom;
int dstx0 = rc.left;
int dstx1 = rc.left + p.frame.left;
int dstx2 = rc.right - p.frame.right;
int dstx3 = rc.right;
int dsty0 = rc.top;
int dsty1 = rc.top + p.frame.top;
int dsty2 = rc.bottom - p.frame.bottom;
int dsty3 = rc.bottom;
//Log.d("x bounds: ", x0, ", ", x1, ", ", x2, ", ", x3, " dst ", dstx0, ", ", dstx1, ", ", dstx2, ", ", dstx3);
//Log.d("y bounds: ", y0, ", ", y1, ", ", y2, ", ", y3, " dst ", dsty0, ", ", dsty1, ", ", dsty2, ", ", dsty3);
correctFrameBounds(x1, x2, dstx1, dstx2);
correctFrameBounds(y1, y2, dsty1, dsty2);
//correctFrameBounds(x1, x2);
//correctFrameBounds(y1, y2);
//correctFrameBounds(dstx1, dstx2);
//correctFrameBounds(dsty1, dsty2);
if (y0 < y1 && dsty0 < dsty1) {
// top row
if (x0 < x1 && dstx0 < dstx1)
buf.drawFragment(dstx0, dsty0, _image.get, Rect(x0, y0, x1, y1)); // top left
if (x1 < x2 && dstx1 < dstx2)
buf.drawRescaled(Rect(dstx1, dsty0, dstx2, dsty1), _image.get, Rect(x1, y0, x2, y1)); // top center
if (x2 < x3 && dstx2 < dstx3)
buf.drawFragment(dstx2, dsty0, _image.get, Rect(x2, y0, x3, y1)); // top right
}
if (y1 < y2 && dsty1 < dsty2) {
// middle row
if (x0 < x1 && dstx0 < dstx1)
buf.drawRescaled(Rect(dstx0, dsty1, dstx1, dsty2), _image.get, Rect(x0, y1, x1, y2)); // middle center
if (x1 < x2 && dstx1 < dstx2)
buf.drawRescaled(Rect(dstx1, dsty1, dstx2, dsty2), _image.get, Rect(x1, y1, x2, y2)); // center
if (x2 < x3 && dstx2 < dstx3)
buf.drawRescaled(Rect(dstx2, dsty1, dstx3, dsty2), _image.get, Rect(x2, y1, x3, y2)); // middle center
}
if (y2 < y3 && dsty2 < dsty3) {
// bottom row
if (x0 < x1 && dstx0 < dstx1)
buf.drawFragment(dstx0, dsty2, _image.get, Rect(x0, y2, x1, y3)); // bottom left
if (x1 < x2 && dstx1 < dstx2)
buf.drawRescaled(Rect(dstx1, dsty2, dstx2, dsty3), _image.get, Rect(x1, y2, x2, y3)); // bottom center
if (x2 < x3 && dstx2 < dstx3)
buf.drawFragment(dstx2, dsty2, _image.get, Rect(x2, y2, x3, y3)); // bottom right
}
}
} else if (_tiled) {
// tiled
} else {
// rescaled or normal
if (rc.width != _image.width || rc.height != _image.height)
buf.drawRescaled(rc, _image.get, Rect(0, 0, _image.width, _image.height));
else
buf.drawImage(rc.left, rc.top, _image);
}
}
}
string attrValue(Element item, string attrname, string attrname2) {
if (attrname in item.tag.attr)
return item.tag.attr[attrname];
if (attrname2 in item.tag.attr)
return item.tag.attr[attrname2];
return null;
}
string attrValue(ref string[string] attr, string attrname, string attrname2) {
if (attrname in attr)
return attr[attrname];
if (attrname2 in attr)
return attr[attrname2];
return null;
}
void extractStateFlag(ref string[string] attr, string attrName, string attrName2, State state, ref uint stateMask, ref uint stateValue) {
string value = attrValue(attr, attrName, attrName2);
if (value !is null) {
if (value.equal("true"))
stateValue |= state;
stateMask |= state;
}
}
/// converts XML attribute name to State (see http://developer.android.com/guide/topics/resources/drawable-resource.html#StateList)
void extractStateFlags(ref string[string] attr, ref uint stateMask, ref uint stateValue) {
extractStateFlag(attr, "state_pressed", "android:state_pressed", State.Pressed, stateMask, stateValue);
extractStateFlag(attr, "state_focused", "android:state_focused", State.Focused, stateMask, stateValue);
extractStateFlag(attr, "state_hovered", "android:state_hovered", State.Hovered, stateMask, stateValue);
extractStateFlag(attr, "state_selected", "android:state_selected", State.Selected, stateMask, stateValue);
extractStateFlag(attr, "state_checkable", "android:state_checkable", State.Checkable, stateMask, stateValue);
extractStateFlag(attr, "state_checked", "android:state_checked", State.Checked, stateMask, stateValue);
extractStateFlag(attr, "state_enabled", "android:state_enabled", State.Enabled, stateMask, stateValue);
extractStateFlag(attr, "state_activated", "android:state_activated", State.Activated, stateMask, stateValue);
extractStateFlag(attr, "state_window_focused", "android:state_window_focused", State.WindowFocused, stateMask, stateValue);
}
/*
sample:
(prefix android: is optional)
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android"
android:constantSize=["true" | "false"]
android:dither=["true" | "false"]
android:variablePadding=["true" | "false"] >
<item
android:drawable="@[package:]drawable/drawable_resource"
android:state_pressed=["true" | "false"]
android:state_focused=["true" | "false"]
android:state_hovered=["true" | "false"]
android:state_selected=["true" | "false"]
android:state_checkable=["true" | "false"]
android:state_checked=["true" | "false"]
android:state_enabled=["true" | "false"]
android:state_activated=["true" | "false"]
android:state_window_focused=["true" | "false"] />
</selector>
*/
/// Drawable which is drawn depending on state (see http://developer.android.com/guide/topics/resources/drawable-resource.html#StateList)
class StateDrawable : Drawable {
static struct StateItem {
uint stateMask;
uint stateValue;
DrawableRef drawable;
@property bool matchState(uint state) {
return (stateMask & state) == stateValue;
}
}
protected StateItem[] _stateList;
void addState(uint stateMask, uint stateValue, string resourceId) {
StateItem item;
item.stateMask = stateMask;
item.stateValue = stateValue;
item.drawable = drawableCache.get(resourceId);
_stateList ~= item;
}
void addState(uint stateMask, uint stateValue, DrawableRef drawable) {
StateItem item;
item.stateMask = stateMask;
item.stateValue = stateValue;
item.drawable = drawable;
_stateList ~= item;
}
bool load(Element element) {
foreach(item; element.elements) {
if (item.tag.name.equal("item")) {
string drawableId = attrValue(item, "drawable", "android:drawable");
if (drawableId !is null) {
uint stateMask, stateValue;
extractStateFlags(item.tag.attr, stateMask, stateValue);
if (drawableId !is null) {
addState(stateMask, stateValue, drawableId);
}
}
}
}
return _stateList.length > 0;
}
/// load from XML file
bool load(string filename) {
import std.file;
import std.string;
try {
string s = cast(string)std.file.read(filename);
// Check for well-formedness
check(s);
// Make a DOM tree
auto doc = new Document(s);
return load(doc);
} catch (CheckException e) {
Log.e("Invalid XML file ", filename);
return false;
} catch (Throwable e) {
Log.e("Cannot read drawable resource from file ", filename);
return false;
}
}
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
foreach(ref item; _stateList)
if (item.matchState(state)) {
item.drawable.drawTo(buf, rc, state, tilex0, tiley0);
return;
}
}
@property override int width() {
return (_stateList.length > 0) ? _stateList[0].drawable.width : 0;
}
@property override int height() {
return (_stateList.length > 0) ? _stateList[0].drawable.height : 0;
}
@property override Rect padding() {
return (_stateList.length > 0) ? _stateList[0].drawable.padding : Rect(0,0,0,0);
}
}
alias DrawableRef = Ref!Drawable;
/// decoded image cache
class ImageCache {
static class ImageCacheItem {
string _filename;
DrawBufRef _drawbuf;
bool _error; // flag to avoid loading of file if it has been failed once
bool _used;
this(string filename) {
_filename = filename;
}
@property ref DrawBufRef get() {
if (!_drawbuf.isNull || _error) {
_used = true;
return _drawbuf;
}
_drawbuf = loadImage(_filename);
_used = true;
if (_drawbuf.isNull)
_error = true;
return _drawbuf;
}
/// remove from memory, will cause reload on next access
void compact() {
if (!_drawbuf.isNull)
_drawbuf.clear();
}
/// mark as not used
void checkpoint() {
_used = false;
}
/// cleanup if unused since last checkpoint
void cleanup() {
if (!_used)
compact();
}
}
ImageCacheItem[string] _map;
/// get and cache image
ref DrawBufRef get(string filename) {
if (filename in _map) {
return _map[filename].get;
}
ImageCacheItem item = new ImageCacheItem(filename);
_map[filename] = item;
return item.get;
}
// clear usage flags for all entries
void checkpoint() {
foreach (item; _map)
item.checkpoint();
}
// removes entries not used after last call of checkpoint() or cleanup()
void cleanup() {
foreach (item; _map)
item.cleanup();
}
this() {
Log.i("Creating ImageCache");
}
~this() {
Log.i("Destroying ImageCache");
foreach (ref item; _map) {
destroy(item);
item = null;
}
_map.clear();
}
}
__gshared ImageCache _imageCache;
/// image cache singleton
@property ImageCache imageCache() { return _imageCache; }
/// image cache singleton
@property void imageCache(ImageCache cache) {
if (_imageCache !is null)
destroy(_imageCache);
_imageCache = cache;
}
__gshared DrawableCache _drawableCache;
/// drawable cache singleton
@property DrawableCache drawableCache() { return _drawableCache; }
/// drawable cache singleton
@property void drawableCache(DrawableCache cache) {
if (_drawableCache !is null)
destroy(_drawableCache);
_drawableCache = cache;
}
shared static this() {
_imageCache = new ImageCache();
_drawableCache = new DrawableCache();
}
class DrawableCache {
static class DrawableCacheItem {
string _id;
string _filename;
bool _tiled;
bool _error;
bool _used;
DrawableRef _drawable;
//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);
}
~this() {
_drawable.clear();
//Log.d("Destroyed DrawableCacheItem, count=", --_instanceCount);
}
/// remove from memory, will cause reload on next access
void compact() {
if (!_drawable.isNull)
_drawable.clear();
}
/// mark as not used
void checkpoint() {
_used = false;
}
/// cleanup if unused since last checkpoint
void cleanup() {
if (!_used)
compact();
}
/// returns drawable (loads from file if necessary)
@property ref DrawableRef drawable() {
_used = true;
if (!_drawable.isNull || _error)
return _drawable;
if (_filename !is null) {
// reload from file
if (_filename.endsWith(".xml")) {
// XML drawables support
StateDrawable d = new StateDrawable();
if (!d.load(_filename)) {
destroy(d);
_error = true;
} else {
_drawable = d;
}
} else {
// PNG/JPEG drawables support
DrawBufRef image = imageCache.get(_filename);
if (!image.isNull) {
bool ninePatch = _filename.endsWith(".9.png");
_drawable = new ImageDrawable(image, _tiled, ninePatch);
} else
_error = true;
}
}
return _drawable;
}
}
void clear() {
Log.d("DrawableCache.clear()");
_idToFileMap.clear();
foreach(DrawableCacheItem item; _idToDrawableMap)
item.drawable.clear();
_idToDrawableMap.clear();
}
// clear usage flags for all entries
void checkpoint() {
foreach (item; _idToDrawableMap)
item.checkpoint();
}
// removes entries not used after last call of checkpoint() or cleanup()
void cleanup() {
foreach (item; _idToDrawableMap)
item.cleanup();
}
string[] _resourcePaths;
string[string] _idToFileMap;
DrawableCacheItem[string] _idToDrawableMap;
ref DrawableRef get(string id) {
if (id in _idToDrawableMap)
return _idToDrawableMap[id].drawable;
string resourceId = id;
bool tiled = false;
if (id.endsWith(".tiled")) {
resourceId = id[0..$-6]; // remove .tiled
tiled = true;
}
string filename = findResource(resourceId);
DrawableCacheItem item = new DrawableCacheItem(id, filename, tiled);
_idToDrawableMap[id] = item;
return item.drawable;
}
@property string[] resourcePaths() {
return _resourcePaths;
}
@property void resourcePaths(string[] paths) {
_resourcePaths = paths;
clear();
}
/// concatenates path with resource id and extension, returns pathname if there is such file, null if file does not exist
private string checkFileName(string path, string id, string extension) {
char[] fn = path.dup;
fn ~= id;
fn ~= extension;
if (exists(fn) && isFile(fn))
return fn.dup;
return null;
}
string findResource(string id) {
if (id in _idToFileMap)
return _idToFileMap[id];
foreach(string path; _resourcePaths) {
string fn;
fn = checkFileName(path, id, ".xml");
if (fn is null)
fn = checkFileName(path, id, ".png");
if (fn is null)
fn = checkFileName(path, id, ".9.png");
if (fn is null)
fn = checkFileName(path, id, ".jpg");
if (fn !is null) {
_idToFileMap[id] = fn;
return fn;
}
}
return null;
}
this() {
Log.i("Creating DrawableCache");
}
~this() {
Log.i("Destroying DrawableCache");
foreach (ref item; _idToDrawableMap) {
destroy(item);
item = null;
}
_idToDrawableMap.clear();
}
}

View File

@ -3,7 +3,8 @@ module dlangui.widgets.styles;
import dlangui.core.types;
import dlangui.graphics.fonts;
import dlangui.graphics.drawbuf;
import dlangui.graphics.images;
//import dlangui.graphics.images;
import dlangui.graphics.resources;
immutable ubyte ALIGN_UNSPECIFIED = 0;
immutable uint COLOR_UNSPECIFIED = 0xFFDEADFF;
@ -20,22 +21,6 @@ immutable int FILL_PARENT = int.max - 1;
immutable int WRAP_CONTENT = int.max - 2;
immutable int WEIGHT_UNSPECIFIED = -1;
/// widget state flags - bits
enum State : uint {
/// state not specified / normal
Normal = 0,
Pressed = 1,
Focused = 2,
Enabled = 4,
Hovered = 8, // mouse pointer is over control, buttons not pressed
Selected = 16,
Checkable = 32,
Checked = 64,
Activated = 128,
WindowFocused = 256,
Parent = 0x10000, // use parent's state
}
enum Align : ubyte {
Unspecified = ALIGN_UNSPECIFIED,
Left = 1,
@ -658,11 +643,12 @@ Theme createDefaultTheme() {
tabUpButtonText.createState(State.Focused, State.Focused).textColor(0x000000);
tabUpButtonText.createState(State.Hovered, State.Hovered).textColor(0xFFE0E0);
Style tabUpButton = res.createSubstyle("TAB_UP_BUTTON");
tabUpButton.backgroundImageId("tab_btn_up_normal");
tabUpButton.createState(State.Selected, State.Selected).backgroundImageId("tab_btn_up_selected");
tabUpButton.createState(State.Selected|State.Focused, State.Selected|State.Focused).backgroundImageId("tab_btn_up_focused_selected");
tabUpButton.createState(State.Focused, State.Focused).backgroundImageId("tab_btn_up_focused");
tabUpButton.createState(State.Hovered, State.Hovered).backgroundImageId("tab_btn_up_hover");
tabUpButton.backgroundImageId("tab_btn_up");
//tabUpButton.backgroundImageId("tab_btn_up_normal");
//tabUpButton.createState(State.Selected, State.Selected).backgroundImageId("tab_btn_up_selected");
//tabUpButton.createState(State.Selected|State.Focused, State.Selected|State.Focused).backgroundImageId("tab_btn_up_focused_selected");
//tabUpButton.createState(State.Focused, State.Focused).backgroundImageId("tab_btn_up_focused");
//tabUpButton.createState(State.Hovered, State.Hovered).backgroundImageId("tab_btn_up_hover");
Style tabHost = res.createSubstyle("TAB_HOST");
tabHost.layoutWidth(FILL_PARENT).layoutHeight(FILL_PARENT);
tabHost.backgroundColor(0xF0F0F0);

View File

@ -4,7 +4,8 @@ public import dlangui.core.types;
public import dlangui.core.events;
public import dlangui.widgets.styles;
public import dlangui.graphics.drawbuf;
public import dlangui.graphics.images;
//public import dlangui.graphics.images;
public import dlangui.graphics.resources;
public import dlangui.graphics.fonts;
public import dlangui.core.i18n;