mirror of https://github.com/buggins/dlangui.git
performace optimizations
This commit is contained in:
parent
90aa9bbb49
commit
970928d5f6
|
@ -36,7 +36,8 @@
|
|||
<OutputName>dlanguilib</OutputName>
|
||||
<ObjectsDirectory>obj/Release</ObjectsDirectory>
|
||||
<Externalconsole>true</Externalconsole>
|
||||
<Target>Executable</Target>
|
||||
<Target>StaticLibrary</Target>
|
||||
<ExtraCompilerArguments>-version=USE_OPENGL</ExtraCompilerArguments>
|
||||
</PropertyGroup>
|
||||
<PropertyGroup Condition=" '$(Configuration)|$(Platform)' == 'Unittest|AnyCPU' ">
|
||||
<DebugSymbols>true</DebugSymbols>
|
||||
|
@ -45,7 +46,7 @@
|
|||
<ObjectsDirectory>obj/Unittest</ObjectsDirectory>
|
||||
<OutputName>dlanguilib</OutputName>
|
||||
<Externalconsole>true</Externalconsole>
|
||||
<Target>Executable</Target>
|
||||
<Target>StaticLibrary</Target>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="src\dlangui\core\logger.d" />
|
||||
|
@ -61,9 +62,6 @@
|
|||
<Compile Include="src\dlangui\widgets\layouts.d" />
|
||||
<Compile Include="src\dlangui\widgets\styles.d" />
|
||||
<Compile Include="src\dlangui\widgets\widget.d" />
|
||||
<Compile Include="3rdparty\libpng\source\libpng\png.d" />
|
||||
<Compile Include="3rdparty\libpng\source\libpng\pngconf.d" />
|
||||
<Compile Include="3rdparty\libpng\source\libpng\pnglibconf.d" />
|
||||
<Compile Include="src\dlangui\platforms\x11\x11app.d" />
|
||||
<Compile Include="..\DerelictFT\source\derelict\freetype\ft.d">
|
||||
<Link>3rdparty\DerelictFT\ft.d</Link>
|
||||
|
|
|
@ -5,7 +5,7 @@ import std.stdio;
|
|||
import std.conv;
|
||||
|
||||
version (linux) {
|
||||
pragma(lib, "png");
|
||||
//pragma(lib, "png");
|
||||
pragma(lib, "xcb");
|
||||
pragma(lib, "xcb-shm");
|
||||
pragma(lib, "xcb-image");
|
||||
|
|
|
@ -80,6 +80,7 @@ class Window {
|
|||
_mainWidget.layout(Rect(0, 0, _dx, _dy));
|
||||
long layoutEnd = currentTimeMillis;
|
||||
Log.d("layout took ", layoutEnd - measureEnd, " ms");
|
||||
checkUpdateNeeded(needDraw, needLayout, animationActive);
|
||||
}
|
||||
long drawStart = currentTimeMillis;
|
||||
_mainWidget.onDraw(buf);
|
||||
|
|
|
@ -27,7 +27,14 @@ version(linux) {
|
|||
|
||||
import derelict.opengl3.gl3;
|
||||
import derelict.opengl3.glx;
|
||||
|
||||
|
||||
// pragma(lib, "xcb");
|
||||
// pragma(lib, "xcb-shm");
|
||||
// pragma(lib, "xcb-image");
|
||||
// pragma(lib, "X11-xcb");
|
||||
// pragma(lib, "X11");
|
||||
// pragma(lib, "dl");
|
||||
|
||||
extern (System)
|
||||
xcb_connection_t *XGetXCBConnection(std.c.linux.X11.Xlib.Display *dpy);
|
||||
enum XEventQueueOwner { XlibOwnsEventQueue = 0, XCBOwnsEventQueue };
|
||||
|
@ -160,12 +167,20 @@ version(linux) {
|
|||
windowCaption = _caption;
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
private int _imageDx;
|
||||
private int _imageDy;
|
||||
void createImage() {
|
||||
Log.i("CRXCBScreen::createImage ", _dx, "x", _dy);
|
||||
if (_image)
|
||||
if (_image) {
|
||||
if (_imageDx == _imageDx && _imageDy == _dy)
|
||||
return; // already have image of proper size
|
||||
Log.i("CRXCBScreen::createImage - destroying existing image");
|
||||
xcb_image_destroy(_image);
|
||||
_image = null;
|
||||
_image = null;
|
||||
}
|
||||
_imageDx = _dx;
|
||||
_imageDy = _dy;
|
||||
Log.i("CRXCBScreen::createImage ", _dx, "x", _dy);
|
||||
xcb_shm_query_version_reply_t * rep_shm;
|
||||
rep_shm = xcb_shm_query_version_reply (_xcbconnection,
|
||||
xcb_shm_query_version(_xcbconnection),
|
||||
|
@ -403,9 +418,9 @@ version(linux) {
|
|||
_drawbuf = new ColorDrawBuf(_dx, _dy);
|
||||
_drawbuf.resize(_dx, _dy);
|
||||
_drawbuf.fill(_backgroundColor);
|
||||
Log.d("calling createImage");
|
||||
//Log.d("calling createImage");
|
||||
createImage();
|
||||
Log.d("done createImage");
|
||||
//Log.d("done createImage");
|
||||
onDraw(_drawbuf);
|
||||
draw(_drawbuf);
|
||||
/*
|
||||
|
|
|
@ -456,8 +456,32 @@ class ScrollBar : AbstractSlider, OnClickHandler {
|
|||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
|
||||
override protected void onPositionChanged() {
|
||||
if (!needLayout)
|
||||
layoutButtons();
|
||||
}
|
||||
|
||||
protected void layoutButtons(Rect irc) {
|
||||
protected void layoutButtons() {
|
||||
Rect irc = _scrollArea;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// vertical
|
||||
int spaceBackSize, spaceForwardSize, indicatorSize;
|
||||
bool indicatorVisible = calcButtonSizes(_scrollArea.height, spaceBackSize, spaceForwardSize, indicatorSize);
|
||||
irc.top += spaceBackSize;
|
||||
irc.bottom -= spaceForwardSize;
|
||||
layoutButtons(irc);
|
||||
} else {
|
||||
// horizontal
|
||||
int spaceBackSize, spaceForwardSize, indicatorSize;
|
||||
bool indicatorVisible = calcButtonSizes(_scrollArea.width, spaceBackSize, spaceForwardSize, indicatorSize);
|
||||
irc.left += spaceBackSize;
|
||||
irc.right -= spaceForwardSize;
|
||||
layoutButtons(irc);
|
||||
}
|
||||
}
|
||||
|
||||
protected void layoutButtons(Rect irc) {
|
||||
Rect r;
|
||||
_indicator.visibility = Visibility.Visible;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
|
@ -500,6 +524,7 @@ class ScrollBar : AbstractSlider, OnClickHandler {
|
|||
}
|
||||
|
||||
override void layout(Rect rc) {
|
||||
_needLayout = false;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
Rect r;
|
||||
|
@ -519,12 +544,6 @@ class ScrollBar : AbstractSlider, OnClickHandler {
|
|||
r.top = backbtnpos;
|
||||
r.bottom = fwdbtnpos;
|
||||
_scrollArea = r;
|
||||
int spaceBackSize, spaceForwardSize, indicatorSize;
|
||||
bool indicatorVisible = calcButtonSizes(r.height, spaceBackSize, spaceForwardSize, indicatorSize);
|
||||
Rect irc = r;
|
||||
irc.top += spaceBackSize;
|
||||
irc.bottom -= spaceForwardSize;
|
||||
layoutButtons(irc);
|
||||
} else {
|
||||
// horizontal
|
||||
int backbtnpos = rc.left + _btnSize;
|
||||
|
@ -540,15 +559,9 @@ class ScrollBar : AbstractSlider, OnClickHandler {
|
|||
r.left = backbtnpos;
|
||||
r.right = fwdbtnpos;
|
||||
_scrollArea = r;
|
||||
int spaceBackSize, spaceForwardSize, indicatorSize;
|
||||
bool indicatorVisible = calcButtonSizes(r.width, spaceBackSize, spaceForwardSize, indicatorSize);
|
||||
Rect irc = r;
|
||||
irc.left += spaceBackSize;
|
||||
irc.right -= spaceForwardSize;
|
||||
layoutButtons(irc);
|
||||
}
|
||||
layoutButtons();
|
||||
_pos = rc;
|
||||
_needLayout = false;
|
||||
}
|
||||
|
||||
override bool onClick(Widget source) {
|
||||
|
|
|
@ -1,378 +1,377 @@
|
|||
module dlangui.widgets.layouts;
|
||||
|
||||
public import dlangui.widgets.widget;
|
||||
|
||||
/// helper for layouts
|
||||
struct LayoutItem {
|
||||
Widget _widget;
|
||||
Orientation _orientation;
|
||||
int _measuredSize; // primary size for orientation
|
||||
int _secondarySize; // other measured size
|
||||
int _layoutSize; // layout size for primary dimension
|
||||
int _minSize; // min size for primary dimension
|
||||
int _maxSize; // max size for primary dimension
|
||||
int _weight; // weight
|
||||
bool _fillParent;
|
||||
@property int measuredSize() { return _measuredSize; }
|
||||
@property int minSize() { return _measuredSize; }
|
||||
@property int maxSize() { return _maxSize; }
|
||||
@property int layoutSize() { return _layoutSize; }
|
||||
@property int secondarySize() { return _layoutSize; }
|
||||
@property bool fillParent() { return _fillParent; }
|
||||
@property int weight() { return _weight; }
|
||||
// just to help GC
|
||||
void clear() {
|
||||
_widget = null;
|
||||
}
|
||||
/// sets item for widget
|
||||
void set(Widget widget, Orientation orientation) {
|
||||
_widget = widget;
|
||||
_orientation = orientation;
|
||||
}
|
||||
/// set item and measure it
|
||||
void measure(int parentWidth, int parentHeight) {
|
||||
_widget.measure(parentWidth, parentHeight);
|
||||
_weight = _widget.layoutWeight;
|
||||
if (_orientation == Orientation.Horizontal) {
|
||||
_secondarySize = _widget.measuredHeight;
|
||||
_measuredSize = _widget.measuredWidth;
|
||||
_minSize = _widget.minWidth;
|
||||
_maxSize = _widget.maxWidth;
|
||||
_layoutSize = _widget.layoutWidth;
|
||||
} else {
|
||||
_secondarySize = _widget.measuredWidth;
|
||||
_measuredSize = _widget.measuredHeight;
|
||||
_minSize = _widget.minHeight;
|
||||
_maxSize = _widget.maxHeight;
|
||||
_layoutSize = _widget.layoutHeight;
|
||||
}
|
||||
_fillParent = _layoutSize == FILL_PARENT;
|
||||
}
|
||||
void layout(ref Rect rc) {
|
||||
_widget.layout(rc);
|
||||
}
|
||||
}
|
||||
|
||||
/// helper class for layouts
|
||||
class LayoutItems {
|
||||
Orientation _orientation;
|
||||
LayoutItem[] _list;
|
||||
int _count;
|
||||
int _totalSize;
|
||||
int _maxSecondarySize;
|
||||
Point _measureParentSize;
|
||||
|
||||
int _layoutWidth;
|
||||
int _layoutHeight;
|
||||
|
||||
void setLayoutParams(Orientation orientation, int layoutWidth, int layoutHeight) {
|
||||
_orientation = orientation;
|
||||
_layoutWidth = layoutWidth;
|
||||
_layoutHeight = layoutHeight;
|
||||
}
|
||||
|
||||
/// fill widget layout list with Visible or Invisible items, measure them
|
||||
Point measure(int parentWidth, int parentHeight) {
|
||||
_totalSize = 0;
|
||||
_maxSecondarySize = 0;
|
||||
_measureParentSize.x = parentWidth;
|
||||
_measureParentSize.y = parentHeight;
|
||||
// measure
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
item.measure(parentWidth, parentHeight);
|
||||
if (_maxSecondarySize < item._secondarySize)
|
||||
_maxSecondarySize = item._secondarySize;
|
||||
_totalSize += item._measuredSize;
|
||||
}
|
||||
return _orientation == Orientation.Horizontal ? Point(_totalSize, _maxSecondarySize) : Point(_maxSecondarySize, _totalSize);
|
||||
}
|
||||
|
||||
/// fill widget layout list with Visible or Invisible items, measure them
|
||||
void setWidgets(ref WidgetList widgets) {
|
||||
// remove old items, if any
|
||||
clear();
|
||||
// reserve space
|
||||
if (_list.length < widgets.count)
|
||||
_list.length = widgets.count;
|
||||
// copy
|
||||
for (int i = 0; i < widgets.count; i++) {
|
||||
Widget item = widgets.get(i);
|
||||
if (item.visibility == Visibility.Gone)
|
||||
continue;
|
||||
_list[_count++].set(item, _orientation);
|
||||
}
|
||||
}
|
||||
|
||||
void layout(Rect rc) {
|
||||
// measure again - available area could be changed
|
||||
if (_measureParentSize.x != rc.width || _measureParentSize.y != rc.height)
|
||||
measure(rc.width, rc.height);
|
||||
int contentSecondarySize = 0;
|
||||
int contentHeight = 0;
|
||||
int totalSize = 0;
|
||||
int delta = 0;
|
||||
int resizableSize = 0;
|
||||
int resizableWeight = 0;
|
||||
int nonresizableSize = 0;
|
||||
int nonresizableWeight = 0;
|
||||
int maxItem = 0; // max item dimention
|
||||
// calc total size
|
||||
int visibleCount = cast(int)_list.length;
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
int weight = item.weight;
|
||||
int size = item.measuredSize;
|
||||
totalSize += size;
|
||||
if (maxItem < item.secondarySize)
|
||||
maxItem = item.secondarySize;
|
||||
if (item.fillParent) {
|
||||
resizableWeight += weight;
|
||||
resizableSize += size * weight;
|
||||
} else {
|
||||
nonresizableWeight += weight;
|
||||
nonresizableSize += size * weight;
|
||||
}
|
||||
}
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
if (_layoutWidth == WRAP_CONTENT && maxItem < rc.width)
|
||||
contentSecondarySize = maxItem;
|
||||
else
|
||||
contentSecondarySize = rc.width;
|
||||
if (_layoutHeight == FILL_PARENT || totalSize > rc.height)
|
||||
delta = rc.height - totalSize; // total space to add to fit
|
||||
} else {
|
||||
if (_layoutHeight == WRAP_CONTENT && maxItem < rc.height)
|
||||
contentSecondarySize = maxItem;
|
||||
else
|
||||
contentSecondarySize = rc.height;
|
||||
if (_layoutWidth == FILL_PARENT || totalSize > rc.width)
|
||||
delta = rc.width - totalSize; // total space to add to fit
|
||||
}
|
||||
// calculate resize options and scale
|
||||
bool needForceResize = false;
|
||||
bool needResize = false;
|
||||
int scaleFactor = 10000; // per weight unit
|
||||
if (delta != 0 && visibleCount > 0) {
|
||||
// need resize of some children
|
||||
needResize = true;
|
||||
// resize all if need to shrink or only resizable are too small to correct delta
|
||||
needForceResize = delta < 0 || resizableWeight == 0; // || resizableSize * 2 / 3 < delta; // do we need resize non-FILL_PARENT items?
|
||||
// calculate scale factor: weight / delta * 10000
|
||||
if (needForceResize)
|
||||
scaleFactor = 10000 * delta / (nonresizableSize + resizableSize);
|
||||
else
|
||||
scaleFactor = 10000 * delta / resizableSize;
|
||||
}
|
||||
//Log.d("VerticalLayout delta=", delta, ", nonres=", nonresizableWeight, ", res=", resizableWeight, ", scale=", scaleFactor);
|
||||
// find last resized - to allow fill space 1 pixel accurate
|
||||
int lastResized = -1;
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
if (item.fillParent || needForceResize) {
|
||||
lastResized = i;
|
||||
}
|
||||
}
|
||||
// final resize and layout of children
|
||||
int position = 0;
|
||||
int deltaTotal = 0;
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
int layoutSize = item.layoutSize;
|
||||
int weight = item.weight;
|
||||
int size = item.measuredSize;
|
||||
if (needResize && (layoutSize == FILL_PARENT || needForceResize)) {
|
||||
// do resize
|
||||
int correction = scaleFactor * weight * size / 10000;
|
||||
deltaTotal += correction;
|
||||
// for last resized, apply additional correction to resolve calculation inaccuracy
|
||||
if (i == lastResized) {
|
||||
correction += delta - deltaTotal;
|
||||
}
|
||||
size += correction;
|
||||
}
|
||||
// apply size
|
||||
Rect childRect = rc;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
childRect.top += position;
|
||||
childRect.bottom = childRect.top + size;
|
||||
childRect.right = childRect.left + contentSecondarySize;
|
||||
item.layout(childRect);
|
||||
} else {
|
||||
// Horizontal
|
||||
childRect.left += position;
|
||||
childRect.right = childRect.left + size;
|
||||
childRect.bottom = childRect.top + contentSecondarySize;
|
||||
item.layout(childRect);
|
||||
}
|
||||
position += size;
|
||||
}
|
||||
}
|
||||
|
||||
void clear() {
|
||||
for (int i = 0; i < _count; i++)
|
||||
_list[i].clear();
|
||||
_count = 0;
|
||||
}
|
||||
~this() {
|
||||
clear();
|
||||
}
|
||||
}
|
||||
|
||||
class LinearLayout : WidgetGroup {
|
||||
protected Orientation _orientation = Orientation.Vertical;
|
||||
/// returns linear layout orientation (Vertical, Horizontal)
|
||||
@property Orientation orientation() { return _orientation; }
|
||||
/// sets linear layout orientation
|
||||
@property LinearLayout orientation(Orientation value) { _orientation = value; requestLayout(); return this; }
|
||||
|
||||
this(string ID = null) {
|
||||
super(ID);
|
||||
_layoutItems = new LayoutItems();
|
||||
}
|
||||
|
||||
LayoutItems _layoutItems;
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
// measure children
|
||||
_layoutItems.setLayoutParams(orientation, layoutWidth, layoutHeight);
|
||||
_layoutItems.setWidgets(_children);
|
||||
Point sz = _layoutItems.measure(pwidth, pheight);
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
if (visibility == Visibility.Gone) {
|
||||
_needLayout = false;
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
_layoutItems.layout(rc);
|
||||
_needLayout = false;
|
||||
}
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Visible)
|
||||
continue;
|
||||
item.onDraw(buf);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
class VerticalLayout : LinearLayout {
|
||||
this(string ID = null) {
|
||||
super(ID);
|
||||
orientation = Orientation.Vertical;
|
||||
}
|
||||
}
|
||||
|
||||
class HorizontalLayout : LinearLayout {
|
||||
this(string ID = null) {
|
||||
super(ID);
|
||||
orientation = Orientation.Horizontal;
|
||||
}
|
||||
}
|
||||
|
||||
/// place all children into same place (usually, only one child should be visible at a time)
|
||||
class FrameLayout : WidgetGroup {
|
||||
this(string ID) {
|
||||
super(ID);
|
||||
}
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
// measure children
|
||||
Point sz;
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Gone) {
|
||||
item.measure(pwidth, pheight);
|
||||
if (sz.x < item.measuredWidth)
|
||||
sz.x = item.measuredWidth;
|
||||
if (sz.y < item.measuredHeight)
|
||||
sz.y = item.measuredHeight;
|
||||
}
|
||||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
if (visibility == Visibility.Gone) {
|
||||
_needLayout = false;
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Gone) {
|
||||
item.layout(rc);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Visible)
|
||||
continue;
|
||||
item.onDraw(buf);
|
||||
}
|
||||
}
|
||||
|
||||
/// make one of children (with specified ID) visible, for the rest, set visibility to otherChildrenVisibility
|
||||
bool showChild(string ID, Visibility otherChildrenVisibility = Visibility.Invisible) {
|
||||
bool found = false;
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.compareId(ID)) {
|
||||
item.visibility = Visibility.Visible;
|
||||
found = true;
|
||||
} else {
|
||||
item.visibility = otherChildrenVisibility;
|
||||
}
|
||||
}
|
||||
return found;
|
||||
}
|
||||
}
|
||||
module dlangui.widgets.layouts;
|
||||
|
||||
public import dlangui.widgets.widget;
|
||||
|
||||
/// helper for layouts
|
||||
struct LayoutItem {
|
||||
Widget _widget;
|
||||
Orientation _orientation;
|
||||
int _measuredSize; // primary size for orientation
|
||||
int _secondarySize; // other measured size
|
||||
int _layoutSize; // layout size for primary dimension
|
||||
int _minSize; // min size for primary dimension
|
||||
int _maxSize; // max size for primary dimension
|
||||
int _weight; // weight
|
||||
bool _fillParent;
|
||||
@property int measuredSize() { return _measuredSize; }
|
||||
@property int minSize() { return _measuredSize; }
|
||||
@property int maxSize() { return _maxSize; }
|
||||
@property int layoutSize() { return _layoutSize; }
|
||||
@property int secondarySize() { return _layoutSize; }
|
||||
@property bool fillParent() { return _fillParent; }
|
||||
@property int weight() { return _weight; }
|
||||
// just to help GC
|
||||
void clear() {
|
||||
_widget = null;
|
||||
}
|
||||
/// sets item for widget
|
||||
void set(Widget widget, Orientation orientation) {
|
||||
_widget = widget;
|
||||
_orientation = orientation;
|
||||
}
|
||||
/// set item and measure it
|
||||
void measure(int parentWidth, int parentHeight) {
|
||||
_widget.measure(parentWidth, parentHeight);
|
||||
_weight = _widget.layoutWeight;
|
||||
if (_orientation == Orientation.Horizontal) {
|
||||
_secondarySize = _widget.measuredHeight;
|
||||
_measuredSize = _widget.measuredWidth;
|
||||
_minSize = _widget.minWidth;
|
||||
_maxSize = _widget.maxWidth;
|
||||
_layoutSize = _widget.layoutWidth;
|
||||
} else {
|
||||
_secondarySize = _widget.measuredWidth;
|
||||
_measuredSize = _widget.measuredHeight;
|
||||
_minSize = _widget.minHeight;
|
||||
_maxSize = _widget.maxHeight;
|
||||
_layoutSize = _widget.layoutHeight;
|
||||
}
|
||||
_fillParent = _layoutSize == FILL_PARENT;
|
||||
}
|
||||
void layout(ref Rect rc) {
|
||||
_widget.layout(rc);
|
||||
}
|
||||
}
|
||||
|
||||
/// helper class for layouts
|
||||
class LayoutItems {
|
||||
Orientation _orientation;
|
||||
LayoutItem[] _list;
|
||||
int _count;
|
||||
int _totalSize;
|
||||
int _maxSecondarySize;
|
||||
Point _measureParentSize;
|
||||
|
||||
int _layoutWidth;
|
||||
int _layoutHeight;
|
||||
|
||||
void setLayoutParams(Orientation orientation, int layoutWidth, int layoutHeight) {
|
||||
_orientation = orientation;
|
||||
_layoutWidth = layoutWidth;
|
||||
_layoutHeight = layoutHeight;
|
||||
}
|
||||
|
||||
/// fill widget layout list with Visible or Invisible items, measure them
|
||||
Point measure(int parentWidth, int parentHeight) {
|
||||
_totalSize = 0;
|
||||
_maxSecondarySize = 0;
|
||||
_measureParentSize.x = parentWidth;
|
||||
_measureParentSize.y = parentHeight;
|
||||
// measure
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
item.measure(parentWidth, parentHeight);
|
||||
if (_maxSecondarySize < item._secondarySize)
|
||||
_maxSecondarySize = item._secondarySize;
|
||||
_totalSize += item._measuredSize;
|
||||
}
|
||||
return _orientation == Orientation.Horizontal ? Point(_totalSize, _maxSecondarySize) : Point(_maxSecondarySize, _totalSize);
|
||||
}
|
||||
|
||||
/// fill widget layout list with Visible or Invisible items, measure them
|
||||
void setWidgets(ref WidgetList widgets) {
|
||||
// remove old items, if any
|
||||
clear();
|
||||
// reserve space
|
||||
if (_list.length < widgets.count)
|
||||
_list.length = widgets.count;
|
||||
// copy
|
||||
for (int i = 0; i < widgets.count; i++) {
|
||||
Widget item = widgets.get(i);
|
||||
if (item.visibility == Visibility.Gone)
|
||||
continue;
|
||||
_list[_count++].set(item, _orientation);
|
||||
}
|
||||
}
|
||||
|
||||
void layout(Rect rc) {
|
||||
// measure again - available area could be changed
|
||||
if (_measureParentSize.x != rc.width || _measureParentSize.y != rc.height)
|
||||
measure(rc.width, rc.height);
|
||||
int contentSecondarySize = 0;
|
||||
int contentHeight = 0;
|
||||
int totalSize = 0;
|
||||
int delta = 0;
|
||||
int resizableSize = 0;
|
||||
int resizableWeight = 0;
|
||||
int nonresizableSize = 0;
|
||||
int nonresizableWeight = 0;
|
||||
int maxItem = 0; // max item dimention
|
||||
// calc total size
|
||||
int visibleCount = cast(int)_list.length;
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
int weight = item.weight;
|
||||
int size = item.measuredSize;
|
||||
totalSize += size;
|
||||
if (maxItem < item.secondarySize)
|
||||
maxItem = item.secondarySize;
|
||||
if (item.fillParent) {
|
||||
resizableWeight += weight;
|
||||
resizableSize += size * weight;
|
||||
} else {
|
||||
nonresizableWeight += weight;
|
||||
nonresizableSize += size * weight;
|
||||
}
|
||||
}
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
if (_layoutWidth == WRAP_CONTENT && maxItem < rc.width)
|
||||
contentSecondarySize = maxItem;
|
||||
else
|
||||
contentSecondarySize = rc.width;
|
||||
if (_layoutHeight == FILL_PARENT || totalSize > rc.height)
|
||||
delta = rc.height - totalSize; // total space to add to fit
|
||||
} else {
|
||||
if (_layoutHeight == WRAP_CONTENT && maxItem < rc.height)
|
||||
contentSecondarySize = maxItem;
|
||||
else
|
||||
contentSecondarySize = rc.height;
|
||||
if (_layoutWidth == FILL_PARENT || totalSize > rc.width)
|
||||
delta = rc.width - totalSize; // total space to add to fit
|
||||
}
|
||||
// calculate resize options and scale
|
||||
bool needForceResize = false;
|
||||
bool needResize = false;
|
||||
int scaleFactor = 10000; // per weight unit
|
||||
if (delta != 0 && visibleCount > 0) {
|
||||
// need resize of some children
|
||||
needResize = true;
|
||||
// resize all if need to shrink or only resizable are too small to correct delta
|
||||
needForceResize = delta < 0 || resizableWeight == 0; // || resizableSize * 2 / 3 < delta; // do we need resize non-FILL_PARENT items?
|
||||
// calculate scale factor: weight / delta * 10000
|
||||
if (needForceResize)
|
||||
scaleFactor = 10000 * delta / (nonresizableSize + resizableSize);
|
||||
else
|
||||
scaleFactor = 10000 * delta / resizableSize;
|
||||
}
|
||||
//Log.d("VerticalLayout delta=", delta, ", nonres=", nonresizableWeight, ", res=", resizableWeight, ", scale=", scaleFactor);
|
||||
// find last resized - to allow fill space 1 pixel accurate
|
||||
int lastResized = -1;
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
if (item.fillParent || needForceResize) {
|
||||
lastResized = i;
|
||||
}
|
||||
}
|
||||
// final resize and layout of children
|
||||
int position = 0;
|
||||
int deltaTotal = 0;
|
||||
for (int i = 0; i < _count; i++) {
|
||||
LayoutItem * item = &_list[i];
|
||||
int layoutSize = item.layoutSize;
|
||||
int weight = item.weight;
|
||||
int size = item.measuredSize;
|
||||
if (needResize && (layoutSize == FILL_PARENT || needForceResize)) {
|
||||
// do resize
|
||||
int correction = scaleFactor * weight * size / 10000;
|
||||
deltaTotal += correction;
|
||||
// for last resized, apply additional correction to resolve calculation inaccuracy
|
||||
if (i == lastResized) {
|
||||
correction += delta - deltaTotal;
|
||||
}
|
||||
size += correction;
|
||||
}
|
||||
// apply size
|
||||
Rect childRect = rc;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
childRect.top += position;
|
||||
childRect.bottom = childRect.top + size;
|
||||
childRect.right = childRect.left + contentSecondarySize;
|
||||
item.layout(childRect);
|
||||
} else {
|
||||
// Horizontal
|
||||
childRect.left += position;
|
||||
childRect.right = childRect.left + size;
|
||||
childRect.bottom = childRect.top + contentSecondarySize;
|
||||
item.layout(childRect);
|
||||
}
|
||||
position += size;
|
||||
}
|
||||
}
|
||||
|
||||
void clear() {
|
||||
for (int i = 0; i < _count; i++)
|
||||
_list[i].clear();
|
||||
_count = 0;
|
||||
}
|
||||
~this() {
|
||||
clear();
|
||||
}
|
||||
}
|
||||
|
||||
class LinearLayout : WidgetGroup {
|
||||
protected Orientation _orientation = Orientation.Vertical;
|
||||
/// returns linear layout orientation (Vertical, Horizontal)
|
||||
@property Orientation orientation() { return _orientation; }
|
||||
/// sets linear layout orientation
|
||||
@property LinearLayout orientation(Orientation value) { _orientation = value; requestLayout(); return this; }
|
||||
|
||||
this(string ID = null) {
|
||||
super(ID);
|
||||
_layoutItems = new LayoutItems();
|
||||
}
|
||||
|
||||
LayoutItems _layoutItems;
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
// measure children
|
||||
_layoutItems.setLayoutParams(orientation, layoutWidth, layoutHeight);
|
||||
_layoutItems.setWidgets(_children);
|
||||
Point sz = _layoutItems.measure(pwidth, pheight);
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
_needLayout = false;
|
||||
if (visibility == Visibility.Gone) {
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
_layoutItems.layout(rc);
|
||||
}
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Visible)
|
||||
continue;
|
||||
item.onDraw(buf);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
class VerticalLayout : LinearLayout {
|
||||
this(string ID = null) {
|
||||
super(ID);
|
||||
orientation = Orientation.Vertical;
|
||||
}
|
||||
}
|
||||
|
||||
class HorizontalLayout : LinearLayout {
|
||||
this(string ID = null) {
|
||||
super(ID);
|
||||
orientation = Orientation.Horizontal;
|
||||
}
|
||||
}
|
||||
|
||||
/// place all children into same place (usually, only one child should be visible at a time)
|
||||
class FrameLayout : WidgetGroup {
|
||||
this(string ID) {
|
||||
super(ID);
|
||||
}
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
// measure children
|
||||
Point sz;
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Gone) {
|
||||
item.measure(pwidth, pheight);
|
||||
if (sz.x < item.measuredWidth)
|
||||
sz.x = item.measuredWidth;
|
||||
if (sz.y < item.measuredHeight)
|
||||
sz.y = item.measuredHeight;
|
||||
}
|
||||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
_needLayout = false;
|
||||
if (visibility == Visibility.Gone) {
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Gone) {
|
||||
item.layout(rc);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Visible)
|
||||
continue;
|
||||
item.onDraw(buf);
|
||||
}
|
||||
}
|
||||
|
||||
/// make one of children (with specified ID) visible, for the rest, set visibility to otherChildrenVisibility
|
||||
bool showChild(string ID, Visibility otherChildrenVisibility = Visibility.Invisible) {
|
||||
bool found = false;
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.compareId(ID)) {
|
||||
item.visibility = Visibility.Visible;
|
||||
found = true;
|
||||
} else {
|
||||
item.visibility = otherChildrenVisibility;
|
||||
}
|
||||
}
|
||||
return found;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,403 +1,402 @@
|
|||
module dlangui.widgets.lists;
|
||||
|
||||
import dlangui.widgets.widget;
|
||||
import dlangui.widgets.controls;
|
||||
|
||||
/// list widget adapter provides items for list widgets
|
||||
interface ListAdapter {
|
||||
/// returns number of widgets in list
|
||||
@property int itemCount();
|
||||
/// return list item widget by item index
|
||||
Widget itemWidget(int index);
|
||||
}
|
||||
|
||||
/// List adapter for simple list of widget instances
|
||||
class WidgetListAdapter : ListAdapter {
|
||||
WidgetList _widgets;
|
||||
/// list of widgets to display
|
||||
@property ref WidgetList widgets() { return _widgets; }
|
||||
/// returns number of widgets in list
|
||||
@property override int itemCount() {
|
||||
return _widgets.count;
|
||||
}
|
||||
/// return list item widget by item index
|
||||
override Widget itemWidget(int index) {
|
||||
return _widgets.get(index);
|
||||
}
|
||||
}
|
||||
|
||||
/// List
|
||||
class ListWidget : WidgetGroup, OnScrollHandler {
|
||||
protected Orientation _orientation = Orientation.Vertical;
|
||||
/// returns linear layout orientation (Vertical, Horizontal)
|
||||
@property Orientation orientation() { return _orientation; }
|
||||
/// sets linear layout orientation
|
||||
@property ListWidget orientation(Orientation value) {
|
||||
_orientation = value;
|
||||
_scrollbar.orientation = value;
|
||||
requestLayout();
|
||||
return this;
|
||||
}
|
||||
|
||||
protected Rect[] _itemRects;
|
||||
protected Point[] _itemSizes;
|
||||
protected bool _needScrollbar;
|
||||
protected Point _sbsz; // scrollbar size
|
||||
protected ScrollBar _scrollbar;
|
||||
protected int _lastMeasureWidth;
|
||||
protected int _lastMeasureHeight;
|
||||
|
||||
/// first visible item index
|
||||
protected int _firstVisibleItem;
|
||||
/// scroll position - offset of scroll area
|
||||
protected int _scrollPosition;
|
||||
/// maximum scroll position
|
||||
protected int _maxScrollPosition;
|
||||
/// client area rectangle (counting padding, margins, and scrollbar)
|
||||
protected Rect _clientRc;
|
||||
/// total height of all items for Vertical orientation, or width for Horizontal
|
||||
protected int _totalSize;
|
||||
|
||||
/// returns rectangle for item (not scrolled, first item starts at 0,0)
|
||||
Rect itemRectNoScroll(int index) {
|
||||
Rect res;
|
||||
res = _itemRects[index];
|
||||
return res;
|
||||
}
|
||||
|
||||
/// returns rectangle for item (scrolled)
|
||||
Rect itemRect(int index) {
|
||||
Rect res = itemRectNoScroll(index);
|
||||
if (_orientation == Orientation.Horizontal) {
|
||||
res.left += _scrollPosition;
|
||||
res.right += _scrollPosition;
|
||||
} else {
|
||||
res.top += _scrollPosition;
|
||||
res.bottom += _scrollPosition;
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
/// returns item index by 0-based offset from top/left of list content
|
||||
int itemByPosition(int pos) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
protected ListAdapter _adapter;
|
||||
/// when true, need to destroy adapter on list destroy
|
||||
protected bool _ownAdapter;
|
||||
|
||||
/// get adapter
|
||||
@property ListAdapter adapter() { return _adapter; }
|
||||
/// set adapter
|
||||
@property ListWidget adapter(ListAdapter adapter) {
|
||||
if (_adapter !is null && _ownAdapter)
|
||||
destroy(_adapter);
|
||||
_adapter = adapter;
|
||||
_ownAdapter = false;
|
||||
onAdapterChanged();
|
||||
return this;
|
||||
}
|
||||
/// set adapter
|
||||
@property ListWidget ownAdapter(ListAdapter adapter) {
|
||||
if (_adapter !is null && _ownAdapter)
|
||||
destroy(_adapter);
|
||||
_adapter = adapter;
|
||||
_ownAdapter = true;
|
||||
onAdapterChanged();
|
||||
return this;
|
||||
}
|
||||
|
||||
/// returns number of widgets in list
|
||||
@property int itemCount() {
|
||||
if (_adapter !is null)
|
||||
return _adapter.itemCount;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/// return list item widget by item index
|
||||
Widget itemWidget(int index) {
|
||||
if (_adapter !is null)
|
||||
return _adapter.itemWidget(index);
|
||||
return null;
|
||||
}
|
||||
|
||||
void onAdapterChanged() {
|
||||
requestLayout();
|
||||
}
|
||||
|
||||
this(string ID = null, Orientation orientation = Orientation.Vertical) {
|
||||
super(ID);
|
||||
_orientation = orientation;
|
||||
_scrollbar = new ScrollBar("listscroll", orientation);
|
||||
_scrollbar.visibility = Visibility.Gone;
|
||||
_scrollbar.onScrollEventListener = &onScrollEvent;
|
||||
addChild(_scrollbar);
|
||||
}
|
||||
|
||||
~this() {
|
||||
if (_adapter !is null && _ownAdapter)
|
||||
destroy(_adapter);
|
||||
_adapter = null;
|
||||
}
|
||||
|
||||
module dlangui.widgets.lists;
|
||||
|
||||
import dlangui.widgets.widget;
|
||||
import dlangui.widgets.controls;
|
||||
|
||||
/// list widget adapter provides items for list widgets
|
||||
interface ListAdapter {
|
||||
/// returns number of widgets in list
|
||||
@property int itemCount();
|
||||
/// return list item widget by item index
|
||||
Widget itemWidget(int index);
|
||||
}
|
||||
|
||||
/// List adapter for simple list of widget instances
|
||||
class WidgetListAdapter : ListAdapter {
|
||||
WidgetList _widgets;
|
||||
/// list of widgets to display
|
||||
@property ref WidgetList widgets() { return _widgets; }
|
||||
/// returns number of widgets in list
|
||||
@property override int itemCount() {
|
||||
return _widgets.count;
|
||||
}
|
||||
/// return list item widget by item index
|
||||
override Widget itemWidget(int index) {
|
||||
return _widgets.get(index);
|
||||
}
|
||||
}
|
||||
|
||||
/// List
|
||||
class ListWidget : WidgetGroup, OnScrollHandler {
|
||||
protected Orientation _orientation = Orientation.Vertical;
|
||||
/// returns linear layout orientation (Vertical, Horizontal)
|
||||
@property Orientation orientation() { return _orientation; }
|
||||
/// sets linear layout orientation
|
||||
@property ListWidget orientation(Orientation value) {
|
||||
_orientation = value;
|
||||
_scrollbar.orientation = value;
|
||||
requestLayout();
|
||||
return this;
|
||||
}
|
||||
|
||||
protected Rect[] _itemRects;
|
||||
protected Point[] _itemSizes;
|
||||
protected bool _needScrollbar;
|
||||
protected Point _sbsz; // scrollbar size
|
||||
protected ScrollBar _scrollbar;
|
||||
protected int _lastMeasureWidth;
|
||||
protected int _lastMeasureHeight;
|
||||
|
||||
/// first visible item index
|
||||
protected int _firstVisibleItem;
|
||||
/// scroll position - offset of scroll area
|
||||
protected int _scrollPosition;
|
||||
/// maximum scroll position
|
||||
protected int _maxScrollPosition;
|
||||
/// client area rectangle (counting padding, margins, and scrollbar)
|
||||
protected Rect _clientRc;
|
||||
/// total height of all items for Vertical orientation, or width for Horizontal
|
||||
protected int _totalSize;
|
||||
|
||||
/// returns rectangle for item (not scrolled, first item starts at 0,0)
|
||||
Rect itemRectNoScroll(int index) {
|
||||
Rect res;
|
||||
res = _itemRects[index];
|
||||
return res;
|
||||
}
|
||||
|
||||
/// returns rectangle for item (scrolled)
|
||||
Rect itemRect(int index) {
|
||||
Rect res = itemRectNoScroll(index);
|
||||
if (_orientation == Orientation.Horizontal) {
|
||||
res.left += _scrollPosition;
|
||||
res.right += _scrollPosition;
|
||||
} else {
|
||||
res.top += _scrollPosition;
|
||||
res.bottom += _scrollPosition;
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
/// returns item index by 0-based offset from top/left of list content
|
||||
int itemByPosition(int pos) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
protected ListAdapter _adapter;
|
||||
/// when true, need to destroy adapter on list destroy
|
||||
protected bool _ownAdapter;
|
||||
|
||||
/// get adapter
|
||||
@property ListAdapter adapter() { return _adapter; }
|
||||
/// set adapter
|
||||
@property ListWidget adapter(ListAdapter adapter) {
|
||||
if (_adapter !is null && _ownAdapter)
|
||||
destroy(_adapter);
|
||||
_adapter = adapter;
|
||||
_ownAdapter = false;
|
||||
onAdapterChanged();
|
||||
return this;
|
||||
}
|
||||
/// set adapter
|
||||
@property ListWidget ownAdapter(ListAdapter adapter) {
|
||||
if (_adapter !is null && _ownAdapter)
|
||||
destroy(_adapter);
|
||||
_adapter = adapter;
|
||||
_ownAdapter = true;
|
||||
onAdapterChanged();
|
||||
return this;
|
||||
}
|
||||
|
||||
/// returns number of widgets in list
|
||||
@property int itemCount() {
|
||||
if (_adapter !is null)
|
||||
return _adapter.itemCount;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/// return list item widget by item index
|
||||
Widget itemWidget(int index) {
|
||||
if (_adapter !is null)
|
||||
return _adapter.itemWidget(index);
|
||||
return null;
|
||||
}
|
||||
|
||||
void onAdapterChanged() {
|
||||
requestLayout();
|
||||
}
|
||||
|
||||
this(string ID = null, Orientation orientation = Orientation.Vertical) {
|
||||
super(ID);
|
||||
_orientation = orientation;
|
||||
_scrollbar = new ScrollBar("listscroll", orientation);
|
||||
_scrollbar.visibility = Visibility.Gone;
|
||||
_scrollbar.onScrollEventListener = &onScrollEvent;
|
||||
addChild(_scrollbar);
|
||||
}
|
||||
|
||||
~this() {
|
||||
if (_adapter !is null && _ownAdapter)
|
||||
destroy(_adapter);
|
||||
_adapter = null;
|
||||
}
|
||||
|
||||
/// handle scroll event
|
||||
override bool onScrollEvent(AbstractSlider source, ScrollEvent event) {
|
||||
int newPosition = _scrollPosition;
|
||||
if (event.action == ScrollAction.SliderMoved) {
|
||||
// scroll
|
||||
newPosition = event.position;
|
||||
} else {
|
||||
// use default handler for page/line up/down events
|
||||
newPosition = event.defaultUpdatePosition();
|
||||
}
|
||||
if (_scrollPosition != newPosition) {
|
||||
_scrollPosition = newPosition;
|
||||
if (_scrollPosition > _maxScrollPosition)
|
||||
_scrollPosition = _maxScrollPosition;
|
||||
if (_scrollPosition < 0)
|
||||
_scrollPosition = 0;
|
||||
invalidate();
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
override bool onScrollEvent(AbstractSlider source, ScrollEvent event) {
|
||||
int newPosition = _scrollPosition;
|
||||
if (event.action == ScrollAction.SliderMoved) {
|
||||
// scroll
|
||||
newPosition = event.position;
|
||||
} else {
|
||||
// use default handler for page/line up/down events
|
||||
newPosition = event.defaultUpdatePosition();
|
||||
}
|
||||
if (_scrollPosition != newPosition) {
|
||||
_scrollPosition = newPosition;
|
||||
if (_scrollPosition > _maxScrollPosition)
|
||||
_scrollPosition = _maxScrollPosition;
|
||||
if (_scrollPosition < 0)
|
||||
_scrollPosition = 0;
|
||||
invalidate();
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
if (visibility == Visibility.Gone) {
|
||||
_measuredWidth = _measuredHeight = 0;
|
||||
return;
|
||||
}
|
||||
if (_itemSizes.length < itemCount)
|
||||
_itemSizes.length = itemCount;
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
_scrollbar.visibility = Visibility.Visible;
|
||||
_scrollbar.measure(pwidth, pheight);
|
||||
|
||||
_lastMeasureWidth = pwidth;
|
||||
_lastMeasureHeight = pheight;
|
||||
|
||||
int sbsize = _orientation == Orientation.Vertical ? _scrollbar.measuredWidth : _scrollbar.measuredHeight;
|
||||
// measure children
|
||||
Point sz;
|
||||
_sbsz.clear;
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
Widget w = itemWidget(i);
|
||||
if (w is null || w.visibility == Visibility.Gone) {
|
||||
_itemSizes[i].x = _itemSizes[i].y = 0;
|
||||
continue;
|
||||
}
|
||||
w.measure(pwidth, pheight);
|
||||
_itemSizes[i].x = w.measuredWidth;
|
||||
_itemSizes[i].y = w.measuredHeight;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
if (sz.x < w.measuredWidth)
|
||||
sz.x = w.measuredWidth;
|
||||
sz.y += w.measuredHeight;
|
||||
} else {
|
||||
// Horizontal
|
||||
w.measure(pwidth, pheight);
|
||||
if (sz.y < w.measuredHeight)
|
||||
sz.y = w.measuredHeight;
|
||||
sz.x += w.measuredWidth;
|
||||
}
|
||||
}
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
if (pheight != SIZE_UNSPECIFIED && sz.y > pheight) {
|
||||
// need scrollbar
|
||||
if (pwidth != SIZE_UNSPECIFIED) {
|
||||
pwidth -= sbsize;
|
||||
_sbsz.x = sbsize;
|
||||
_needScrollbar = true;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (pwidth != SIZE_UNSPECIFIED && sz.x > pwidth) {
|
||||
// need scrollbar
|
||||
if (pheight != SIZE_UNSPECIFIED) {
|
||||
pheight -= sbsize;
|
||||
_sbsz.y = sbsize;
|
||||
_needScrollbar = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (_needScrollbar) {
|
||||
// recalculate with scrollbar
|
||||
sz.x = sz.y = 0;
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
Widget w = itemWidget(i);
|
||||
if (w is null || w.visibility == Visibility.Gone)
|
||||
continue;
|
||||
w.measure(pwidth, pheight);
|
||||
_itemSizes[i].x = w.measuredWidth;
|
||||
_itemSizes[i].y = w.measuredHeight;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
if (sz.x < w.measuredWidth)
|
||||
sz.x = w.measuredWidth;
|
||||
sz.y += w.measuredHeight;
|
||||
} else {
|
||||
// Horizontal
|
||||
w.measure(pwidth, pheight);
|
||||
if (sz.y < w.measuredHeight)
|
||||
sz.y = w.measuredHeight;
|
||||
sz.x += w.measuredWidth;
|
||||
}
|
||||
}
|
||||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x + _sbsz.x, sz.y + _sbsz.y);
|
||||
}
|
||||
|
||||
|
||||
protected void updateItemPositions() {
|
||||
Rect r;
|
||||
int p = 0;
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
if (_itemSizes[i].x == 0 && _itemSizes[i].y == 0)
|
||||
continue;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
int w = _clientRc.width;
|
||||
int h = _itemSizes[i].y;
|
||||
r.top = p;
|
||||
r.bottom = p + h;
|
||||
r.left = 0;
|
||||
r.right = w;
|
||||
_itemRects[i] = r;
|
||||
p += h;
|
||||
} else {
|
||||
// Horizontal
|
||||
int h = _clientRc.height;
|
||||
int w = _itemSizes[i].x;
|
||||
r.top = 0;
|
||||
r.bottom = h;
|
||||
r.left = p;
|
||||
r.right = p + w;
|
||||
_itemRects[i] = r;
|
||||
p += w;
|
||||
}
|
||||
}
|
||||
_totalSize = p;
|
||||
if (_needScrollbar) {
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
_scrollbar.setRange(0, p);
|
||||
_scrollbar.pageSize = _clientRc.height;
|
||||
_scrollbar.position = _scrollPosition;
|
||||
} else {
|
||||
_scrollbar.setRange(0, p);
|
||||
_scrollbar.pageSize = _clientRc.width;
|
||||
_scrollbar.position = _scrollPosition;
|
||||
}
|
||||
}
|
||||
/// maximum scroll position
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
_maxScrollPosition = _totalSize - _clientRc.height;
|
||||
if (_maxScrollPosition < 0)
|
||||
_maxScrollPosition = 0;
|
||||
} else {
|
||||
_maxScrollPosition = _totalSize - _clientRc.width;
|
||||
if (_maxScrollPosition < 0)
|
||||
_maxScrollPosition = 0;
|
||||
}
|
||||
if (_scrollPosition > _maxScrollPosition)
|
||||
_scrollPosition = _maxScrollPosition;
|
||||
if (_scrollPosition < 0)
|
||||
_scrollPosition = 0;
|
||||
if (_needScrollbar) {
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
_scrollbar.position = _scrollPosition;
|
||||
} else {
|
||||
_scrollbar.position = _scrollPosition;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
if (visibility == Visibility.Gone) {
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
|
||||
if (_itemRects.length < itemCount)
|
||||
_itemRects.length = itemCount;
|
||||
|
||||
// measure again if client size has been changed
|
||||
if (_lastMeasureWidth != rc.width || _lastMeasureHeight != rc.height)
|
||||
measure(rc.width, rc.height);
|
||||
|
||||
// layout scrollbar
|
||||
if (_needScrollbar) {
|
||||
_scrollbar.visibility = Visibility.Visible;
|
||||
Rect sbrect = rc;
|
||||
if (_orientation == Orientation.Vertical)
|
||||
sbrect.left = sbrect.right - _sbsz.x;
|
||||
else
|
||||
sbrect.top = sbrect.bottom - _sbsz.y;
|
||||
_scrollbar.layout(sbrect);
|
||||
rc.right -= _sbsz.x;
|
||||
rc.bottom -= _sbsz.y;
|
||||
} else {
|
||||
_scrollbar.visibility = Visibility.Gone;
|
||||
}
|
||||
|
||||
_clientRc = rc;
|
||||
|
||||
// calc item rectangles
|
||||
updateItemPositions();
|
||||
|
||||
_needLayout = false;
|
||||
}
|
||||
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
// draw scrollbar
|
||||
if (_needScrollbar)
|
||||
_scrollbar.onDraw(buf);
|
||||
|
||||
Point scrollOffset;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
scrollOffset.y = _scrollPosition;
|
||||
} else {
|
||||
scrollOffset.x = _scrollPosition;
|
||||
}
|
||||
// todo: scrollOffset
|
||||
// draw items
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
Rect itemrc = _itemRects[i];
|
||||
itemrc.left += rc.left - scrollOffset.x;
|
||||
itemrc.right += rc.left - scrollOffset.x;
|
||||
itemrc.top += rc.top - scrollOffset.y;
|
||||
itemrc.bottom += rc.top - scrollOffset.y;
|
||||
if (itemrc.intersects(rc)) {
|
||||
Widget w = itemWidget(i);
|
||||
if (w is null || w.visibility != Visibility.Visible)
|
||||
continue;
|
||||
w.measure(itemrc.width, itemrc.height);
|
||||
w.layout(itemrc);
|
||||
w.onDraw(buf);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
if (_itemSizes.length < itemCount)
|
||||
_itemSizes.length = itemCount;
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
_scrollbar.visibility = Visibility.Visible;
|
||||
_scrollbar.measure(pwidth, pheight);
|
||||
|
||||
_lastMeasureWidth = pwidth;
|
||||
_lastMeasureHeight = pheight;
|
||||
|
||||
int sbsize = _orientation == Orientation.Vertical ? _scrollbar.measuredWidth : _scrollbar.measuredHeight;
|
||||
// measure children
|
||||
Point sz;
|
||||
_sbsz.clear;
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
Widget w = itemWidget(i);
|
||||
if (w is null || w.visibility == Visibility.Gone) {
|
||||
_itemSizes[i].x = _itemSizes[i].y = 0;
|
||||
continue;
|
||||
}
|
||||
w.measure(pwidth, pheight);
|
||||
_itemSizes[i].x = w.measuredWidth;
|
||||
_itemSizes[i].y = w.measuredHeight;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
if (sz.x < w.measuredWidth)
|
||||
sz.x = w.measuredWidth;
|
||||
sz.y += w.measuredHeight;
|
||||
} else {
|
||||
// Horizontal
|
||||
w.measure(pwidth, pheight);
|
||||
if (sz.y < w.measuredHeight)
|
||||
sz.y = w.measuredHeight;
|
||||
sz.x += w.measuredWidth;
|
||||
}
|
||||
}
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
if (pheight != SIZE_UNSPECIFIED && sz.y > pheight) {
|
||||
// need scrollbar
|
||||
if (pwidth != SIZE_UNSPECIFIED) {
|
||||
pwidth -= sbsize;
|
||||
_sbsz.x = sbsize;
|
||||
_needScrollbar = true;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (pwidth != SIZE_UNSPECIFIED && sz.x > pwidth) {
|
||||
// need scrollbar
|
||||
if (pheight != SIZE_UNSPECIFIED) {
|
||||
pheight -= sbsize;
|
||||
_sbsz.y = sbsize;
|
||||
_needScrollbar = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (_needScrollbar) {
|
||||
// recalculate with scrollbar
|
||||
sz.x = sz.y = 0;
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
Widget w = itemWidget(i);
|
||||
if (w is null || w.visibility == Visibility.Gone)
|
||||
continue;
|
||||
w.measure(pwidth, pheight);
|
||||
_itemSizes[i].x = w.measuredWidth;
|
||||
_itemSizes[i].y = w.measuredHeight;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
if (sz.x < w.measuredWidth)
|
||||
sz.x = w.measuredWidth;
|
||||
sz.y += w.measuredHeight;
|
||||
} else {
|
||||
// Horizontal
|
||||
w.measure(pwidth, pheight);
|
||||
if (sz.y < w.measuredHeight)
|
||||
sz.y = w.measuredHeight;
|
||||
sz.x += w.measuredWidth;
|
||||
}
|
||||
}
|
||||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x + _sbsz.x, sz.y + _sbsz.y);
|
||||
}
|
||||
|
||||
|
||||
protected void updateItemPositions() {
|
||||
Rect r;
|
||||
int p = 0;
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
if (_itemSizes[i].x == 0 && _itemSizes[i].y == 0)
|
||||
continue;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
// Vertical
|
||||
int w = _clientRc.width;
|
||||
int h = _itemSizes[i].y;
|
||||
r.top = p;
|
||||
r.bottom = p + h;
|
||||
r.left = 0;
|
||||
r.right = w;
|
||||
_itemRects[i] = r;
|
||||
p += h;
|
||||
} else {
|
||||
// Horizontal
|
||||
int h = _clientRc.height;
|
||||
int w = _itemSizes[i].x;
|
||||
r.top = 0;
|
||||
r.bottom = h;
|
||||
r.left = p;
|
||||
r.right = p + w;
|
||||
_itemRects[i] = r;
|
||||
p += w;
|
||||
}
|
||||
}
|
||||
_totalSize = p;
|
||||
if (_needScrollbar) {
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
_scrollbar.setRange(0, p);
|
||||
_scrollbar.pageSize = _clientRc.height;
|
||||
_scrollbar.position = _scrollPosition;
|
||||
} else {
|
||||
_scrollbar.setRange(0, p);
|
||||
_scrollbar.pageSize = _clientRc.width;
|
||||
_scrollbar.position = _scrollPosition;
|
||||
}
|
||||
}
|
||||
/// maximum scroll position
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
_maxScrollPosition = _totalSize - _clientRc.height;
|
||||
if (_maxScrollPosition < 0)
|
||||
_maxScrollPosition = 0;
|
||||
} else {
|
||||
_maxScrollPosition = _totalSize - _clientRc.width;
|
||||
if (_maxScrollPosition < 0)
|
||||
_maxScrollPosition = 0;
|
||||
}
|
||||
if (_scrollPosition > _maxScrollPosition)
|
||||
_scrollPosition = _maxScrollPosition;
|
||||
if (_scrollPosition < 0)
|
||||
_scrollPosition = 0;
|
||||
if (_needScrollbar) {
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
_scrollbar.position = _scrollPosition;
|
||||
} else {
|
||||
_scrollbar.position = _scrollPosition;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
_needLayout = false;
|
||||
if (visibility == Visibility.Gone) {
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
|
||||
if (_itemRects.length < itemCount)
|
||||
_itemRects.length = itemCount;
|
||||
|
||||
// measure again if client size has been changed
|
||||
if (_lastMeasureWidth != rc.width || _lastMeasureHeight != rc.height)
|
||||
measure(rc.width, rc.height);
|
||||
|
||||
// layout scrollbar
|
||||
if (_needScrollbar) {
|
||||
_scrollbar.visibility = Visibility.Visible;
|
||||
Rect sbrect = rc;
|
||||
if (_orientation == Orientation.Vertical)
|
||||
sbrect.left = sbrect.right - _sbsz.x;
|
||||
else
|
||||
sbrect.top = sbrect.bottom - _sbsz.y;
|
||||
_scrollbar.layout(sbrect);
|
||||
rc.right -= _sbsz.x;
|
||||
rc.bottom -= _sbsz.y;
|
||||
} else {
|
||||
_scrollbar.visibility = Visibility.Gone;
|
||||
}
|
||||
|
||||
_clientRc = rc;
|
||||
|
||||
// calc item rectangles
|
||||
updateItemPositions();
|
||||
}
|
||||
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
// draw scrollbar
|
||||
if (_needScrollbar)
|
||||
_scrollbar.onDraw(buf);
|
||||
|
||||
Point scrollOffset;
|
||||
if (_orientation == Orientation.Vertical) {
|
||||
scrollOffset.y = _scrollPosition;
|
||||
} else {
|
||||
scrollOffset.x = _scrollPosition;
|
||||
}
|
||||
// todo: scrollOffset
|
||||
// draw items
|
||||
for (int i = 0; i < itemCount; i++) {
|
||||
Rect itemrc = _itemRects[i];
|
||||
itemrc.left += rc.left - scrollOffset.x;
|
||||
itemrc.right += rc.left - scrollOffset.x;
|
||||
itemrc.top += rc.top - scrollOffset.y;
|
||||
itemrc.bottom += rc.top - scrollOffset.y;
|
||||
if (itemrc.intersects(rc)) {
|
||||
Widget w = itemWidget(i);
|
||||
if (w is null || w.visibility != Visibility.Visible)
|
||||
continue;
|
||||
w.measure(itemrc.width, itemrc.height);
|
||||
w.layout(itemrc);
|
||||
w.onDraw(buf);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
|
|
@ -1,12 +1,12 @@
|
|||
module dlangui.widgets.tabs;
|
||||
|
||||
import dlangui.widgets.layouts;
|
||||
import dlangui.widgets.controls;
|
||||
|
||||
interface TabHandler {
|
||||
void onTabChanged(string newActiveTabId, string previousTabId);
|
||||
}
|
||||
|
||||
module dlangui.widgets.tabs;
|
||||
|
||||
import dlangui.widgets.layouts;
|
||||
import dlangui.widgets.controls;
|
||||
|
||||
interface TabHandler {
|
||||
void onTabChanged(string newActiveTabId, string previousTabId);
|
||||
}
|
||||
|
||||
class TabItem {
|
||||
private string _iconRes;
|
||||
private string _id;
|
||||
|
@ -229,109 +229,109 @@ class TabControl : WidgetGroup {
|
|||
}
|
||||
return true;
|
||||
}
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
// measure children
|
||||
Point sz;
|
||||
_moreButton.measure(pwidth, pheight);
|
||||
sz.x = _moreButton.measuredWidth;
|
||||
sz.y = _moreButton.measuredHeight;
|
||||
pwidth -= sz.x;
|
||||
for (int i = 1; i < _children.count; i++) {
|
||||
Widget tab = _children.get(i);
|
||||
tab.visibility = Visibility.Visible;
|
||||
tab.measure(pwidth, pheight);
|
||||
if (sz.y < tab.measuredHeight)
|
||||
sz.y = tab.measuredHeight;
|
||||
if (sz.x + tab.measuredWidth > pwidth)
|
||||
break;
|
||||
sz.x += tab.measuredWidth;
|
||||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
_needLayout = false;
|
||||
if (visibility == Visibility.Gone) {
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
// more button
|
||||
Rect moreRc = rc;
|
||||
moreRc.left = rc.right - _moreButton.measuredWidth;
|
||||
_moreButton.layout(moreRc);
|
||||
rc.right -= _moreButton.measuredWidth;
|
||||
// tabs
|
||||
int maxw = rc.width;
|
||||
// measure and update visibility
|
||||
TabItemWidget[] sorted = sortedItems();
|
||||
int w = 0;
|
||||
for (int i = 0; i < sorted.length; i++) {
|
||||
TabItemWidget widget = sorted[i];
|
||||
widget.visibility = Visibility.Visible;
|
||||
widget.measure(rc.width, rc.height);
|
||||
if (w + widget.measuredWidth < maxw) {
|
||||
w += widget.measuredWidth;
|
||||
} else {
|
||||
widget.visibility = Visibility.Gone;
|
||||
}
|
||||
}
|
||||
// layout visible items
|
||||
for (int i = 1; i < _children.count; i++) {
|
||||
TabItemWidget widget = cast(TabItemWidget)_children.get(i);
|
||||
if (widget.visibility != Visibility.Visible)
|
||||
continue;
|
||||
w = widget.measuredWidth;
|
||||
rc.right = rc.left + w;
|
||||
widget.layout(rc);
|
||||
rc.left += w;
|
||||
}
|
||||
}
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Visible)
|
||||
continue;
|
||||
item.onDraw(buf);
|
||||
}
|
||||
}
|
||||
|
||||
protected string _selectedTabId;
|
||||
|
||||
void selectTab(int index) {
|
||||
if (_children.get(index + 1).compareId(_selectedTabId))
|
||||
return; // already selected
|
||||
string previousSelectedTab = _selectedTabId;
|
||||
for (int i = 1; i < _children.count; i++) {
|
||||
if (index == i - 1) {
|
||||
_children.get(i).state = State.Selected;
|
||||
_selectedTabId = _children.get(i).id;
|
||||
} else {
|
||||
_children.get(i).state = State.Normal;
|
||||
}
|
||||
}
|
||||
if (_onTabChanged !is null)
|
||||
_onTabChanged(_selectedTabId, previousSelectedTab);
|
||||
}
|
||||
/// Measure widget according to desired width and height constraints. (Step 1 of two phase layout).
|
||||
override void measure(int parentWidth, int parentHeight) {
|
||||
Rect m = margins;
|
||||
Rect p = padding;
|
||||
// calc size constraints for children
|
||||
int pwidth = parentWidth;
|
||||
int pheight = parentHeight;
|
||||
if (parentWidth != SIZE_UNSPECIFIED)
|
||||
pwidth -= m.left + m.right + p.left + p.right;
|
||||
if (parentHeight != SIZE_UNSPECIFIED)
|
||||
pheight -= m.top + m.bottom + p.top + p.bottom;
|
||||
// measure children
|
||||
Point sz;
|
||||
_moreButton.measure(pwidth, pheight);
|
||||
sz.x = _moreButton.measuredWidth;
|
||||
sz.y = _moreButton.measuredHeight;
|
||||
pwidth -= sz.x;
|
||||
for (int i = 1; i < _children.count; i++) {
|
||||
Widget tab = _children.get(i);
|
||||
tab.visibility = Visibility.Visible;
|
||||
tab.measure(pwidth, pheight);
|
||||
if (sz.y < tab.measuredHeight)
|
||||
sz.y = tab.measuredHeight;
|
||||
if (sz.x + tab.measuredWidth > pwidth)
|
||||
break;
|
||||
sz.x += tab.measuredWidth;
|
||||
}
|
||||
measuredContent(parentWidth, parentHeight, sz.x, sz.y);
|
||||
}
|
||||
/// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
override void layout(Rect rc) {
|
||||
_needLayout = false;
|
||||
if (visibility == Visibility.Gone) {
|
||||
return;
|
||||
}
|
||||
_pos = rc;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
// more button
|
||||
Rect moreRc = rc;
|
||||
moreRc.left = rc.right - _moreButton.measuredWidth;
|
||||
_moreButton.layout(moreRc);
|
||||
rc.right -= _moreButton.measuredWidth;
|
||||
// tabs
|
||||
int maxw = rc.width;
|
||||
// measure and update visibility
|
||||
TabItemWidget[] sorted = sortedItems();
|
||||
int w = 0;
|
||||
for (int i = 0; i < sorted.length; i++) {
|
||||
TabItemWidget widget = sorted[i];
|
||||
widget.visibility = Visibility.Visible;
|
||||
widget.measure(rc.width, rc.height);
|
||||
if (w + widget.measuredWidth < maxw) {
|
||||
w += widget.measuredWidth;
|
||||
} else {
|
||||
widget.visibility = Visibility.Gone;
|
||||
}
|
||||
}
|
||||
// layout visible items
|
||||
for (int i = 1; i < _children.count; i++) {
|
||||
TabItemWidget widget = cast(TabItemWidget)_children.get(i);
|
||||
if (widget.visibility != Visibility.Visible)
|
||||
continue;
|
||||
w = widget.measuredWidth;
|
||||
rc.right = rc.left + w;
|
||||
widget.layout(rc);
|
||||
rc.left += w;
|
||||
}
|
||||
}
|
||||
/// Draw widget at its position to buffer
|
||||
override void onDraw(DrawBuf buf) {
|
||||
if (visibility != Visibility.Visible)
|
||||
return;
|
||||
super.onDraw(buf);
|
||||
Rect rc = _pos;
|
||||
applyMargins(rc);
|
||||
applyPadding(rc);
|
||||
auto saver = ClipRectSaver(buf, rc);
|
||||
for (int i = 0; i < _children.count; i++) {
|
||||
Widget item = _children.get(i);
|
||||
if (item.visibility != Visibility.Visible)
|
||||
continue;
|
||||
item.onDraw(buf);
|
||||
}
|
||||
}
|
||||
|
||||
protected string _selectedTabId;
|
||||
|
||||
void selectTab(int index) {
|
||||
if (_children.get(index + 1).compareId(_selectedTabId))
|
||||
return; // already selected
|
||||
string previousSelectedTab = _selectedTabId;
|
||||
for (int i = 1; i < _children.count; i++) {
|
||||
if (index == i - 1) {
|
||||
_children.get(i).state = State.Selected;
|
||||
_selectedTabId = _children.get(i).id;
|
||||
} else {
|
||||
_children.get(i).state = State.Normal;
|
||||
}
|
||||
}
|
||||
if (_onTabChanged !is null)
|
||||
_onTabChanged(_selectedTabId, previousSelectedTab);
|
||||
}
|
||||
}
|
||||
|
||||
/// container for widgets controlled by TabControl
|
||||
|
@ -396,12 +396,25 @@ class TabHost : FrameLayout, TabHandler {
|
|||
return this;
|
||||
}
|
||||
/// select tab
|
||||
void selectTab(string ID) {
|
||||
int index = _tabControl.tabIndex(ID);
|
||||
if (index != -1) {
|
||||
_tabControl.selectTab(index);
|
||||
}
|
||||
}
|
||||
void selectTab(string ID) {
|
||||
int index = _tabControl.tabIndex(ID);
|
||||
if (index != -1) {
|
||||
_tabControl.selectTab(index);
|
||||
}
|
||||
}
|
||||
// /// request relayout of widget and its children
|
||||
// override void requestLayout() {
|
||||
// Log.d("TabHost.requestLayout called");
|
||||
// super.requestLayout();
|
||||
// //_needLayout = true;
|
||||
// }
|
||||
// /// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
// override void layout(Rect rc) {
|
||||
// Log.d("TabHost.layout() called");
|
||||
// super.layout(rc);
|
||||
// Log.d("after layout(): needLayout = ", needLayout);
|
||||
// }
|
||||
|
||||
}
|
||||
|
||||
/// compound widget - contains from TabControl widget (tabs header) and TabHost (content pages)
|
||||
|
@ -444,7 +457,13 @@ class TabWidget : VerticalLayout, TabHandler {
|
|||
return this;
|
||||
}
|
||||
/// select tab
|
||||
void selectTab(string ID) {
|
||||
_tabHost.selectTab(ID);
|
||||
}
|
||||
void selectTab(string ID) {
|
||||
_tabHost.selectTab(ID);
|
||||
}
|
||||
// /// Set widget rectangle to specified value and layout widget contents. (Step 2 of two phase layout).
|
||||
// override void layout(Rect rc) {
|
||||
// Log.d("TabWidget.layout() called");
|
||||
// super.layout(rc);
|
||||
// Log.d("after layout(): tabhost.needLayout = ", _tabHost.needLayout);
|
||||
// }
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue