Merge branch 'master' of https://github.com/buggins/dlangui into again_adding_WIDGET_STYLE_CONSOLE

This commit is contained in:
Denis Feklushkin 2017-10-20 06:35:57 +07:00
commit 3a97a60ee8
8 changed files with 316 additions and 106 deletions

View File

@ -38,7 +38,8 @@
"./examples/tetris/", "./examples/tetris/",
"./examples/opengl/", "./examples/opengl/",
"./examples/ircclient/", "./examples/ircclient/",
"./examples/spreadsheet/" "./examples/spreadsheet/",
"./examples/dragon/"
], ],
"configurations": [ "configurations": [
@ -64,7 +65,7 @@
{ {
"name": "console", "name": "console",
"versions": ["USE_CONSOLE", "EmbedStandardResources"], "versions": ["USE_CONSOLE", "EmbedStandardResources"],
"excludedSourceFiles": ["3rdparty/*"] "excludedSourceFiles": ["3rdparty/*GL*", "3rdparty/android", "3rdparty/dimage", "3rdparty/fontconfig/*", "3rdparty/icontheme", "3rdparty/jni.d"]
}, },
{ {
"name": "external", "name": "external",

View File

@ -8,36 +8,33 @@ import dlangui.widgets.scroll;
class DragonView : ScrollWidget { class DragonView : ScrollWidget {
int _scaleX; private {
int _scaleY; int _scaleX;
int _middleX; int _scaleY;
int _middleY; int _middleX;
int _dx; int _middleY;
int _dy; int _dx;
int _x0; int _dy;
int _y0; int _x0;
int _length = 1000; int _y0;
int _dir0 = 0; // either 0 or 1 int _length = 256;
int _straightLen = 10; int _dir0 = 0; // either 0 or 1
int _roundLen = 4; int _straightLen = 10;
uint _bgcolor = 0x101010; int _roundLen = 4;
uint _grid1color = 0x303030; uint _bgcolor = 0x101010;
uint _grid2color = 0x202020; uint _grid1color = 0x303030;
uint _grid3color = 0x181818; uint _grid2color = 0x202020;
uint _curve1color = 0x4050FF; uint _grid3color = 0x181818;
uint _curve2color = 0xFF4040; uint _curve1color = 0x4050FF;
uint _curve3color = 0x30FF20; uint _curve2color = 0xFF4040;
uint _curve4color = 0xC000D0; uint _curve3color = 0x30C020;
Point[8] _directionVectors = [ uint _curve4color = 0xC000D0;
Point(4, 0), bool[4] _partVisible = [true, true, true, true];
Point(3, -3),
Point(0, -4), bool _needRepaint = true;
Point(-3, -3), Point[8] _directionVectors;
Point(-4, 0), }
Point(-3, 3),
Point(0, 4),
Point(3, 3),
];
ColorDrawBuf _drawBuf; ColorDrawBuf _drawBuf;
@ -55,7 +52,7 @@ class DragonView : ScrollWidget {
_dy = dy; _dy = dy;
_fullScrollableArea.right = dx; _fullScrollableArea.right = dx;
_fullScrollableArea.bottom = dy; _fullScrollableArea.bottom = dy;
_visibleScrollableArea.left = dx / 2 - 300; _visibleScrollableArea.left = dx / 2 - 400;
_visibleScrollableArea.top = dy / 2 - 300; _visibleScrollableArea.top = dy / 2 - 300;
_visibleScrollableArea.right = _visibleScrollableArea.left + 400; _visibleScrollableArea.right = _visibleScrollableArea.left + 400;
_visibleScrollableArea.bottom = _visibleScrollableArea.top + 400; _visibleScrollableArea.bottom = _visibleScrollableArea.top + 400;
@ -67,6 +64,7 @@ class DragonView : ScrollWidget {
} }
_middleX = _fullScrollableArea.width / 2; _middleX = _fullScrollableArea.width / 2;
_middleY = _fullScrollableArea.height / 2; _middleY = _fullScrollableArea.height / 2;
_needRepaint = true;
drawCurve(); drawCurve();
} }
@ -111,6 +109,18 @@ class DragonView : ScrollWidget {
} }
_x0 = vectors[1].x; _x0 = vectors[1].x;
_y0 = vectors[1].y; _y0 = vectors[1].y;
repaint();
}
bool getPartVisible(int n) {
return _partVisible[n & 3];
}
void setPartVisible(int n, bool flgVisible) {
n = n & 3;
if (_partVisible[n] != flgVisible) {
_partVisible[n] = flgVisible;
repaint();
}
} }
@property int straightLen() { @property int straightLen() {
@ -120,7 +130,6 @@ class DragonView : ScrollWidget {
if (_straightLen != n) { if (_straightLen != n) {
_straightLen = n; _straightLen = n;
setVectors(); setVectors();
drawCurve();
} }
return this; return this;
} }
@ -131,7 +140,6 @@ class DragonView : ScrollWidget {
if (_roundLen != n) { if (_roundLen != n) {
_roundLen = n; _roundLen = n;
setVectors(); setVectors();
drawCurve();
} }
return this; return this;
} }
@ -141,26 +149,38 @@ class DragonView : ScrollWidget {
@property DragonView length(int n) { @property DragonView length(int n) {
if (_length != n) { if (_length != n) {
_length = n; _length = n;
drawCurve(); repaint();
} }
return this; return this;
} }
void repaint() {
_needRepaint = true;
invalidate();
}
@property int rotation() { @property int rotation() {
return _dir0; return _dir0;
} }
@property DragonView rotation(int angle) { @property DragonView rotation(int angle) {
if (_dir0 != (angle & 7)) { if (_dir0 != (angle & 7)) {
_dir0 = angle & 7; _dir0 = angle & 7;
drawCurve(); _needRepaint = true;
repaint();
} }
return this; return this;
} }
/// Draw widget at its position to buffer
override void onDraw(DrawBuf buf) {
if (_needRepaint)
drawCurve();
super.onDraw(buf);
}
void drawLine(Point pt1, Point pt2, uint color) { void drawLine(Point pt1, Point pt2, uint color) {
pt1.x += _middleX; pt1.x += _middleX;
pt2.x += _middleX; pt2.x += _middleX;
pt1.y += _middleY; pt1.y = _middleY - pt1.y;
pt2.y += _middleY; pt2.y = _middleY - pt2.y;
_drawBuf.drawLine(pt1, pt2, color); _drawBuf.drawLine(pt1, pt2, color);
} }
@ -214,56 +234,63 @@ class DragonView : ScrollWidget {
} }
void drawSegment(ref Point currentPoint, ref int currentDir, int n, uint color, int mirror) { void drawSegment(ref Point currentPoint, ref int currentDir, int n, uint color, int mirror) {
int mx = _middleX + _scaleX * 2;
int my = _middleY + _scaleY * 2;
bool insideView = currentPoint.x >= -mx && currentPoint.x <= mx && currentPoint.y >= -my && currentPoint.y <= my;
int delta = getDirectionDelta(n) * mirror; int delta = getDirectionDelta(n) * mirror;
Point nextPoint = currentPoint + _directionVectors[currentDir]; Point nextPoint = currentPoint + _directionVectors[currentDir];
drawLine(currentPoint, nextPoint, color); if (insideView)
drawLine(currentPoint, nextPoint, color);
currentPoint = nextPoint; currentPoint = nextPoint;
currentDir = (currentDir + delta) & 7; currentDir = (currentDir + delta) & 7;
nextPoint = currentPoint + _directionVectors[currentDir]; nextPoint = currentPoint + _directionVectors[currentDir];
drawLine(currentPoint, nextPoint, color); if (insideView)
drawLine(currentPoint, nextPoint, color);
currentPoint = nextPoint; currentPoint = nextPoint;
currentDir = (currentDir + delta) & 7; currentDir = (currentDir + delta) & 7;
} }
void drawCurve() { void drawLines() {
drawBackground();
// segment 1
int dir; int dir;
Point p0; Point p0;
//Point p0 = Point(_directionVectors[_dir0].y, _directionVectors[_dir0 + 1].y ); Point pt;
if (_dir0 == 0) if (_dir0 == 0)
p0 = Point(0, _directionVectors[_dir0 + 1].y); p0 = Point(0, _directionVectors[_dir0 + 1].y);
else else
p0 = Point(-_directionVectors[0].x / 2, _directionVectors[_dir0 + 1].y / 2); p0 = Point(-_directionVectors[0].x / 2, _directionVectors[_dir0 + 1].y / 2);
//Point p0 = Point(-_directionVectors[0].x * 0, -_directionVectors[0].y / 2); // segment 1
Point pt; if (_partVisible[0]) {
///* dir = 0 + _dir0;
dir = 0 + _dir0; pt = p0 - (_directionVectors[dir] + _directionVectors[dir + 1]);
//Point pt = Point(_directionVectors[dir + 1].x - _scaleX, _directionVectors[dir].y); for(int i = 0; i < _length; i++)
pt = p0 - (_directionVectors[dir] + _directionVectors[dir + 1]); drawSegment(pt, dir, i, _curve1color, 1);
for(int i = 0; i < _length; i++) }
drawSegment(pt, dir, i, _curve1color, 1);
// segment 2 // segment 2
///* if (_partVisible[1]) {
dir = 4 + _dir0; dir = 4 + _dir0;
//pt = Point(-_directionVectors[dir + 1].x - _directionVectors[dir].x - _scaleX, _directionVectors[dir].y); pt = p0 + _directionVectors[dir + 1];
pt = p0 + _directionVectors[dir + 1];//_directionVectors[dir].y for(int i = -1; i > -_length; i--)
for(int i = -1; i > -_length; i--) drawSegment(pt, dir, i, _curve2color, -1);
drawSegment(pt, dir, i, _curve2color, -1); }
//*/
///*
// segment 3 // segment 3
dir = 4 + _dir0; if (_partVisible[2]) {
pt = p0 - (_directionVectors[dir - 1] + _directionVectors[dir]); dir = 4 + _dir0;
for(int i = 0; i < _length; i++) pt = p0 - (_directionVectors[dir - 1] + _directionVectors[dir]);
drawSegment(pt, dir, i, _curve3color, 1); for(int i = 0; i < _length; i++)
drawSegment(pt, dir, i, _curve3color, 1);
}
// segment 4 // segment 4
dir = 0 + _dir0; if (_partVisible[3]) {
pt = p0 + _directionVectors[(dir - 1) & 7]; dir = 0 + _dir0;
for(int i = -1; i > -_length; i--) pt = p0 + _directionVectors[(dir - 1) & 7];
drawSegment(pt, dir, i, _curve4color, -1); for(int i = -1; i > -_length; i--)
//*/ drawSegment(pt, dir, i, _curve4color, -1);
invalidate(); }
}
void drawCurve() {
drawBackground();
drawLines();
_needRepaint = false;
} }
/// calculate full content size in pixels /// calculate full content size in pixels
@ -289,13 +316,46 @@ class DragonView : ScrollWidget {
} }
immutable SLIDER_ACCELERATION_STEP = 64;
int sliderToSize(int n) {
int limit = 0;
int total = 0;
int factor = 1;
for(;;) {
int newlimit = limit + SLIDER_ACCELERATION_STEP;
if (n < newlimit) {
return total + (n - limit) * factor;
}
total += SLIDER_ACCELERATION_STEP * factor;
limit = newlimit;
factor *= 2;
}
}
int sizeToSlider(int n) {
int limit = 0;
int total = 0;
int factor = 1;
for(;;) {
int newlimit = limit + SLIDER_ACCELERATION_STEP;
int newtotal = total + SLIDER_ACCELERATION_STEP * factor;
if (n < newtotal) {
return limit + (n - total) / factor;
}
total = newtotal;
limit = newlimit;
factor *= 2;
}
}
/// entry point for dlangui based application /// entry point for dlangui based application
extern (C) int UIAppMain(string[] args) { extern (C) int UIAppMain(string[] args) {
// create window // create window
Log.d("Creating window"); Window window = Platform.instance.createWindow("DlangUI example : Dragon Curve"d, null, WindowFlag.Resizable, 800, 600);
Window window = Platform.instance.createWindow("DlangUI example - Dragon Curve", null); int n = sliderToSize(1000);
Log.d("Window created"); int n2 = sizeToSlider(n);
DragonView dragon = new DragonView("DRAGON_VIEW"); DragonView dragon = new DragonView("DRAGON_VIEW");
@ -309,7 +369,7 @@ extern (C) int UIAppMain(string[] args) {
dragon.roundLen = event.position; dragon.roundLen = event.position;
break; break;
case "size": case "size":
dragon.length = event.position; dragon.length = sliderToSize(event.position);
break; break;
default: default:
break; break;
@ -338,10 +398,30 @@ extern (C) int UIAppMain(string[] args) {
cbRotate.checkChange = delegate(Widget w, bool check) { cbRotate.checkChange = delegate(Widget w, bool check) {
dragon.rotation(check ? 1 : 0); return true; dragon.rotation(check ? 1 : 0); return true;
}; };
auto cbPartVisible0 = new CheckBox(null, " A1"d);
controls1.addChild(cbPartVisible0).checked(dragon.getPartVisible(0));
cbPartVisible0.checkChange = delegate(Widget w, bool check) {
dragon.setPartVisible(0, check); return true;
};
auto cbPartVisible1 = new CheckBox(null, " A2"d);
controls1.addChild(cbPartVisible1).checked(dragon.getPartVisible(1));
cbPartVisible1.checkChange = delegate(Widget w, bool check) {
dragon.setPartVisible(1, check); return true;
};
auto cbPartVisible2 = new CheckBox(null, " B1"d);
controls1.addChild(cbPartVisible2).checked(dragon.getPartVisible(2));
cbPartVisible2.checkChange = delegate(Widget w, bool check) {
dragon.setPartVisible(2, check); return true;
};
auto cbPartVisible3 = new CheckBox(null, " B2"d);
controls1.addChild(cbPartVisible3).checked(dragon.getPartVisible(3));
cbPartVisible3.checkChange = delegate(Widget w, bool check) {
dragon.setPartVisible(3, check); return true;
};
controls1.addChild(new TextWidget(null," Size"d)); controls1.addChild(new TextWidget(null," Size"d));
auto sliderSize = new SliderWidget("size"); auto sliderSize = new SliderWidget("size");
sliderSize.setRange(2, 10000).position(dragon.length).layoutWeight(10).fillHorizontal; sliderSize.setRange(2, 1000).position(sizeToSlider(dragon.length)).layoutWeight(10).fillHorizontal;
sliderSize.scrollEvent = onScrollEvent; sliderSize.scrollEvent = onScrollEvent;
controls1.addChild(sliderSize); controls1.addChild(sliderSize);

View File

@ -166,7 +166,7 @@ struct Listener(RETURN_T, T1...)
} }
/// disconnect all listeners /// disconnect all listeners
final void clear() { final void clear() {
_listener = null; _listener = null;
} }
alias get this; alias get this;
} }
@ -241,7 +241,7 @@ struct Signal(T1) if (is(T1 == interface) && __traits(allMembers, T1).length ==
} }
/// disconnect all listeners /// disconnect all listeners
final void clear() { final void clear() {
_listeners.clear(); _listeners.clear();
} }
} }
@ -297,6 +297,6 @@ struct Signal(RETURN_T, T1...)
} }
/// disconnect all listeners /// disconnect all listeners
final void clear() { final void clear() {
_listeners.clear(); _listeners.clear();
} }
} }

