implement CheckBox and RadioButton controls
|
@ -1,5 +1,10 @@
|
||||||
Debug
|
Debug
|
||||||
Release
|
Release
|
||||||
|
Unittest
|
||||||
ui.log
|
ui.log
|
||||||
obj
|
obj
|
||||||
|
*.suo
|
||||||
|
Thumbs.db
|
||||||
|
.dub
|
||||||
|
bin
|
||||||
|
*.obj
|
||||||
|
|
|
@ -65,6 +65,7 @@ Hello World
|
||||||
// setup resource directories - will use only existing directories
|
// setup resource directories - will use only existing directories
|
||||||
drawableCache.setResourcePaths(resourceDirs);
|
drawableCache.setResourcePaths(resourceDirs);
|
||||||
|
|
||||||
|
// optinally setup internatilnalization (if used)
|
||||||
// setup i18n - look for i18n directory inside one of passed directories
|
// setup i18n - look for i18n directory inside one of passed directories
|
||||||
i18n.findTranslationsDir(resourceDirs);
|
i18n.findTranslationsDir(resourceDirs);
|
||||||
// select translation file - for english language
|
// select translation file - for english language
|
||||||
|
@ -72,7 +73,10 @@ Hello World
|
||||||
|
|
||||||
// create window
|
// create window
|
||||||
Window window = Platform.instance.createWindow("My Window", null);
|
Window window = Platform.instance.createWindow("My Window", null);
|
||||||
|
// create some widget to show in window
|
||||||
window.mainWidget = (new Button()).text("Hello world"d);
|
window.mainWidget = (new Button()).text("Hello world"d);
|
||||||
|
// show window
|
||||||
|
window.show();
|
||||||
// run message loop
|
// run message loop
|
||||||
return Platform.instance.enterMessageLoop();
|
return Platform.instance.enterMessageLoop();
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,12 +54,12 @@
|
||||||
<objname />
|
<objname />
|
||||||
<libname />
|
<libname />
|
||||||
<doDocComments>0</doDocComments>
|
<doDocComments>0</doDocComments>
|
||||||
<docdir />
|
<docdir>docs</docdir>
|
||||||
<docname />
|
<docname>docs/index.html</docname>
|
||||||
<modules_ddoc />
|
<modules_ddoc />
|
||||||
<ddocfiles />
|
<ddocfiles />
|
||||||
<doHdrGeneration>0</doHdrGeneration>
|
<doHdrGeneration>0</doHdrGeneration>
|
||||||
<hdrdir />
|
<hdrdir>include</hdrdir>
|
||||||
<hdrname />
|
<hdrname />
|
||||||
<doXGeneration>1</doXGeneration>
|
<doXGeneration>1</doXGeneration>
|
||||||
<xfilename>$(IntDir)\$(TargetName).json</xfilename>
|
<xfilename>$(IntDir)\$(TargetName).json</xfilename>
|
||||||
|
|
|
@ -12,8 +12,10 @@ extern (C) int UIAppMain(string[] args) {
|
||||||
// resource directory search paths
|
// resource directory search paths
|
||||||
string[] resourceDirs = [
|
string[] resourceDirs = [
|
||||||
appendPath(exePath, "../../../res/"), // for Visual D and DUB builds
|
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/"),// 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
|
// setup resource directories - will use only existing directories
|
||||||
drawableCache.setResourcePaths(resourceDirs);
|
drawableCache.setResourcePaths(resourceDirs);
|
||||||
|
@ -70,9 +72,9 @@ extern (C) int UIAppMain(string[] args) {
|
||||||
LinearLayout hlayout = new HorizontalLayout();
|
LinearLayout hlayout = new HorizontalLayout();
|
||||||
//hlayout.addChild((new Button()).text("<<")); //.textColor(0x40FF4000)
|
//hlayout.addChild((new Button()).text("<<")); //.textColor(0x40FF4000)
|
||||||
hlayout.addChild((new TextWidget()).text("Several").alignment(Align.Center));
|
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 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 TextWidget()).text("in horizontal layout"));
|
||||||
hlayout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5)).alignment(Align.Center));
|
hlayout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5)).alignment(Align.Center));
|
||||||
//hlayout.addChild((new Button()).text(">>")); //.textColor(0x40FF4000)
|
//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 1").textColor(0x40FF4000)); //
|
||||||
vlayout.addChild((new TextWidget()).text("VLayout line 2").textColor(0x40FF8000));
|
vlayout.addChild((new TextWidget()).text("VLayout line 2").textColor(0x40FF8000));
|
||||||
vlayout.addChild((new TextWidget()).text("VLayout line 2").textColor(0x40008000));
|
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);
|
vlayout.layoutWidth(FILL_PARENT);
|
||||||
vlayoutgroup.addChild(vlayout);
|
vlayoutgroup.addChild(vlayout);
|
||||||
vlayoutgroup.layoutWidth(FILL_PARENT);
|
vlayoutgroup.layoutWidth(FILL_PARENT);
|
||||||
|
@ -94,12 +99,13 @@ extern (C) int UIAppMain(string[] args) {
|
||||||
ScrollBar sb = new ScrollBar("hscroll", Orientation.Horizontal);
|
ScrollBar sb = new ScrollBar("hscroll", Orientation.Horizontal);
|
||||||
layout.addChild(sb.layoutHeight(WRAP_CONTENT).layoutWidth(FILL_PARENT));
|
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 TextWidget()).textColor(0x40FF4000).text("Text widget"));
|
||||||
layout.addChild((new ImageWidget()).drawableId("exit").padding(Rect(5,5,5,5)));
|
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 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 RadioButton("BTN3", "Some radio button"d)));
|
||||||
layout.addChild((new TextWidget()).textColor(0x004000).text("Text widget3 with very long text"));
|
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("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; });
|
layout.childById("BTN2").onClickListener = (delegate (Widget w) { Log.d("onClick ", w.id); return true; });
|
||||||
|
|
|
@ -25,10 +25,12 @@ extern (C) int UIAppMain(string[] args) {
|
||||||
// create window
|
// create window
|
||||||
Window window = Platform.instance.createWindow("My Window", null);
|
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.show();
|
||||||
//window.windowCaption = "New Window Caption";
|
|
||||||
// run message loop
|
// run message loop
|
||||||
return Platform.instance.enterMessageLoop();
|
return Platform.instance.enterMessageLoop();
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,65 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright (C) 2011 The Android Open Source Project
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<selector xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
|
||||||
|
<!-- Enabled states -->
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_window_focused="false"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_on_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_window_focused="false"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_off_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_pressed="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_on_pressed_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_pressed="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_off_pressed_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_focused="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_on_focused_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_focused="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_off_focused_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="false"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_off_holo_light" />
|
||||||
|
<item android:state_checked="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_check_on_holo_light" />
|
||||||
|
|
||||||
|
|
||||||
|
<!-- Disabled states -->
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_window_focused="false"
|
||||||
|
android:drawable="@drawable/btn_check_on_disabled_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_window_focused="false"
|
||||||
|
android:drawable="@drawable/btn_check_off_disabled_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_focused="true"
|
||||||
|
android:drawable="@drawable/btn_check_on_disabled_focused_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_focused="true"
|
||||||
|
android:drawable="@drawable/btn_check_off_disabled_focused_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="false" android:drawable="@drawable/btn_check_off_disabled_holo_light" />
|
||||||
|
<item android:state_checked="true" android:drawable="@drawable/btn_check_on_disabled_holo_light" />
|
||||||
|
|
||||||
|
</selector>
|
|
@ -0,0 +1,59 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright (C) 2011 The Android Open Source Project
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<selector xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<item android:state_checked="true" android:state_window_focused="false"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_on_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_window_focused="false"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_off_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_pressed="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_on_pressed_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_pressed="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_off_pressed_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_focused="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_on_focused_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_focused="true"
|
||||||
|
android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_off_focused_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="false" android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_off_holo_light" />
|
||||||
|
<item android:state_checked="true" android:state_enabled="true"
|
||||||
|
android:drawable="@drawable/btn_radio_on_holo_light" />
|
||||||
|
|
||||||
|
<!-- Disabled states -->
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_window_focused="false"
|
||||||
|
android:drawable="@drawable/btn_radio_on_disabled_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_window_focused="false"
|
||||||
|
android:drawable="@drawable/btn_radio_off_disabled_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="true" android:state_focused="true"
|
||||||
|
android:drawable="@drawable/btn_radio_on_disabled_focused_holo_light" />
|
||||||
|
<item android:state_checked="false" android:state_focused="true"
|
||||||
|
android:drawable="@drawable/btn_radio_off_disabled_focused_holo_light" />
|
||||||
|
|
||||||
|
<item android:state_checked="false" android:drawable="@drawable/btn_radio_off_disabled_holo_light" />
|
||||||
|
<item android:state_checked="true" android:drawable="@drawable/btn_radio_on_disabled_holo_light" />
|
||||||
|
|
||||||
|
</selector>
|
|
@ -0,0 +1,11 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<selector xmlns:android="http://schemas.android.com/apk/res/android" >
|
||||||
|
<item
|
||||||
|
android:drawable="main_menu_item_background_selected"
|
||||||
|
android:state_selected="true" />
|
||||||
|
<item
|
||||||
|
android:drawable="main_menu_item_background_hover"
|
||||||
|
android:state_hovered="true" />
|
||||||
|
<item
|
||||||
|
android:drawable="@null" />
|
||||||
|
</selector>
|
After Width: | Height: | Size: 368 B |
After Width: | Height: | Size: 329 B |
After Width: | Height: | Size: 425 B |
After Width: | Height: | Size: 323 B |
After Width: | Height: | Size: 491 B |
After Width: | Height: | Size: 1.6 KiB |
After Width: | Height: | Size: 632 B |
After Width: | Height: | Size: 508 B |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 1.5 KiB |
After Width: | Height: | Size: 1.9 KiB |
After Width: | Height: | Size: 919 B |
After Width: | Height: | Size: 479 B |
After Width: | Height: | Size: 1.0 KiB |
After Width: | Height: | Size: 533 B |
After Width: | Height: | Size: 1.8 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 736 B |
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 578 B |
After Width: | Height: | Size: 1.3 KiB |
|
@ -21,8 +21,10 @@ extern (C) int UIAppMain(string[] args) {
|
||||||
appendPath(exePath, "../../../../res/"), // for Mono-D 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/") // when res dir is located at the same directory as executable
|
||||||
];
|
];
|
||||||
|
|
||||||
// setup resource directories - will use only existing directories
|
// setup resource directories - will use only existing directories
|
||||||
drawableCache.setResourcePaths(resourceDirs);
|
drawableCache.setResourcePaths(resourceDirs);
|
||||||
|
|
||||||
// setup i18n - look for i18n directory inside one of passed directories
|
// setup i18n - look for i18n directory inside one of passed directories
|
||||||
i18n.findTranslationsDir(resourceDirs);
|
i18n.findTranslationsDir(resourceDirs);
|
||||||
// select translation file - for english language
|
// select translation file - for english language
|
||||||
|
@ -30,7 +32,10 @@ extern (C) int UIAppMain(string[] args) {
|
||||||
|
|
||||||
// create window
|
// create window
|
||||||
Window window = Platform.instance.createWindow("My Window", null);
|
Window window = Platform.instance.createWindow("My Window", null);
|
||||||
|
// create some widget to show in window
|
||||||
window.mainWidget = (new Button()).text("Hello world"d);
|
window.mainWidget = (new Button()).text("Hello world"d);
|
||||||
|
// show window
|
||||||
|
window.show();
|
||||||
// run message loop
|
// run message loop
|
||||||
return Platform.instance.enterMessageLoop();
|
return Platform.instance.enterMessageLoop();
|
||||||
}
|
}
|
||||||
|
|
|
@ -351,6 +351,8 @@ class StateDrawable : Drawable {
|
||||||
foreach(item; element.elements) {
|
foreach(item; element.elements) {
|
||||||
if (item.tag.name.equal("item")) {
|
if (item.tag.name.equal("item")) {
|
||||||
string drawableId = attrValue(item, "drawable", "android:drawable");
|
string drawableId = attrValue(item, "drawable", "android:drawable");
|
||||||
|
if (drawableId.startsWith("@drawable/"))
|
||||||
|
drawableId = drawableId[10 .. $];
|
||||||
ColorTransform transform;
|
ColorTransform transform;
|
||||||
transform.addBefore = colorTransformFromStringAdd(attrValue(item, "color_transform_add1", "android:transform_color_add1"));
|
transform.addBefore = colorTransformFromStringAdd(attrValue(item, "color_transform_add1", "android:transform_color_add1"));
|
||||||
transform.multiply = colorTransformFromStringMult(attrValue(item, "color_transform_mul", "android:transform_color_mul"));
|
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) {
|
override void drawTo(DrawBuf buf, Rect rc, uint state = 0, int tilex0 = 0, int tiley0 = 0) {
|
||||||
foreach(ref item; _stateList)
|
foreach(ref item; _stateList)
|
||||||
if (item.matchState(state)) {
|
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);
|
item.drawable.drawTo(buf, rc, state, tilex0, tiley0);
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -620,24 +626,28 @@ class DrawableCache {
|
||||||
return _drawable;
|
return _drawable;
|
||||||
if (_filename !is null) {
|
if (_filename !is null) {
|
||||||
// reload from file
|
// reload from file
|
||||||
if (_filename.endsWith(".xml")) {
|
if (_filename.endsWith(".xml") || _filename.endsWith(".XML")) {
|
||||||
// XML drawables support
|
// XML drawables support
|
||||||
StateDrawable d = new StateDrawable();
|
StateDrawable d = new StateDrawable();
|
||||||
if (!d.load(_filename)) {
|
if (!d.load(_filename)) {
|
||||||
|
Log.e("failed to load .xml drawable from ", _filename);
|
||||||
destroy(d);
|
destroy(d);
|
||||||
_error = true;
|
_error = true;
|
||||||
} else {
|
} else {
|
||||||
|
Log.d("loaded .xml drawable from ", _filename);
|
||||||
_drawable = d;
|
_drawable = d;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// PNG/JPEG drawables support
|
// PNG/JPEG drawables support
|
||||||
DrawBufRef image = imageCache.get(_filename, transform);
|
DrawBufRef image = imageCache.get(_filename, transform);
|
||||||
if (!image.isNull) {
|
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);
|
_transformed[transform] = new ImageDrawable(image, _tiled, ninePatch);
|
||||||
return _transformed[transform];
|
return _transformed[transform];
|
||||||
} else
|
} else {
|
||||||
|
Log.e("failed to load image from ", _filename);
|
||||||
_error = true;
|
_error = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return _drawable;
|
return _drawable;
|
||||||
|
@ -743,6 +753,8 @@ class DrawableCache {
|
||||||
if (fn !is null) {
|
if (fn !is null) {
|
||||||
_idToFileMap[id] = fn;
|
_idToFileMap[id] = fn;
|
||||||
return fn;
|
return fn;
|
||||||
|
} else {
|
||||||
|
Log.w("resource ", id, " is not found");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -286,9 +286,11 @@ class Win32Font : Font {
|
||||||
lf.lfFaceName[def.face.length] = 0;
|
lf.lfFaceName[def.face.length] = 0;
|
||||||
lf.lfHeight = -size;
|
lf.lfHeight = -size;
|
||||||
lf.lfItalic = italic;
|
lf.lfItalic = italic;
|
||||||
lf.lfOutPrecision = OUT_TT_ONLY_PRECIS;
|
lf.lfOutPrecision = OUT_OUTLINE_PRECIS; //OUT_TT_ONLY_PRECIS;
|
||||||
lf.lfClipPrecision = CLIP_DEFAULT_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;
|
lf.lfPitchAndFamily = def.pitchAndFamily;
|
||||||
_hfont = CreateFontIndirectA(&lf);
|
_hfont = CreateFontIndirectA(&lf);
|
||||||
_drawbuf = new Win32ColorDrawBuf(1, 1);
|
_drawbuf = new Win32ColorDrawBuf(1, 1);
|
||||||
|
|
|
@ -30,14 +30,33 @@ Authors: $(WEB coolreader.org, Vadim Lopatin)
|
||||||
module dlangui.widgets.controls;
|
module dlangui.widgets.controls;
|
||||||
|
|
||||||
import dlangui.widgets.widget;
|
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
|
/// static text widget
|
||||||
class TextWidget : Widget {
|
class TextWidget : Widget {
|
||||||
this(string ID = null) {
|
this(string ID = null, string textResourceId = null) {
|
||||||
super(ID);
|
super(ID);
|
||||||
styleId = "TEXT";
|
styleId = "TEXT";
|
||||||
|
_text = textResourceId;
|
||||||
|
}
|
||||||
|
this(string ID, dstring rawText) {
|
||||||
|
super(ID);
|
||||||
|
styleId = "TEXT";
|
||||||
|
_text = rawText;
|
||||||
}
|
}
|
||||||
protected UIString _text;
|
protected UIString _text;
|
||||||
/// get widget text
|
/// get widget text
|
||||||
|
@ -49,7 +68,7 @@ class TextWidget : Widget {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
/// set text to show
|
/// set text to show
|
||||||
@property Widget text(ref UIString s) {
|
override @property Widget text(ref UIString s) {
|
||||||
_text = s;
|
_text = s;
|
||||||
requestLayout();
|
requestLayout();
|
||||||
return this;
|
return this;
|
||||||
|
@ -152,7 +171,11 @@ class ImageWidget : Widget {
|
||||||
sz.x = img.width;
|
sz.x = img.width;
|
||||||
sz.y = img.height;
|
sz.y = img.height;
|
||||||
applyAlign(rc, sz);
|
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
|
/// button with image only
|
||||||
class ImageButton : ImageWidget {
|
class ImageButton : ImageWidget {
|
||||||
this(string ID = null, string drawableId = null) {
|
this(string ID = null, string drawableId = null) {
|
||||||
super(ID);
|
super(ID, drawableId);
|
||||||
styleId = "BUTTON";
|
styleId = "BUTTON";
|
||||||
_drawableId = drawableId;
|
_drawableId = drawableId;
|
||||||
|
clickable = true;
|
||||||
focusable = true;
|
focusable = true;
|
||||||
trackHover = 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 {
|
class Button : Widget {
|
||||||
protected UIString _text;
|
protected UIString _text;
|
||||||
override @property dstring text() { return _text; }
|
override @property dstring text() { return _text; }
|
||||||
override @property Widget text(dstring s) { _text = s; requestLayout(); return this; }
|
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; }
|
@property Widget textResource(string s) { _text = s; requestLayout(); return this; }
|
||||||
this(string ID = null) {
|
this(string ID = null) {
|
||||||
super(ID);
|
super(ID);
|
||||||
|
|
|
@ -523,10 +523,11 @@ class Theme : Style {
|
||||||
_backgroundColor = 0xFFFFFFFF; // transparent
|
_backgroundColor = 0xFFFFFFFF; // transparent
|
||||||
_textColor = 0x000000; // black
|
_textColor = 0x000000; // black
|
||||||
_align = Align.TopLeft;
|
_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;
|
_fontStyle = FONT_STYLE_NORMAL;
|
||||||
_fontWeight = 400;
|
_fontWeight = 400;
|
||||||
_fontFace = "Arial"; // TODO: from settings
|
//_fontFace = "Arial"; // TODO: from settings
|
||||||
|
_fontFace = "Verdana"; // TODO: from settings
|
||||||
_fontFamily = FontFamily.SansSerif;
|
_fontFamily = FontFamily.SansSerif;
|
||||||
_minHeight = 0;
|
_minHeight = 0;
|
||||||
_minWidth = 0;
|
_minWidth = 0;
|
||||||
|
@ -635,10 +636,18 @@ Theme createDefaultTheme() {
|
||||||
Log.d("Creating default theme");
|
Log.d("Creating default theme");
|
||||||
Theme res = new Theme("default");
|
Theme res = new Theme("default");
|
||||||
//res.fontSize(14);
|
//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 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);
|
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_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 | 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.Enabled, 0).backgroundImageId("btn_default_small_normal_disable");
|
||||||
//button.createState(State.Pressed, State.Pressed).backgroundImageId("btn_default_small_pressed");
|
//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.Selected, State.Selected).backgroundColor(0x00F8F9Fa);
|
||||||
menuItem.createState(State.Hovered, State.Hovered).backgroundColor(0xC0FFFF00);
|
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 poopupMenu = res.createSubstyle("POPUP_MENU").backgroundImageId("popup_menu_background_normal");
|
||||||
|
|
||||||
Style listItem = res.createSubstyle("LIST_ITEM");
|
Style listItem = res.createSubstyle("LIST_ITEM");
|
||||||
|
|
|
@ -259,6 +259,8 @@ class Widget {
|
||||||
@property dstring text() { return ""; }
|
@property dstring text() { return ""; }
|
||||||
/// sets widget content text (override to support this)
|
/// sets widget content text (override to support this)
|
||||||
@property Widget text(dstring s) { return 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
|
// Layout and drawing related methods
|
||||||
|
@ -337,12 +339,39 @@ class Widget {
|
||||||
return _pos.isPointInside(x, y);
|
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;
|
protected bool _focusable;
|
||||||
@property bool focusable() { return _focusable; }
|
@property bool focusable() { return _focusable; }
|
||||||
@property Widget focusable(bool flg) { _focusable = flg; return this; }
|
@property Widget focusable(bool flg) { _focusable = flg; return this; }
|
||||||
|
|
||||||
@property bool focused() {
|
@property bool focused() {
|
||||||
return (window !is null && window.focusedWidget is this && (state & State.Focused));
|
return (window !is null && window.focusedWidget is this && (state & State.Focused));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/// returns true if this widget and all its parents are visible
|
/// returns true if this widget and all its parents are visible
|
||||||
@property bool visible() {
|
@property bool visible() {
|
||||||
if (visibility != Visibility.Visible)
|
if (visibility != Visibility.Visible)
|
||||||
|
@ -390,9 +419,15 @@ class Widget {
|
||||||
// =======================================================
|
// =======================================================
|
||||||
// Events
|
// 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.
|
/// process key event, return true if event is processed.
|
||||||
bool onKeyEvent(KeyEvent event) {
|
bool onKeyEvent(KeyEvent event) {
|
||||||
if (onClickListener.assigned) {
|
if (clickable) {
|
||||||
// support onClick event initiated by Space or Return keys
|
// support onClick event initiated by Space or Return keys
|
||||||
if (event.action == KeyAction.KeyDown) {
|
if (event.action == KeyAction.KeyDown) {
|
||||||
if (event.keyCode == KeyCode.SPACE || event.keyCode == KeyCode.RETURN) {
|
if (event.keyCode == KeyCode.SPACE || event.keyCode == KeyCode.RETURN) {
|
||||||
|
@ -403,7 +438,7 @@ class Widget {
|
||||||
if (event.action == KeyAction.KeyUp) {
|
if (event.action == KeyAction.KeyUp) {
|
||||||
if (event.keyCode == KeyCode.SPACE || event.keyCode == KeyCode.RETURN) {
|
if (event.keyCode == KeyCode.SPACE || event.keyCode == KeyCode.RETURN) {
|
||||||
resetState(State.Pressed);
|
resetState(State.Pressed);
|
||||||
onClickListener(this);
|
handleClick();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -415,7 +450,7 @@ class Widget {
|
||||||
bool onMouseEvent(MouseEvent event) {
|
bool onMouseEvent(MouseEvent event) {
|
||||||
//Log.d("onMouseEvent ", id, " ", event.action, " (", event.x, ",", event.y, ")");
|
//Log.d("onMouseEvent ", id, " ", event.action, " (", event.x, ",", event.y, ")");
|
||||||
// support onClick
|
// support onClick
|
||||||
if (onClickListener.assigned) {
|
if (clickable) {
|
||||||
if (event.action == MouseAction.ButtonDown && event.button == MouseButton.Left) {
|
if (event.action == MouseAction.ButtonDown && event.button == MouseButton.Left) {
|
||||||
setState(State.Pressed);
|
setState(State.Pressed);
|
||||||
if (focusable)
|
if (focusable)
|
||||||
|
@ -424,7 +459,7 @@ class Widget {
|
||||||
}
|
}
|
||||||
if (event.action == MouseAction.ButtonUp && event.button == MouseButton.Left) {
|
if (event.action == MouseAction.ButtonUp && event.button == MouseButton.Left) {
|
||||||
resetState(State.Pressed);
|
resetState(State.Pressed);
|
||||||
onClickListener(this);
|
handleClick();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (event.action == MouseAction.FocusOut || event.action == MouseAction.Cancel) {
|
if (event.action == MouseAction.FocusOut || event.action == MouseAction.Cancel) {
|
||||||
|
|