diff --git a/.gitignore b/.gitignore index dc269a50..863ad45e 100644 --- a/.gitignore +++ b/.gitignore @@ -1,5 +1,10 @@ Debug Release +Unittest ui.log obj - +*.suo +Thumbs.db +.dub +bin +*.obj diff --git a/README.md b/README.md index 9264bbdd..ce90255d 100644 --- a/README.md +++ b/README.md @@ -65,6 +65,7 @@ Hello World // setup resource directories - will use only existing directories drawableCache.setResourcePaths(resourceDirs); + // optinally setup internatilnalization (if used) // setup i18n - look for i18n directory inside one of passed directories i18n.findTranslationsDir(resourceDirs); // select translation file - for english language @@ -72,7 +73,10 @@ Hello World // create window Window window = Platform.instance.createWindow("My Window", null); + // create some widget to show in window window.mainWidget = (new Button()).text("Hello world"d); + // show window + window.show(); // run message loop return Platform.instance.enterMessageLoop(); } diff --git a/dlanguilib.visualdproj b/dlanguilib.visualdproj index 2d9c732d..68797e73 100644 --- a/dlanguilib.visualdproj +++ b/dlanguilib.visualdproj @@ -54,12 +54,12 @@ 0 - - + docs + docs/index.html 0 - + include 1 $(IntDir)\$(TargetName).json diff --git a/examples/example1/src/main.d b/examples/example1/src/main.d index 34e75830..c132388b 100644 --- a/examples/example1/src/main.d +++ b/examples/example1/src/main.d @@ -12,8 +12,10 @@ extern (C) int UIAppMain(string[] args) { // resource directory search paths string[] resourceDirs = [ appendPath(exePath, "../../../res/"), // for Visual D and DUB builds + appendPath(exePath, "../../../res/mdpi/"), // for Visual D and DUB builds appendPath(exePath, "../../../../res/"),// for Mono-D builds - appendPath(exePath, "res/") // when res dir is located at the same directory as executable + appendPath(exePath, "../../../../res/mdpi/"),// for Mono-D builds + appendPath(exePath, "res/mdpi/") // when res dir is located at the same directory as executable ]; // setup resource directories - will use only existing directories drawableCache.setResourcePaths(resourceDirs); @@ -70,9 +72,9 @@ extern (C) int UIAppMain(string[] args) { LinearLayout hlayout = new HorizontalLayout(); //hlayout.addChild((new Button()).text("<<")); //.textColor(0x40FF4000) hlayout.addChild((new TextWidget()).text("Several").alignment(Align.Center)); - hlayout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5)).alignment(Align.Center)); + hlayout.addChild((new ImageWidget()).drawableId("btn_radio").padding(Rect(5,5,5,5)).alignment(Align.Center)); hlayout.addChild((new TextWidget()).text("items").alignment(Align.Center)); - hlayout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5)).alignment(Align.Center)); + hlayout.addChild((new ImageWidget()).drawableId("btn_check").padding(Rect(5,5,5,5)).alignment(Align.Center)); hlayout.addChild((new TextWidget()).text("in horizontal layout")); hlayout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5)).alignment(Align.Center)); //hlayout.addChild((new Button()).text(">>")); //.textColor(0x40FF4000) @@ -84,6 +86,9 @@ extern (C) int UIAppMain(string[] args) { vlayout.addChild((new TextWidget()).text("VLayout line 1").textColor(0x40FF4000)); // vlayout.addChild((new TextWidget()).text("VLayout line 2").textColor(0x40FF8000)); vlayout.addChild((new TextWidget()).text("VLayout line 2").textColor(0x40008000)); + vlayout.addChild(new RadioButton("rb1", "Radio button 1"d)); + vlayout.addChild(new RadioButton("rb2", "Radio button 2"d)); + vlayout.addChild(new RadioButton("rb3", "Radio button 3"d)); vlayout.layoutWidth(FILL_PARENT); vlayoutgroup.addChild(vlayout); vlayoutgroup.layoutWidth(FILL_PARENT); @@ -94,12 +99,13 @@ extern (C) int UIAppMain(string[] args) { ScrollBar sb = new ScrollBar("hscroll", Orientation.Horizontal); layout.addChild(sb.layoutHeight(WRAP_CONTENT).layoutWidth(FILL_PARENT)); - layout.addChild((new Button("BTN2")).textColor(0x000000FF).text("Button2")); + layout.addChild((new CheckBox("BTN2", "Some checkbox"d))); layout.addChild((new TextWidget()).textColor(0x40FF4000).text("Text widget")); layout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5))); layout.addChild((new TextWidget()).textColor(0xFF4000).text("Text widget2").padding(Rect(5,5,5,5)).margins(Rect(5,5,5,5)).backgroundColor(0xA0A0A0)); - layout.addChild((new Button("BTN3")).textColor(0x000000FF).text("Button3").layoutHeight(FILL_PARENT)); - layout.addChild((new TextWidget()).textColor(0x004000).text("Text widget3 with very long text")); + layout.addChild((new RadioButton("BTN3", "Some radio button"d))); + layout.addChild((new TextWidget(null, "Text widget3 with very long text"d)).textColor(0x004000)); + layout.addChild(new VSpacer()); // vertical spacer to fill extra space layout.childById("BTN1").onClickListener = (delegate (Widget w) { Log.d("onClick ", w.id); return true; }); layout.childById("BTN2").onClickListener = (delegate (Widget w) { Log.d("onClick ", w.id); return true; }); diff --git a/examples/helloworld/src/app.d b/examples/helloworld/src/app.d index b33b09b1..f60a5202 100644 --- a/examples/helloworld/src/app.d +++ b/examples/helloworld/src/app.d @@ -25,10 +25,12 @@ extern (C) int UIAppMain(string[] args) { // create window Window window = Platform.instance.createWindow("My Window", null); - window.mainWidget = (new Button()).text("sample button"); + // create some widget to show in window + window.mainWidget = (new Button()).text("Hello world"d); + // show window window.show(); - //window.windowCaption = "New Window Caption"; + // run message loop return Platform.instance.enterMessageLoop(); } diff --git a/res/btn_check.xml b/res/btn_check.xml new file mode 100644 index 00000000..c8c2bee0 --- /dev/null +++ b/res/btn_check.xml @@ -0,0 +1,65 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/res/btn_radio.xml b/res/btn_radio.xml new file mode 100644 index 00000000..e2f1f84c --- /dev/null +++ b/res/btn_radio.xml @@ -0,0 +1,59 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/res/btn_radio_background.xml b/res/btn_radio_background.xml new file mode 100644 index 00000000..005b6fc7 --- /dev/null +++ b/res/btn_radio_background.xml @@ -0,0 +1,11 @@ + + + + + + diff --git a/res/mdpi/btn_check_off_disabled_focused_holo_light.png b/res/mdpi/btn_check_off_disabled_focused_holo_light.png new file mode 100644 index 00000000..42442e8f Binary files /dev/null and b/res/mdpi/btn_check_off_disabled_focused_holo_light.png differ diff --git a/res/mdpi/btn_check_off_disabled_holo_light.png b/res/mdpi/btn_check_off_disabled_holo_light.png new file mode 100644 index 00000000..db190430 Binary files /dev/null and b/res/mdpi/btn_check_off_disabled_holo_light.png differ diff --git a/res/mdpi/btn_check_off_focused_holo_light.png b/res/mdpi/btn_check_off_focused_holo_light.png new file mode 100644 index 00000000..3fc71fd2 Binary files /dev/null and b/res/mdpi/btn_check_off_focused_holo_light.png differ diff --git a/res/mdpi/btn_check_off_holo_light.png b/res/mdpi/btn_check_off_holo_light.png new file mode 100644 index 00000000..5e04a57b Binary files /dev/null and b/res/mdpi/btn_check_off_holo_light.png differ diff --git a/res/mdpi/btn_check_off_normal_holo_light.png b/res/mdpi/btn_check_off_normal_holo_light.png new file mode 100644 index 00000000..b39ad3d6 Binary files /dev/null and b/res/mdpi/btn_check_off_normal_holo_light.png differ diff --git a/res/mdpi/btn_check_off_pressed_holo_light.png b/res/mdpi/btn_check_off_pressed_holo_light.png new file mode 100644 index 00000000..1109c20f Binary files /dev/null and b/res/mdpi/btn_check_off_pressed_holo_light.png differ diff --git a/res/mdpi/btn_check_on_disabled_focused_holo_light.png b/res/mdpi/btn_check_on_disabled_focused_holo_light.png new file mode 100644 index 00000000..997045de Binary files /dev/null and b/res/mdpi/btn_check_on_disabled_focused_holo_light.png differ diff --git a/res/mdpi/btn_check_on_disabled_holo_light.png b/res/mdpi/btn_check_on_disabled_holo_light.png new file mode 100644 index 00000000..20e2aab1 Binary files /dev/null and b/res/mdpi/btn_check_on_disabled_holo_light.png differ diff --git a/res/mdpi/btn_check_on_focused_holo_light.png b/res/mdpi/btn_check_on_focused_holo_light.png new file mode 100644 index 00000000..a3b49165 Binary files /dev/null and b/res/mdpi/btn_check_on_focused_holo_light.png differ diff --git a/res/mdpi/btn_check_on_holo_light.png b/res/mdpi/btn_check_on_holo_light.png new file mode 100644 index 00000000..f657c5b2 Binary files /dev/null and b/res/mdpi/btn_check_on_holo_light.png differ diff --git a/res/mdpi/btn_check_on_pressed_holo_light.png b/res/mdpi/btn_check_on_pressed_holo_light.png new file mode 100644 index 00000000..e9f5f06c Binary files /dev/null and b/res/mdpi/btn_check_on_pressed_holo_light.png differ diff --git a/res/mdpi/btn_radio_off_disabled_focused_holo_light.png b/res/mdpi/btn_radio_off_disabled_focused_holo_light.png new file mode 100644 index 00000000..4dac84cf Binary files /dev/null and b/res/mdpi/btn_radio_off_disabled_focused_holo_light.png differ diff --git a/res/mdpi/btn_radio_off_disabled_holo_light.png b/res/mdpi/btn_radio_off_disabled_holo_light.png new file mode 100644 index 00000000..a67375ec Binary files /dev/null and b/res/mdpi/btn_radio_off_disabled_holo_light.png differ diff --git a/res/mdpi/btn_radio_off_focused_holo_light.png b/res/mdpi/btn_radio_off_focused_holo_light.png new file mode 100644 index 00000000..6753d087 Binary files /dev/null and b/res/mdpi/btn_radio_off_focused_holo_light.png differ diff --git a/res/mdpi/btn_radio_off_holo_light.png b/res/mdpi/btn_radio_off_holo_light.png new file mode 100644 index 00000000..665cb171 Binary files /dev/null and b/res/mdpi/btn_radio_off_holo_light.png differ diff --git a/res/mdpi/btn_radio_off_pressed_holo_light.png b/res/mdpi/btn_radio_off_pressed_holo_light.png new file mode 100644 index 00000000..2a7d0d52 Binary files /dev/null and b/res/mdpi/btn_radio_off_pressed_holo_light.png differ diff --git a/res/mdpi/btn_radio_on_disabled_focused_holo_light.png b/res/mdpi/btn_radio_on_disabled_focused_holo_light.png new file mode 100644 index 00000000..c9be37e9 Binary files /dev/null and b/res/mdpi/btn_radio_on_disabled_focused_holo_light.png differ diff --git a/res/mdpi/btn_radio_on_disabled_holo_light.png b/res/mdpi/btn_radio_on_disabled_holo_light.png new file mode 100644 index 00000000..4583c3e1 Binary files /dev/null and b/res/mdpi/btn_radio_on_disabled_holo_light.png differ diff --git a/res/mdpi/btn_radio_on_focused_holo_light.png b/res/mdpi/btn_radio_on_focused_holo_light.png new file mode 100644 index 00000000..db3b30a1 Binary files /dev/null and b/res/mdpi/btn_radio_on_focused_holo_light.png differ diff --git a/res/mdpi/btn_radio_on_holo_light.png b/res/mdpi/btn_radio_on_holo_light.png new file mode 100644 index 00000000..53dc1536 Binary files /dev/null and b/res/mdpi/btn_radio_on_holo_light.png differ diff --git a/res/mdpi/btn_radio_on_pressed_holo_light.png b/res/mdpi/btn_radio_on_pressed_holo_light.png new file mode 100644 index 00000000..34dd059b Binary files /dev/null and b/res/mdpi/btn_radio_on_pressed_holo_light.png differ diff --git a/src/dlangui/all.d b/src/dlangui/all.d index bdb984fa..1f7495be 100644 --- a/src/dlangui/all.d +++ b/src/dlangui/all.d @@ -21,8 +21,10 @@ extern (C) int UIAppMain(string[] args) { appendPath(exePath, "../../../../res/"), // for Mono-D builds appendPath(exePath, "res/") // when res dir is located at the same directory as executable ]; + // setup resource directories - will use only existing directories drawableCache.setResourcePaths(resourceDirs); + // setup i18n - look for i18n directory inside one of passed directories i18n.findTranslationsDir(resourceDirs); // select translation file - for english language @@ -30,7 +32,10 @@ extern (C) int UIAppMain(string[] args) { // create window Window window = Platform.instance.createWindow("My Window", null); + // create some widget to show in window window.mainWidget = (new Button()).text("Hello world"d); + // show window + window.show(); // run message loop return Platform.instance.enterMessageLoop(); } diff --git a/src/dlangui/graphics/resources.d b/src/dlangui/graphics/resources.d index 6099b40a..6ce5605c 100644 --- a/src/dlangui/graphics/resources.d +++ b/src/dlangui/graphics/resources.d @@ -351,6 +351,8 @@ class StateDrawable : Drawable { foreach(item; element.elements) { if (item.tag.name.equal("item")) { string drawableId = attrValue(item, "drawable", "android:drawable"); + if (drawableId.startsWith("@drawable/")) + drawableId = drawableId[10 .. $]; ColorTransform transform; transform.addBefore = colorTransformFromStringAdd(attrValue(item, "color_transform_add1", "android:transform_color_add1")); transform.multiply = colorTransformFromStringMult(attrValue(item, "color_transform_mul", "android:transform_color_mul")); @@ -394,8 +396,12 @@ class StateDrawable : Drawable { override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) { foreach(ref item; _stateList) if (item.matchState(state)) { - if (!item.drawable.isNull) + if (!item.drawable.isNull) { + if (state & State.Checked) { + Log.d("Found item for checked state: ", item.stateMask, " ", item.stateValue); + } item.drawable.drawTo(buf, rc, state, tilex0, tiley0); + } return; } } @@ -620,24 +626,28 @@ class DrawableCache { return _drawable; if (_filename !is null) { // reload from file - if (_filename.endsWith(".xml")) { + if (_filename.endsWith(".xml") || _filename.endsWith(".XML")) { // XML drawables support StateDrawable d = new StateDrawable(); if (!d.load(_filename)) { + Log.e("failed to load .xml drawable from ", _filename); destroy(d); _error = true; } else { + Log.d("loaded .xml drawable from ", _filename); _drawable = d; } } else { // PNG/JPEG drawables support DrawBufRef image = imageCache.get(_filename, transform); if (!image.isNull) { - bool ninePatch = _filename.endsWith(".9.png"); + bool ninePatch = _filename.endsWith(".9.png") || _filename.endsWith(".9.PNG"); _transformed[transform] = new ImageDrawable(image, _tiled, ninePatch); return _transformed[transform]; - } else + } else { + Log.e("failed to load image from ", _filename); _error = true; + } } } return _drawable; @@ -743,6 +753,8 @@ class DrawableCache { if (fn !is null) { _idToFileMap[id] = fn; return fn; + } else { + Log.w("resource ", id, " is not found"); } } return null; diff --git a/src/dlangui/platforms/windows/win32fonts.d b/src/dlangui/platforms/windows/win32fonts.d index 1b2f5a07..3a7400eb 100644 --- a/src/dlangui/platforms/windows/win32fonts.d +++ b/src/dlangui/platforms/windows/win32fonts.d @@ -286,9 +286,11 @@ class Win32Font : Font { lf.lfFaceName[def.face.length] = 0; lf.lfHeight = -size; lf.lfItalic = italic; - lf.lfOutPrecision = OUT_TT_ONLY_PRECIS; + lf.lfOutPrecision = OUT_OUTLINE_PRECIS; //OUT_TT_ONLY_PRECIS; lf.lfClipPrecision = CLIP_DEFAULT_PRECIS; - lf.lfQuality = ANTIALIASED_QUALITY; + //lf.lfQuality = NONANTIALIASED_QUALITY; //ANTIALIASED_QUALITY; + //lf.lfQuality = PROOF_QUALITY; //ANTIALIASED_QUALITY; + lf.lfQuality = size < 18 ? NONANTIALIASED_QUALITY : PROOF_QUALITY; //ANTIALIASED_QUALITY; lf.lfPitchAndFamily = def.pitchAndFamily; _hfont = CreateFontIndirectA(&lf); _drawbuf = new Win32ColorDrawBuf(1, 1); diff --git a/src/dlangui/widgets/controls.d b/src/dlangui/widgets/controls.d index 9405faf1..e0b97828 100644 --- a/src/dlangui/widgets/controls.d +++ b/src/dlangui/widgets/controls.d @@ -30,14 +30,33 @@ Authors: $(WEB coolreader.org, Vadim Lopatin) module dlangui.widgets.controls; import dlangui.widgets.widget; +import dlangui.widgets.layouts; +/// vertical spacer to fill empty space in vertical layouts +class VSpacer : Widget { + this() { + styleId = "VSpacer"; + } +} +/// horizontal spacer to fill empty space in horizontal layouts +class HSpacer : Widget { + this() { + styleId = "HSpacer"; + } +} /// static text widget class TextWidget : Widget { - this(string ID = null) { + this(string ID = null, string textResourceId = null) { super(ID); styleId = "TEXT"; + _text = textResourceId; + } + this(string ID, dstring rawText) { + super(ID); + styleId = "TEXT"; + _text = rawText; } protected UIString _text; /// get widget text @@ -49,7 +68,7 @@ class TextWidget : Widget { return this; } /// set text to show - @property Widget text(ref UIString s) { + override @property Widget text(ref UIString s) { _text = s; requestLayout(); return this; @@ -152,7 +171,11 @@ class ImageWidget : Widget { sz.x = img.width; sz.y = img.height; applyAlign(rc, sz); - img.drawTo(buf, rc, state); + uint st = state; + if (state & State.Checked) { + Log.d("Drawing image for checked state"); + } + img.drawTo(buf, rc, st); } } } @@ -160,18 +183,98 @@ class ImageWidget : Widget { /// button with image only class ImageButton : ImageWidget { this(string ID = null, string drawableId = null) { - super(ID); + super(ID, drawableId); styleId = "BUTTON"; _drawableId = drawableId; + clickable = true; focusable = true; trackHover = true; } } +/// button with image and text +class ImageTextButton : HorizontalLayout { + protected ImageWidget _icon; + protected TextWidget _label; + override @property dstring text() { return _label.text; } + override @property Widget text(dstring s) { _label.text = s; requestLayout(); return this; } + override @property Widget text(ref UIString s) { _label.text = s; requestLayout(); return this; } + this(string ID = null, string drawableId = null, string textResourceId = null) { + super(ID); + styleId = "BUTTON"; + _icon = new ImageWidget("icon", drawableId); + _label = new TextWidget("label", textResourceId); + _label.styleId = "BUTTON_LABEL"; + _icon.state = State.Parent; + _label.state = State.Parent; + addChild(_icon); + addChild(_label); + clickable = true; + focusable = true; + trackHover = true; + } + this(string ID, string drawableId, dstring rawText) { + super(ID); + styleId = "BUTTON"; + _icon = new ImageWidget("icon", drawableId); + _label = new TextWidget("label", rawText); + _label.styleId = "BUTTON_LABEL"; + _icon.styleId = "BUTTON_ICON"; + _icon.state = State.Parent; + _label.state = State.Parent; + addChild(_icon); + addChild(_label); + clickable = true; + focusable = true; + trackHover = true; + } +} + +/// checkbox +class CheckBox : ImageTextButton { + this(string ID = null, string textResourceId = null) { + super(ID, "btn_check", textResourceId); + styleId = "TRANSPARENT_BUTTON_BACKGROUND"; + checkable = true; + } + this(string ID, dstring labelText) { + super(ID, "btn_check", labelText); + styleId = "TRANSPARENT_BUTTON_BACKGROUND"; + checkable = true; + } + // called to process click and notify listeners + override protected bool handleClick() { + checked = !checked; + return super.handleClick(); + } +} + +/// radio button +class RadioButton : ImageTextButton { + this(string ID = null, string textResourceId = null) { + super(ID, "btn_radio", textResourceId); + styleId = "TRANSPARENT_BUTTON_BACKGROUND"; + checkable = true; + } + this(string ID, dstring labelText) { + super(ID, "btn_radio", labelText); + styleId = "TRANSPARENT_BUTTON_BACKGROUND"; + checkable = true; + } + // called to process click and notify listeners + override protected bool handleClick() { + checked = true; + return super.handleClick(); + } + +} + +/// Text only button class Button : Widget { protected UIString _text; override @property dstring text() { return _text; } override @property Widget text(dstring s) { _text = s; requestLayout(); return this; } + override @property Widget text(ref UIString s) { _text = s; requestLayout(); return this; } @property Widget textResource(string s) { _text = s; requestLayout(); return this; } this(string ID = null) { super(ID); diff --git a/src/dlangui/widgets/styles.d b/src/dlangui/widgets/styles.d index 6cceabe3..18d8dfdc 100644 --- a/src/dlangui/widgets/styles.d +++ b/src/dlangui/widgets/styles.d @@ -523,10 +523,11 @@ class Theme : Style { _backgroundColor = 0xFFFFFFFF; // transparent _textColor = 0x000000; // black _align = Align.TopLeft; - _fontSize = 24; // TODO: from settings or screen properties / DPI + _fontSize = 14; // TODO: from settings or screen properties / DPI _fontStyle = FONT_STYLE_NORMAL; _fontWeight = 400; - _fontFace = "Arial"; // TODO: from settings + //_fontFace = "Arial"; // TODO: from settings + _fontFace = "Verdana"; // TODO: from settings _fontFamily = FontFamily.SansSerif; _minHeight = 0; _minWidth = 0; @@ -635,10 +636,18 @@ Theme createDefaultTheme() { Log.d("Creating default theme"); Theme res = new Theme("default"); //res.fontSize(14); - res.fontSize(24); + version (Windows) { + res.fontFace = "Verdana"; + } + //res.fontFace = "Arial Narrow"; + res.fontSize = 13; // TODO: choose based on DPI Style button = res.createSubstyle("BUTTON").backgroundImageId("btn_default_small").alignment(Align.Center); - Style buttonTransparent = res.createSubstyle("BUTTON_TRANSPARENT").backgroundImageId("btn_default_small_transparent").alignment(Align.Center); - Style text = res.createSubstyle("TEXT").margins(Rect(2,2,2,2)).padding(Rect(1,1,1,1)); + res.createSubstyle("BUTTON_TRANSPARENT").backgroundImageId("btn_default_small_transparent").alignment(Align.Center); + res.createSubstyle("BUTTON_LABEL").layoutWidth(FILL_PARENT).alignment(Align.Left|Align.VCenter); + res.createSubstyle("BUTTON_ICON").alignment(Align.Center); + res.createSubstyle("TEXT").margins(Rect(2,2,2,2)).padding(Rect(1,1,1,1)); + res.createSubstyle("HSPACER").layoutWidth(FILL_PARENT).layoutWeight(100); + res.createSubstyle("VSPACER").layoutHeight(FILL_PARENT).layoutWeight(100); //button.createState(State.Enabled | State.Focused, State.Focused).backgroundImageId("btn_default_small_normal_disable_focused"); //button.createState(State.Enabled, 0).backgroundImageId("btn_default_small_normal_disable"); //button.createState(State.Pressed, State.Pressed).backgroundImageId("btn_default_small_pressed"); @@ -692,6 +701,12 @@ Theme createDefaultTheme() { menuItem.createState(State.Selected, State.Selected).backgroundColor(0x00F8F9Fa); menuItem.createState(State.Hovered, State.Hovered).backgroundColor(0xC0FFFF00); + Style transparentButtonBackground = res.createSubstyle("TRANSPARENT_BUTTON_BACKGROUND").padding(Rect(4,2,4,2)); //.backgroundColor(0xE0E080) ; + transparentButtonBackground.createState(State.Focused, State.Focused).backgroundColor(0x40C0C000); + transparentButtonBackground.createState(State.Pressed, State.Pressed).backgroundColor(0x4080C000); + transparentButtonBackground.createState(State.Selected, State.Selected).backgroundColor(0x00F8F9Fa); + transparentButtonBackground.createState(State.Hovered, State.Hovered).backgroundColor(0xC0FFFF00); + Style poopupMenu = res.createSubstyle("POPUP_MENU").backgroundImageId("popup_menu_background_normal"); Style listItem = res.createSubstyle("LIST_ITEM"); diff --git a/src/dlangui/widgets/widget.d b/src/dlangui/widgets/widget.d index f246258d..a3f199b5 100644 --- a/src/dlangui/widgets/widget.d +++ b/src/dlangui/widgets/widget.d @@ -259,6 +259,8 @@ class Widget { @property dstring text() { return ""; } /// sets widget content text (override to support this) @property Widget text(dstring s) { return this; } + /// sets widget content text (override to support this) + @property Widget text(ref UIString s) { return this; } //================================================================== // Layout and drawing related methods @@ -337,12 +339,39 @@ class Widget { return _pos.isPointInside(x, y); } + protected bool _clickable; + @property bool clickable() { return _clickable; } + @property Widget clickable(bool flg) { _clickable = flg; return this; } + + protected bool _checkable; + @property bool checkable() { return _checkable; } + @property Widget checkable(bool flg) { _checkable = flg; return this; } + + protected bool _checked; + /// get checked state + @property bool checked() { return (state & State.Checked) != 0; } + /// set checked state + @property Widget checked(bool flg) { + if (flg != checked) { + if (flg) + setState(State.Checked); + else + resetState(State.Checked); + invalidate(); + } + return this; + } + protected bool _focusable; @property bool focusable() { return _focusable; } @property Widget focusable(bool flg) { _focusable = flg; return this; } + @property bool focused() { return (window !is null && window.focusedWidget is this && (state & State.Focused)); } + + + /// returns true if this widget and all its parents are visible @property bool visible() { if (visibility != Visibility.Visible) @@ -390,9 +419,15 @@ class Widget { // ======================================================= // Events + // called to process click and notify listeners + protected bool handleClick() { + bool res = onClickListener(this); + return res; + } + /// process key event, return true if event is processed. bool onKeyEvent(KeyEvent event) { - if (onClickListener.assigned) { + if (clickable) { // support onClick event initiated by Space or Return keys if (event.action == KeyAction.KeyDown) { if (event.keyCode == KeyCode.SPACE || event.keyCode == KeyCode.RETURN) { @@ -403,7 +438,7 @@ class Widget { if (event.action == KeyAction.KeyUp) { if (event.keyCode == KeyCode.SPACE || event.keyCode == KeyCode.RETURN) { resetState(State.Pressed); - onClickListener(this); + handleClick(); return true; } } @@ -415,7 +450,7 @@ class Widget { bool onMouseEvent(MouseEvent event) { //Log.d("onMouseEvent ", id, " ", event.action, " (", event.x, ",", event.y, ")"); // support onClick - if (onClickListener.assigned) { + if (clickable) { if (event.action == MouseAction.ButtonDown && event.button == MouseButton.Left) { setState(State.Pressed); if (focusable) @@ -424,7 +459,7 @@ class Widget { } if (event.action == MouseAction.ButtonUp && event.button == MouseButton.Left) { resetState(State.Pressed); - onClickListener(this); + handleClick(); return true; } if (event.action == MouseAction.FocusOut || event.action == MouseAction.Cancel) {