View File

@ -513,6 +513,55 @@ struct Ref(T) { // if (T is RefCountedObject)
} }
/**
This struct allows to not execute some code if some variables was not changed since the last check.
Used for optimizations.
Reference types, arrays and pointers are compared by reference.
*/
struct CalcSaver(Params...) {
import std.typecons : Tuple;
Tuple!Params values;
bool check(Params args) {
bool changed;
foreach (i, arg; args) {
if (values[i] !is arg) {
values[i] = arg;
changed = true;
}
}
return changed;
}
}
///
unittest {
class A { }
CalcSaver!(uint, double[], A) saver;
uint x = 5;
double[] arr = [1, 2, 3];
A a = new A();
assert(saver.check(x, arr, a));
// values are not changing
assert(!saver.check(x, arr, a));
assert(!saver.check(x, arr, a));
assert(!saver.check(x, arr, a));
assert(!saver.check(x, arr, a));
x = 8;
arr ~= 25;
a = new A();
// values are changed
assert(saver.check(x, arr, a));
assert(!saver.check(x, arr, a));
}
//================================================================================ //================================================================================
// some utility functions // some utility functions
@ -594,3 +643,76 @@ dstring normalizeEndOfLineCharacters(dstring s) {
} }
return cast(dstring)res; return cast(dstring)res;
} }
/// C malloc allocated array wrapper
struct MallocBuf(T) {
import core.stdc.stdlib : realloc, free;
private T * _allocated;
private uint _allocatedSize;
private uint _length;
/// get pointer
@property T * ptr() { return _allocated; }
/// get length
@property uint length() { return _length; }
/// set new length
@property void length(uint len) {
if (len > _allocatedSize) {
reserve(_allocatedSize ? len * 2 : len);
}
_length = len;
}
/// const array[index];
T opIndex(uint index) const {
assert(index < _length);
return _allocated[index];
}
/// ref array[index];
ref T opIndex(uint index) {
assert(index < _length);
return _allocated[index];
}
/// array[index] = value;
void opIndexAssign(uint index, T value) {
assert(index < _length);
_allocated[index] = value;
}
/// array[index] = value;
void opIndexAssign(uint index, T[] values) {
assert(index + values.length < _length);
_allocated[index .. index + values.length] = values[];
}
/// array[a..b]
T[] opSlice(uint a, uint b) {
assert(a <= b && b <= _length);
return _allocated[a .. b];
}
/// array[]
T[] opSlice() {
return _allocated ? _allocated[0 .. _length] : null;
}
/// array[$]
uint opDollar() { return _length; }
~this() {
clear();
}
/// free allocated memory, set length to 0
void clear() {
if (_allocated)
free(_allocated);
_allocatedSize = 0;
_length = 0;
}
/// make sure buffer capacity is at least (size) items
void reserve(uint size) {
if (_allocatedSize < size) {
_allocated = cast(T*)realloc(_allocated, T.sizeof * size);
_allocatedSize = size;
}
}
/// fill buffer with specified value
void fill(T value) {
if (_length) {
_allocated[0 .. _length] = value;
}
}
}

View File

@ -1233,14 +1233,15 @@ class ColorDrawBufBase : DrawBuf {
} }
class GrayDrawBuf : DrawBuf { class GrayDrawBuf : DrawBuf {
int _dx; protected int _dx;
int _dy; protected int _dy;
/// returns buffer bits per pixel /// returns buffer bits per pixel
override @property int bpp() { return 8; } override @property int bpp() { return 8; }
@property override int width() { return _dx; } @property override int width() { return _dx; }
@property override int height() { return _dy; } @property override int height() { return _dy; }
ubyte[] _buf; protected MallocBuf!ubyte _buf;
this(int width, int height) { this(int width, int height) {
resize(width, height); resize(width, height);
} }
@ -1473,7 +1474,7 @@ class GrayDrawBuf : DrawBuf {
} }
class ColorDrawBuf : ColorDrawBufBase { class ColorDrawBuf : ColorDrawBufBase {
uint[] _buf; protected MallocBuf!uint _buf;
/// create ARGB8888 draw buf of specified width and height /// create ARGB8888 draw buf of specified width and height
this(int width, int height) { this(int width, int height) {
@ -1482,9 +1483,8 @@ class ColorDrawBuf : ColorDrawBufBase {
/// create copy of ColorDrawBuf /// create copy of ColorDrawBuf
this(ColorDrawBuf v) { this(ColorDrawBuf v) {
this(v.width, v.height); this(v.width, v.height);
//_buf.length = v._buf.length; if (auto len = _buf.length)
foreach(i; 0 .. _buf.length) _buf.ptr[0 .. len] = v._buf.ptr[0 .. len];
_buf[i] = v._buf[i];
} }
/// create resized copy of ColorDrawBuf /// create resized copy of ColorDrawBuf
this(ColorDrawBuf v, int dx, int dy) { this(ColorDrawBuf v, int dx, int dy) {
@ -1494,7 +1494,7 @@ class ColorDrawBuf : ColorDrawBufBase {
} }
void invertAndPreMultiplyAlpha() { void invertAndPreMultiplyAlpha() {
foreach(ref pixel; _buf) { foreach(ref pixel; _buf[]) {
uint a = (pixel >> 24) & 0xFF; uint a = (pixel >> 24) & 0xFF;
uint r = (pixel >> 16) & 0xFF; uint r = (pixel >> 16) & 0xFF;
uint g = (pixel >> 8) & 0xFF; uint g = (pixel >> 8) & 0xFF;
@ -1510,12 +1510,12 @@ class ColorDrawBuf : ColorDrawBufBase {
} }
void invertAlpha() { void invertAlpha() {
foreach(ref pixel; _buf) foreach(ref pixel; _buf[])
pixel ^= 0xFF000000; pixel ^= 0xFF000000;
} }
void invertByteOrder() { void invertByteOrder() {
foreach(ref pixel; _buf) { foreach(ref pixel; _buf[]) {
pixel = (pixel & 0xFF00FF00) | pixel = (pixel & 0xFF00FF00) |
((pixel & 0xFF0000) >> 16) | ((pixel & 0xFF0000) >> 16) |
((pixel & 0xFF) << 16); ((pixel & 0xFF) << 16);
@ -1524,7 +1524,7 @@ class ColorDrawBuf : ColorDrawBufBase {
// for passing of image to OpenGL texture // for passing of image to OpenGL texture
void invertAlphaAndByteOrder() { void invertAlphaAndByteOrder() {
foreach(ref pixel; _buf) { foreach(ref pixel; _buf[]) {
pixel = ((pixel & 0xFF00FF00) | pixel = ((pixel & 0xFF00FF00) |
((pixel & 0xFF0000) >> 16) | ((pixel & 0xFF0000) >> 16) |
((pixel & 0xFF) << 16)); ((pixel & 0xFF) << 16));
@ -1643,9 +1643,9 @@ class ColorDrawBuf : ColorDrawBufBase {
uint[] tmpbuf; uint[] tmpbuf;
tmpbuf.length = _buf.length; tmpbuf.length = _buf.length;
// do horizontal blur // do horizontal blur
blurOneDimension(_buf, tmpbuf, blurSize, true); blurOneDimension(_buf[], tmpbuf, blurSize, true);
// then do vertical blur // then do vertical blur
blurOneDimension(tmpbuf, _buf, blurSize, false); blurOneDimension(tmpbuf, _buf[], blurSize, false);
} }
} }

View File

@ -115,20 +115,26 @@ class TextWidget : Widget {
return this; return this;
} }
override void measure(int parentWidth, int parentHeight) { private CalcSaver!(Font, dstring, uint, uint) _measureSaver;
override void measure(int parentWidth, int parentHeight) {
FontRef font = font(); FontRef font = font();
//auto measureStart = std.datetime.Clock.currAppTick; uint w = (maxLines == 1) ? MAX_WIDTH_UNSPECIFIED :
Point sz; parentWidth - margins.left - margins.right - padding.left - padding.right;
if (maxLines == 1) { uint flags = textFlags;
sz = font.textSize(text, MAX_WIDTH_UNSPECIFIED, 4, 0, textFlags);
} else { // optimization: do not measure if nothing changed
sz = font.measureMultilineText(text,maxLines,parentWidth-margins.left-margins.right-padding.left-padding.right, 4, 0, textFlags); if (_measureSaver.check(font.get, text, w, flags) || _needLayout) {
Point sz;
if (maxLines == 1) {
sz = font.textSize(text, w, 4, 0, flags);
} else {
sz = font.measureMultilineText(text, maxLines, w, 4, 0, flags);
}
// it's not very correct, but in such simple widget it doesn't make issues
measuredContent(SIZE_UNSPECIFIED, SIZE_UNSPECIFIED, sz.x, sz.y);
_needLayout = false;
} }
//auto measureEnd = std.datetime.Clock.currAppTick;
//auto duration = measureEnd - measureStart;
//if (duration.length > 10)
// Log.d("TextWidget measureText took ", duration.length, " ticks");
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
} }
override void onDraw(DrawBuf buf) { override void onDraw(DrawBuf buf) {

View File

@ -643,7 +643,7 @@ class FrameLayout : WidgetGroupDefaultDrawing {
applyPadding(rc); applyPadding(rc);
for (int i = 0; i < _children.count; i++) { for (int i = 0; i < _children.count; i++) {
Widget item = _children.get(i); Widget item = _children.get(i);
if (item.visibility != Visibility.Gone) { if (item.visibility == Visibility.Visible) {
item.layout(rc); item.layout(rc);
} }
} }
@ -657,6 +657,7 @@ class FrameLayout : WidgetGroupDefaultDrawing {
Widget item = _children.get(i); Widget item = _children.get(i);
if (item.compareId(ID)) { if (item.compareId(ID)) {
item.visibility = Visibility.Visible; item.visibility = Visibility.Visible;
item.requestLayout();
foundWidget = item; foundWidget = item;
found = true; found = true;
} else { } else {

View File

@ -1 +1 @@
v0.9.167 v0.9.169