mirror of
https://thingvellir.net/git/overte
synced 2025-03-27 23:52:03 +01:00
1048 lines
32 KiB
JavaScript
1048 lines
32 KiB
JavaScript
//
|
|
// uiwidgets.js
|
|
// examples/libraries
|
|
//
|
|
// Created by Seiji Emery, 8/10/15
|
|
// Copyright 2015 High Fidelity, Inc
|
|
//
|
|
// Distributed under the Apache License, Version 2.0.
|
|
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
|
//
|
|
|
|
|
|
(function(){
|
|
|
|
// Setup externals
|
|
(function() {
|
|
|
|
// We need a Vec2 impl, with add() and a clone function. If this is not part of hifi, we'll just add it:
|
|
if (this.Vec2 == undefined) {
|
|
var Vec2 = this.Vec2 = function (x, y) {
|
|
this.x = x || 0.0;
|
|
this.y = y || 0.0;
|
|
}
|
|
Vec2.sum = function (a, b) {
|
|
return new Vec2(a.x + b.x, a.y + b.y);
|
|
}
|
|
Vec2.clone = function (v) {
|
|
return new Vec2(v.x, v.y);
|
|
}
|
|
} else if (this.Vec2.clone == undefined) {
|
|
this.Vec2.clone = function (v) {
|
|
return { 'x': v.x || 0.0, 'y': v.y || 0.0 };
|
|
}
|
|
} else {}
|
|
})();
|
|
|
|
var Rect = function (xmin, ymin, xmax, ymax) {
|
|
this.x0 = xmin;
|
|
this.y0 = ymin;
|
|
this.x1 = xmax;
|
|
this.y1 = ymax;
|
|
}
|
|
Rect.prototype.grow = function (pt) {
|
|
this.x0 = Math.min(this.x0, pt.x);
|
|
this.y0 = Math.min(this.y0, pt.y);
|
|
this.x1 = Math.max(this.x1, pt.x);
|
|
this.y1 = Math.max(this.y1, pt.y);
|
|
}
|
|
Rect.prototype.getWidth = function () {
|
|
return this.x1 - this.x0;
|
|
}
|
|
Rect.prototype.getHeight = function () {
|
|
return this.y1 - this.y0;
|
|
}
|
|
Rect.prototype.getTopLeft = function () {
|
|
return { 'x': this.x0, 'y': this.y0 };
|
|
}
|
|
Rect.prototype.getBtmRight = function () {
|
|
return { 'x': this.x1, 'y': this.y1 };
|
|
}
|
|
Rect.prototype.getCenter = function () {
|
|
return {
|
|
'x': 0.5 * (this.x1 + this.x0),
|
|
'y': 0.5 * (this.y1 + this.y0)
|
|
};
|
|
}
|
|
|
|
var __trace = new Array();
|
|
var __traceDepth = 0;
|
|
|
|
var assert = function (cond, expr) {
|
|
if (!cond) {
|
|
var callstack = "";
|
|
var maxRecursion = 10;
|
|
caller = arguments.callee.caller;
|
|
while (maxRecursion > 0 && caller) {
|
|
--maxRecursion;
|
|
callstack += ">> " + caller.toString();
|
|
caller = caller.caller;
|
|
}
|
|
throw new Error("assertion failed: " + expr + " (" + cond + ")" + "\n" +
|
|
"Called from: " + callstack + " " +
|
|
"Traceback: \n\t" + __trace.join("\n\t"));
|
|
}
|
|
}
|
|
var traceEnter = function(fcn) {
|
|
var l = __trace.length;
|
|
// print("TRACE ENTER: " + (l+1));
|
|
s = "";
|
|
for (var i = 0; i < __traceDepth+1; ++i)
|
|
s += "-";
|
|
++__traceDepth;
|
|
__trace.push(s + fcn);
|
|
__trace.push(__trace.pop() + ":" + this);
|
|
return {
|
|
'exit': function () {
|
|
--__traceDepth;
|
|
// while (__trace.length != l)
|
|
// __trace.pop();
|
|
}
|
|
};
|
|
}
|
|
|
|
/// UI namespace
|
|
var UI = this.UI = {};
|
|
|
|
var rgb = UI.rgb = function (r, g, b) {
|
|
if (typeof(r) == 'string') {
|
|
rs = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i.exec(r);
|
|
if (rs) {
|
|
r = parseInt(rs[0], 16);
|
|
g = parseInt(rs[1], 16);
|
|
b = parseInt(rs[2], 16);
|
|
}
|
|
}
|
|
if (typeof(r) != 'number' || typeof(g) != 'number' || typeof(b) != 'number') {
|
|
ui.err("Invalid args to UI.rgb (" + r + ", " + g + ", " + b + ")");
|
|
return null;
|
|
}
|
|
return { 'red': r, 'green': g, 'blue': b };
|
|
}
|
|
var rgba = UI.rgba = function (r, g, b, a) {
|
|
if (typeof(r) == 'string')
|
|
return rgb(r);
|
|
return { 'red': r || 0, 'green': g || 0, 'blue': b || 0, 'a': a };
|
|
}
|
|
|
|
/// Protected UI state
|
|
var ui = {
|
|
defaultVisible: true,
|
|
widgetList: new Array(),
|
|
attachmentList: new Array()
|
|
};
|
|
|
|
ui.complain = function (msg) {
|
|
print("WARNING (uiwidgets.js): " + msg);
|
|
}
|
|
ui.errorHandler = function (err) {
|
|
print(err);
|
|
}
|
|
ui.assert = function (condition, message) {
|
|
if (!condition) {
|
|
message = "FAILED ASSERT (uiwidgets.js): " + message || "(" + condition + ")";
|
|
ui.errorHandler(message);
|
|
if (typeof(Error) !== 'undefined')
|
|
throw new Error(message);
|
|
throw message;
|
|
}
|
|
}
|
|
|
|
UI.setDefaultVisibility = function (visible) {
|
|
ui.defaultVisible = visible;
|
|
}
|
|
|
|
/// Wrapper around the overlays impl
|
|
function makeOverlay(type, properties) {
|
|
var overlay = Overlays.addOverlay(type, properties);
|
|
return {
|
|
'update': function (properties) {
|
|
var _TRACE = traceEnter.call(this, "Overlay.update");
|
|
Overlays.editOverlay(overlay, properties);
|
|
_TRACE.exit();
|
|
},
|
|
'destroy': function () {
|
|
var _TRACE = traceEnter.call(this, "Overlay.destroy");
|
|
Overlays.deleteOverlay(overlay);
|
|
_TRACE.exit();
|
|
},
|
|
'getId': function () {
|
|
return overlay;
|
|
}
|
|
}
|
|
}
|
|
|
|
var COLOR_WHITE = rgb(255, 255, 255);
|
|
var COLOR_GRAY = rgb(125, 125, 125);
|
|
|
|
/// Base widget class.
|
|
var Widget = function () {};
|
|
|
|
// Shared methods:
|
|
var __widgetId = 0;
|
|
Widget.prototype.constructor = function () {
|
|
this.position = { 'x': 0.0, 'y': 0.0 };
|
|
this.dimensions = null;
|
|
|
|
this.visible = ui.defaultVisible;
|
|
this.parentVisible = null;
|
|
|
|
this.actions = {};
|
|
this._dirty = true;
|
|
this.parent = null;
|
|
|
|
this.id = __widgetId++;
|
|
ui.widgetList.push(this);
|
|
}
|
|
Widget.prototype.setPosition = function (x, y) {
|
|
if (arguments.length == 1 && typeof(arguments[0]) == 'object') {
|
|
x = arguments[0].x;
|
|
y = arguments[0].y;
|
|
}
|
|
if (typeof(x) != 'number' || typeof(y) != 'number') {
|
|
ui.complain("invalid arguments to " + this + ".setPosition: '" + arguments + "' (expected (x, y) or (vec2))");
|
|
} else {
|
|
this.position.x = x;
|
|
this.position.y = y;
|
|
}
|
|
}
|
|
Widget.prototype.setVisible = function (visible) {
|
|
this.visible = visible;
|
|
this.parentVisible = null; // set dirty
|
|
}
|
|
Widget.prototype.isVisible = function () {
|
|
if (this.parentVisible === null)
|
|
this.parentVisible = this.parent ? this.parent.isVisible() : true;
|
|
return this.visible && this.parentVisible;
|
|
}
|
|
// Store lists of actions (multiple callbacks per key)
|
|
Widget.prototype.addAction = function (action, callback) {
|
|
if (!this.actions[action])
|
|
this.actions[action] = [ callback ];
|
|
else
|
|
this.actions[action].push(callback);
|
|
}
|
|
Widget.prototype.clearLayout = function () {
|
|
this.dimensions = null;
|
|
this.parentVisible = null;
|
|
}
|
|
// Overridden methods:
|
|
Widget.prototype.toString = function () {
|
|
return "[Widget " + this.id + " ]";
|
|
}
|
|
Widget.prototype.getOverlay = function () {
|
|
return null;
|
|
}
|
|
Widget.prototype.getWidth = function () {
|
|
return 0;
|
|
}
|
|
Widget.prototype.getHeight = function () {
|
|
return 0;
|
|
}
|
|
Widget.prototype.hasOverlay = function () {
|
|
return false;
|
|
}
|
|
Widget.prototype.applyLayout = function () {};
|
|
Widget.prototype.updateOverlays = function () {};
|
|
|
|
/// Implements a simple auto-layouted container of methods.
|
|
/// @param properties
|
|
/// dir: [string]
|
|
/// layout direction.
|
|
/// Can be one of [ '+x', '+y', '-x', '-y' ] for 2d directions.
|
|
/// border: { x: _, y: _ }
|
|
/// Adds spacing to the widget on all sides (aka. margin). Defaults to 0.
|
|
/// padding: { x: _, y: _ }
|
|
/// Padding in between each widget. Only one axis is used (the layout direction).
|
|
/// visible: true | false
|
|
/// Acts as both a widget (logical) property and is used for overlays.
|
|
/// Hiding this will hide all child widgets (non-destructively).
|
|
/// Do not access this directly -- use setVisible(value) and isVisible() instead.
|
|
/// background: [object]
|
|
/// Properties to use for the background overlay (if defined).
|
|
///
|
|
var WidgetStack = UI.WidgetStack = function (properties) {
|
|
var _TRACE = traceEnter.call(this, "WidgetStack.constructor()");
|
|
Widget.prototype.constructor.call(this);
|
|
assert(ui.widgetList[ui.widgetList.length-1] === this, "ui.widgetList.back() == this");
|
|
|
|
properties = properties || {};
|
|
properties['dir'] = properties['dir'] || '+y';
|
|
|
|
var dir = undefined;
|
|
switch(properties['dir']) {
|
|
case '+y': dir = { 'x': 0.0, 'y': 1.0 }; break;
|
|
case '-y': dir = { 'x': 0.0, 'y': -1.0 }; break;
|
|
case '+x': dir = { 'x': 1.0, 'y': 0.0 }; break;
|
|
case '-x': dir = { 'x': -1.0, 'y': 0.0 }; break;
|
|
default: ui.complain("Unrecognized UI.WidgetStack property 'dir': \"" + dir + "\"");
|
|
}
|
|
dir = dir || { 'x': 1.0, 'y': 0.0 };
|
|
|
|
this.layoutDir = dir;
|
|
this.border = properties.border || { 'x': 0.0, 'y': 0.0 };
|
|
this.padding = properties.padding || { 'x': 0.0, 'y': 0.0 };
|
|
this.visible = properties.visible != undefined ? properties.visible : this.visible;
|
|
|
|
if (properties.background) {
|
|
var background = properties.background;
|
|
background.x = this.position ? this.position.x : 0;
|
|
background.y = this.position ? this.position.y : 0;
|
|
background.width = background.width || 100.0;
|
|
background.height = background.height || 100.0;
|
|
background.backgroundColor = background.backgroundColor || COLOR_GRAY;
|
|
background.backgroundAlpha = background.backgroundAlpha || 0.5;
|
|
background.textColor = background.textColor || COLOR_WHITE;
|
|
background.alpha = background.alpha || 1.0;
|
|
background.visible = this.visible;
|
|
this.backgroundOverlay = makeOverlay("text", background);
|
|
} else {
|
|
this.backgroundOverlay = null;
|
|
}
|
|
|
|
this.widgets = new Array();
|
|
|
|
_TRACE.exit();
|
|
}
|
|
WidgetStack.prototype = new Widget();
|
|
WidgetStack.prototype.constructor = WidgetStack;
|
|
|
|
WidgetStack.prototype.toString = function () {
|
|
return "[WidgetStack " + this.id + " ]";
|
|
}
|
|
|
|
WidgetStack.prototype.add = function (widget) {
|
|
this.widgets.push(widget);
|
|
widget.parent = this;
|
|
return widget;
|
|
}
|
|
WidgetStack.prototype.hasOverlay = function (overlayId) {
|
|
return this.backgroundOverlay && this.backgroundOverlay.getId() === overlayId;
|
|
}
|
|
WidgetStack.prototype.getOverlay = function () {
|
|
return this.backgroundOverlay;
|
|
}
|
|
WidgetStack.prototype.destroy = function () {
|
|
if (this.backgroundOverlay) {
|
|
this.backgroundOverlay.destroy();
|
|
this.backgroundOverlay = null;
|
|
}
|
|
}
|
|
WidgetStack.prototype.setColor = function (color) {
|
|
if (arguments.length != 1) {
|
|
color = rgba.apply(arguments);
|
|
}
|
|
this.backgroundOverlay.update({
|
|
'color': color,
|
|
'alpha': color.a
|
|
});
|
|
}
|
|
var sumOf = function (list, f) {
|
|
var sum = 0.0;
|
|
list.forEach(function (elem) {
|
|
sum += f(elem);
|
|
})
|
|
return sum;
|
|
}
|
|
WidgetStack.prototype.calculateDimensions = function () {
|
|
var totalWidth = 0.0, maxWidth = 0.0;
|
|
var totalHeight = 0.0, maxHeight = 0.0;
|
|
this.widgets.forEach(function (widget) {
|
|
totalWidth += widget.getWidth() + this.padding.x;
|
|
maxWidth = Math.max(maxWidth, widget.getWidth());
|
|
|
|
totalHeight += widget.getHeight() + this.padding.y;
|
|
maxHeight = Math.max(maxHeight, widget.getHeight());
|
|
}, this);
|
|
|
|
this.dimensions = {
|
|
x: this.border.x * 2 + Math.max(totalWidth * this.layoutDir.x - this.padding.x, maxWidth),
|
|
y: this.border.y * 2 + Math.max(totalHeight * this.layoutDir.y - this.padding.y, maxHeight)
|
|
};
|
|
}
|
|
WidgetStack.prototype.getWidth = function () {
|
|
if (!this.dimensions)
|
|
this.calculateDimensions();
|
|
return this.dimensions.x;
|
|
}
|
|
WidgetStack.prototype.getHeight = function () {
|
|
if (!this.dimensions)
|
|
this.calculateDimensions();
|
|
return this.dimensions.y;
|
|
}
|
|
WidgetStack.prototype.applyLayout = function () {
|
|
var x = this.position.x + this.border.x;
|
|
var y = this.position.y + this.border.y;
|
|
|
|
this.widgets.forEach(function (widget) {
|
|
widget.setPosition(x, y);
|
|
x += (widget.getWidth() + this.padding.x) * this.layoutDir.x;
|
|
y += (widget.getHeight() + this.padding.y) * this.layoutDir.y;
|
|
widget._parentVisible = this.isVisible();
|
|
}, this);
|
|
}
|
|
WidgetStack.prototype.updateOverlays = function () {
|
|
if (this.backgroundOverlay) {
|
|
this.backgroundOverlay.update({
|
|
width: this.getWidth(),
|
|
height: this.getHeight(),
|
|
x: this.position.x,
|
|
y: this.position.y,
|
|
visible: this.isVisible()
|
|
});
|
|
}
|
|
}
|
|
|
|
|
|
/// GUI Textured Rect
|
|
var Image = UI.Image = function (properties) {
|
|
Widget.prototype.constructor.call(this);
|
|
|
|
this.visible = properties.visible != undefined ? properties.visible : this.visible;
|
|
this.width = properties.width || 1.0;
|
|
this.height = properties.height || 1.0;
|
|
|
|
var imageProperties = {
|
|
'color': properties.color || COLOR_GRAY,
|
|
'alpha': properties.alpha || 1.0,
|
|
'imageURL': properties.imageURL,
|
|
'width': this.width,
|
|
'height': this.height,
|
|
'x': this.position ? this.position.x : 0.0,
|
|
'y': this.position ? this.position.y : 0.0,
|
|
'visible': this.visible
|
|
}
|
|
this.imageOverlay = makeOverlay("image", imageProperties);
|
|
}
|
|
Image.prototype = new Widget();
|
|
Image.prototype.constructor = Image;
|
|
Image.prototype.toString = function () {
|
|
return "[UI.Image " + this.id + " ]";
|
|
}
|
|
Image.prototype.getHeight = function () {
|
|
return this.height;
|
|
}
|
|
Image.prototype.getWidth = function () {
|
|
return this.width;
|
|
}
|
|
Image.prototype.hasOverlay = function (overlayId) {
|
|
return this.imageOverlay.getId() === overlayId;
|
|
}
|
|
Image.prototype.getOverlay = function () {
|
|
return this.imageOverlay;
|
|
}
|
|
Image.prototype.destroy = function () {
|
|
if (this.imageOverlay) {
|
|
this.imageOverlay.destroy();
|
|
this.imageOverlay = null;
|
|
}
|
|
}
|
|
Image.prototype.setColor = function (color) {
|
|
if (arguments.length != 1) {
|
|
color = rgba.apply(arguments);
|
|
}
|
|
this.imageOverlay.update({
|
|
'color': color,
|
|
'alpha': color.a
|
|
});
|
|
}
|
|
Image.prototype.getWidth = function () {
|
|
return this.width;
|
|
}
|
|
Image.prototype.getHeight = function () {
|
|
return this.height;
|
|
}
|
|
Image.prototype.updateOverlays = function () {
|
|
this.imageOverlay.update({
|
|
width: this.width,
|
|
height: this.height,
|
|
x: this.position.x,
|
|
y: this.position.y,
|
|
visible: this.isVisible()
|
|
});
|
|
}
|
|
|
|
|
|
/// GUI Rect. Internally implemented using a text overlay.
|
|
var Box = UI.Box = function (properties) {
|
|
Widget.prototype.constructor.call(this);
|
|
|
|
properties = properties || {};
|
|
properties.width = properties.width || 10;
|
|
properties.height = properties.height || 10;
|
|
properties.visible = properties.visible !== undefined ? properties.visible : this.visible;
|
|
properties.x = this.position.x;
|
|
properties.y = this.position.y;
|
|
|
|
this.width = properties.width;
|
|
this.height = properties.height;
|
|
this.visible = properties.visible;
|
|
|
|
this.overlay = makeOverlay("text", properties);
|
|
};
|
|
Box.prototype = new Widget();
|
|
Box.prototype.constructor = Box;
|
|
Box.prototype.toString = function () {
|
|
return "[UI.Box " + this.id + " ]";
|
|
}
|
|
Box.prototype.getWidth = function () {
|
|
return this.width;
|
|
}
|
|
Box.prototype.getHeight = function () {
|
|
return this.height;
|
|
}
|
|
Box.prototype.destroy = function () {
|
|
if (this.overlay) {
|
|
this.overlay.destroy();
|
|
this.overlay = null;
|
|
}
|
|
}
|
|
Box.prototype.hasOverlay = function (overlayId) {
|
|
return /*this.overlay &&*/ this.overlay.getId() === overlayId;
|
|
}
|
|
Box.prototype.getOverlay = function () {
|
|
return this.overlay;
|
|
}
|
|
Box.prototype.updateOverlays = function () {
|
|
this.overlay.update({
|
|
x: this.position.x,
|
|
y: this.position.y,
|
|
width: this.width,
|
|
height: this.height,
|
|
visible: this.isVisible()
|
|
});
|
|
}
|
|
|
|
var Label = UI.Label = function (properties) {
|
|
properties = properties || {};
|
|
if (properties.text === undefined || properties.text === null)
|
|
properties.text = "< bad UI.Label call (text) >";
|
|
|
|
if (!properties.width) {
|
|
ui.complain("UI.Label constructor expected width property, not " + properties.width);
|
|
this.text = "< bad UI.Label call (width) >";
|
|
properties.width = 220;
|
|
}
|
|
if (!properties.height) {
|
|
ui.complain("UI.Label constructor expected height property, not " + properties.height);
|
|
this.text = "< bad UI.Label call (height) >";
|
|
properties.height = 20;
|
|
}
|
|
properties.color = properties.color || COLOR_WHITE;
|
|
properties.alpha = properties.alpha || properties.color.a || 1.0;
|
|
properties.backgroundAlpha = properties.backgroundAlpha || 0.0;
|
|
|
|
Box.prototype.constructor.call(this, properties);
|
|
};
|
|
Label.prototype = new Box();
|
|
Label.prototype.constructor = Label;
|
|
Label.prototype.toString = function () {
|
|
return "[UI.Label " + this.id + " ]";
|
|
}
|
|
Label.prototype.setText = function (text) {
|
|
this.text = text;
|
|
this.overlay.update({
|
|
text: text
|
|
});
|
|
}
|
|
|
|
/// Slider widget.
|
|
/// @param properties:
|
|
/// onValueChanged
|
|
var Slider = UI.Slider = function (properties) {
|
|
Box.prototype.constructor.call(this, properties);
|
|
// print("CONSTRUCTING SLIDER")
|
|
this.value = properties.value || 0.0;
|
|
this.maxValue = properties.maxValue || 1.0;
|
|
this.minValue = properties.minValue || 0.0;
|
|
this.padding = properties.padding || {
|
|
x: 4, y: 4
|
|
}
|
|
this.onValueChanged = properties.onValueChanged || function () {};
|
|
|
|
this.slider = new Box(properties.slider);
|
|
this.slider.parent = this;
|
|
|
|
var clickOffset = { x: 0.0, y: 0.0 }; // offset relative to slider knob
|
|
var widget = this;
|
|
var updateDrag = function (event) {
|
|
var rx = Math.max(event.x * 1.0 - widget.position.x - clickOffset.x, 0.0);
|
|
var width = Math.max(widget.width - widget.slider.width - widget.padding.x * 2.0, 0.0);
|
|
var v = Math.min(rx, width) / (width || 1);
|
|
|
|
// print("dragging slider: rx = " + rx + ", width = " + width + ", v = " + v);
|
|
|
|
widget.value = widget.minValue + (widget.maxValue - widget.minValue) * v;
|
|
widget.onValueChanged(widget.value);
|
|
UI.updateLayout();
|
|
}
|
|
var startDrag = function (event) {
|
|
// calculate position of slider knob
|
|
var x0 = widget.position.x + widget.padding.x;
|
|
var width = (widget.width - widget.slider.width - widget.padding.x * 2.0);
|
|
var normalizedValue = (widget.value - widget.minValue) / (widget.maxValue - widget.minValue)
|
|
|
|
var sliderX = x0 + normalizedValue * width;
|
|
var sliderWidth = widget.slider.width;
|
|
|
|
if (event.x >= sliderX && event.x <= sliderX + sliderWidth) {
|
|
// print("Start drag -- on slider knob");
|
|
clickOffset.x = event.x - sliderX;
|
|
} else if (event.x >= x0 && event.x <= x0 + width) {
|
|
// print("Start drag -- on slider bar");
|
|
clickOffset.x = sliderWidth * 0.5;
|
|
} else {
|
|
clickOffset.x = 0.0;
|
|
// print("Start drag -- out of bounds!");
|
|
// print("event.x = " + event.x);
|
|
// print("x0 = " + x0 + ", x1 = " + (x0 + width) + " (width = " + width + ")");
|
|
// print("s0 = " + sliderX + ", s1 = " + (sliderX + sliderWidth) + "(slider width = " + sliderWidth + ")");
|
|
// print("widget = " + widget);
|
|
// print("widget.slider = " + widget.slider);
|
|
// print("widget.width = " + widget.width + ", widget.slider.width = " + widget.slider.width);
|
|
}
|
|
updateDrag(event);
|
|
}
|
|
|
|
this.addAction('onMouseDown', startDrag);
|
|
this.addAction('onDragBegin', updateDrag);
|
|
this.addAction('onDragUpdate', updateDrag);
|
|
this.slider.actions = this.actions;
|
|
};
|
|
Slider.prototype = new Box();
|
|
Slider.prototype.constructor = Slider;
|
|
|
|
Slider.prototype.toString = function () {
|
|
return "[UI.Slider " + this.id + " ]";
|
|
}
|
|
Slider.prototype.applyLayout = function () {
|
|
if (!this.slider) {
|
|
// ui.complain("Slider.applyLayout on " + this + " failed");
|
|
return;
|
|
}
|
|
var val = (this.value - this.minValue) / (this.maxValue - this.minValue);
|
|
this.slider.position.x = this.position.x + this.padding.x + (this.width - this.slider.width - this.padding.x * 2.0) * val;
|
|
this.slider.position.y = this.position.y + /*this.padding.y +*/ (this.height - this.slider.height) * 0.5;
|
|
}
|
|
Slider.prototype.getValue = function () {
|
|
return this.value;
|
|
}
|
|
Slider.prototype.setValue = function (value) {
|
|
this.value = value;
|
|
this.onValueChanged(value);
|
|
UI.updateLayout();
|
|
}
|
|
|
|
|
|
var Checkbox = UI.Checkbox = function (properties) {
|
|
Box.prototype.constructor.call(this, properties);
|
|
|
|
this.checked = properties.checked !== undefined ? properties.checked : true;
|
|
this.padding = properties.padding || { x: 4, y: 4 };
|
|
|
|
properties.checkMark = properties.checkMark || {};
|
|
|
|
// Keep square
|
|
var r = Math.min(this.width - this.padding.x * 2.0, this.height - this.padding.y * 2.0);
|
|
properties.checkMark.width = properties.checkMark.height = r;
|
|
properties.checkMark.visible = false;
|
|
properties.checkMark.backgroundColor = properties.checkMark.backgroundColor ||
|
|
properties.checkMark.color || rgb(77, 185, 77);
|
|
properties.checkMark.backgroundAlpha = properties.checkMark.backgroundAlpha ||
|
|
properties.checkMark.alpha || 1.0;
|
|
this.checkMark = new Box(properties.checkMark);
|
|
this.checkMark.setVisible(this.checked);
|
|
this.checkMark.setPosition(
|
|
this.position.x + (this.width - this.checkMark.width) * 0.5,
|
|
this.position.y + (this.height - this.checkMark.height) * 0.5
|
|
);
|
|
this.checkMark.parent = this;
|
|
|
|
this.onValueChanged = properties.onValueChanged || function () {};
|
|
|
|
var widget = this;
|
|
this.addAction('onClick', function (event) {
|
|
widget.setChecked(!widget.isChecked());
|
|
});
|
|
this.checkMark.addAction('onClick', function (event) {
|
|
widget.setChecked(!widget.isChecked());
|
|
});
|
|
};
|
|
Checkbox.prototype = new Box();
|
|
Checkbox.prototype.constructor = Checkbox;
|
|
Checkbox.prototype.toString = function () {
|
|
return "[UI.Checkbox " + this.id + "]";
|
|
}
|
|
Checkbox.prototype.isChecked = function () {
|
|
return this.checked;
|
|
}
|
|
Checkbox.prototype.setChecked = function (value) {
|
|
this.checked = value;
|
|
this.checkMark.setVisible(this.checked);
|
|
|
|
this.onValueChanged(value);
|
|
UI.updateLayout();
|
|
}
|
|
Checkbox.prototype.applyLayout = function () {
|
|
this.checkMark && this.checkMark.setPosition(
|
|
this.position.x + (this.width - this.checkMark.width) * 0.5,
|
|
this.position.y + (this.height - this.checkMark.height) * 0.5
|
|
);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UI.addAttachment = function (target, rel, update) {
|
|
attachment = {
|
|
target: target,
|
|
rel: rel,
|
|
applyLayout: update
|
|
};
|
|
ui.attachmentList.push(attachment);
|
|
return attachment;
|
|
}
|
|
|
|
|
|
UI.updateLayout = function () {
|
|
if (ui.visible)
|
|
ui.updateDebugUI();
|
|
|
|
// Recalc dimensions
|
|
ui.widgetList.forEach(function (widget) {
|
|
widget.clearLayout();
|
|
});
|
|
|
|
function insertAndPush (list, index, elem) {
|
|
if (list[index])
|
|
list[index].push(elem);
|
|
else
|
|
list[index] = [ elem ];
|
|
}
|
|
|
|
// Generate attachment lookup
|
|
var attachmentDeps = {};
|
|
ui.attachmentList.forEach(function(attachment) {
|
|
insertAndPush(attachmentDeps, attachment.target.id, {
|
|
dep: attachment.rel,
|
|
eval: attachment.applyLayout
|
|
});
|
|
});
|
|
updated = {};
|
|
|
|
// Walk the widget list and relayout everything
|
|
function recalcLayout (widget) {
|
|
// Short circuit if we've already updated
|
|
if (updated[widget.id])
|
|
return;
|
|
|
|
// Walk up the tree + update top level first
|
|
if (widget.parent)
|
|
recalcLayout(widget.parent);
|
|
|
|
// Resolve and apply attachment dependencies
|
|
if (attachmentDeps[widget.id]) {
|
|
attachmentDeps[widget.id].forEach(function (attachment) {
|
|
recalcLayout(attachment.dep);
|
|
attachment.eval(widget, attachment.dep);
|
|
});
|
|
}
|
|
|
|
widget.applyLayout();
|
|
updated[widget.id] = true;
|
|
}
|
|
ui.widgetList.forEach(recalcLayout);
|
|
|
|
ui.widgetList.forEach(function (widget) {
|
|
widget.updateOverlays();
|
|
});
|
|
}
|
|
|
|
UI.setDefaultVisibility = function(visibility) {
|
|
ui.defaultVisible = visibility;
|
|
};
|
|
|
|
function dispatchEvent(actions, widget, event) {
|
|
var _TRACE = traceEnter.call(this, "UI.dispatchEvent()");
|
|
actions.forEach(function(action) {
|
|
action.call(widget, event);
|
|
});
|
|
_TRACE.exit();
|
|
}
|
|
|
|
// Focus state
|
|
ui.focusedWidget = null;
|
|
ui.clickedWidget = null;
|
|
ui.draggedWidget = null;
|
|
|
|
// Debugging ui
|
|
var statusPos = new Vec2(15, 20);
|
|
var statusDim = new Vec2(500, 20);
|
|
function makeStatusWidget(defaultText, alpha) {
|
|
var label = new Box({
|
|
text: defaultText,
|
|
width: statusDim.x,
|
|
height: statusDim.y,
|
|
color: COLOR_WHITE,
|
|
alpha: alpha || 0.98,
|
|
backgroundAlpha: 0.0,
|
|
visible: ui.debug.visible
|
|
});
|
|
label.updateText = function (text) {
|
|
label.getOverlay().update({
|
|
text: text
|
|
});
|
|
}
|
|
label.setPosition(statusPos.x, statusPos.y);
|
|
statusPos.y += statusDim.y;
|
|
return label;
|
|
}
|
|
|
|
ui.debug = {};
|
|
ui.debug.visible = false;
|
|
ui.focusStatus = makeStatusWidget("<UI focus>");
|
|
ui.eventStatus = makeStatusWidget("<UI events>", 0.85);
|
|
|
|
UI.debug = {
|
|
eventList: {
|
|
position: { x: 20, y: 20 }, width: 1, height: 1
|
|
},
|
|
widgetList: {
|
|
position: { x: 500, y: 20 }, width: 1, height: 1
|
|
},
|
|
setVisible: function (visible) {
|
|
if (ui.debug.visible != visible) {
|
|
ui.focusStatus.setVisible(visible);
|
|
ui.eventStatus.setVisible(visible);
|
|
if (visible) {
|
|
ui.debug.showWidgetList(UI.debug.widgetList.position.x, UI.debug.widgetList.position.y);
|
|
} else {
|
|
ui.debug.hideWidgetList();
|
|
}
|
|
UI.updateLayout();
|
|
}
|
|
ui.debug.visible = visible;
|
|
},
|
|
isVisible: function () {
|
|
return ui.debug.visible;
|
|
}
|
|
}
|
|
|
|
// Add debug list of all widgets + attachments
|
|
var widgetListHeader = makeStatusWidget("Widgets: ", 0.98);
|
|
var widgetList = makeStatusWidget("<widget list>", 0.85);
|
|
var attachmentListHeader = makeStatusWidget("Attachments: ", 0.98);
|
|
var attachmentList = makeStatusWidget("<attachment list>", 0.85);
|
|
var lineHeight = 20;
|
|
|
|
ui.debug.relayout = function () {
|
|
var x = UI.debug.widgetList.position.x, y = UI.debug.widgetList.position.y;
|
|
|
|
widgetListHeader.setPosition(x, y); y += lineHeight;
|
|
widgetList.updateText(ui.widgetList.map(function (widget) {
|
|
return "" + widget + " actions: " + (Object.keys(widget.actions).join(", ") || "none");
|
|
}).join("\n") || "no widgets");
|
|
widgetList.setPosition(x, y);
|
|
y += lineHeight * (ui.widgetList.length || 1);
|
|
|
|
attachmentListHeader.setPosition(x, y); y += lineHeight;
|
|
attachmentList.updateText(ui.attachmentList.map(function (attachment) {
|
|
return "[attachment target: " + attachment.target + ", to: " + attachment.rel + "]";
|
|
}).join('\n') || "no attachments");
|
|
attachmentList.setPosition(x, y);
|
|
y += lineHeight * (ui.widgetList.length || 1);
|
|
// UI.updateLayout();
|
|
}
|
|
|
|
var defaultX = 500;
|
|
var defaultY = 20;
|
|
ui.debug.showWidgetList = function (x, y) {
|
|
widgetListHeader.setVisible(true);
|
|
widgetList.setVisible(true);
|
|
attachmentListHeader.setVisible(true);
|
|
attachmentList.setVisible(true);
|
|
ui.debug.relayout(x || defaultX, y || defaultY);
|
|
}
|
|
ui.debug.hideWidgetList = function () {
|
|
widgetListHeader.setVisible(false);
|
|
widgetList.setVisible(false);
|
|
attachmentListHeader.setVisible(false);
|
|
attachmentList.setVisible(false);
|
|
}
|
|
|
|
ui.eventStatus.lastPos = {
|
|
x: ui.eventStatus.position.x,
|
|
y: ui.eventStatus.position.y
|
|
};
|
|
ui.updateDebugUI = function () {
|
|
ui.debug.relayout();
|
|
|
|
var dx = ui.eventStatus.position.x - ui.eventStatus.lastPos.x;
|
|
var dy = ui.eventStatus.position.y - ui.eventStatus.lastPos.y;
|
|
|
|
ui.focusStatus.position.x += dx;
|
|
ui.focusStatus.position.y += dy;
|
|
ui.eventStatus.position.x += dx;
|
|
ui.eventStatus.position.y += dy;
|
|
|
|
ui.eventStatus.lastPos.x = ui.eventStatus.position.x;
|
|
ui.eventStatus.lastPos.y = ui.eventStatus.position.y;
|
|
}
|
|
|
|
var eventList = [];
|
|
var maxEvents = 8;
|
|
|
|
ui.logEvent = function (event) {
|
|
eventList.push(event);
|
|
if (eventList.length >= maxEvents)
|
|
eventList.shift();
|
|
ui.eventStatus.updateText(eventList.join('\n'));
|
|
}
|
|
|
|
|
|
// Tries to find a widget with an overlay matching overlay.
|
|
// Used by getFocusedWidget(), dispatchEvent(), etc
|
|
var getWidgetWithOverlay = function (overlay) {
|
|
var foundWidget = null;
|
|
ui.widgetList.forEach(function(widget) {
|
|
if (widget.hasOverlay(overlay)) {
|
|
foundWidget = widget;
|
|
return;
|
|
}
|
|
});
|
|
|
|
ui.focusStatus.updateText("Widget focus: " + foundWidget);
|
|
return foundWidget;
|
|
}
|
|
|
|
var getFocusedWidget = function (event) {
|
|
return getWidgetWithOverlay(Overlays.getOverlayAtPoint({ 'x': event.x, 'y': event.y }));
|
|
}
|
|
|
|
var dispatchEvent = function (action, event, widget) {
|
|
function dispatchActions (actions) {
|
|
var dispatchedActions = false;
|
|
ui.logEvent("dispatching to [" + actions.join(", ") + "]");
|
|
actions.forEach(function(action) {
|
|
action(event, widget);
|
|
ui.logEvent("dispatched to " + action);
|
|
dispatchedActions = true;
|
|
});
|
|
if (!dispatchedActions) {
|
|
// ui.logEvent("No actions to dispatch");
|
|
}
|
|
}
|
|
|
|
if (widget.actions[action]) {
|
|
ui.logEvent("Dispatching action '" + action + "'' to " + widget)
|
|
dispatchActions(widget.actions[action]);
|
|
} else {
|
|
for (var parent = widget.parent; parent != null; parent = parent.parent) {
|
|
if (parent.actions[action]) {
|
|
ui.logEvent("Dispatching action '" + action + "'' to parent widget " + widget);
|
|
dispatchActions(parent.actions[action]);
|
|
return;
|
|
}
|
|
}
|
|
ui.logEvent("No action '" + action + "' in " + widget);
|
|
}
|
|
}
|
|
|
|
function hasAction (widget, action) {
|
|
// print("widget = " + widget);
|
|
// print("action = " + action);
|
|
// if (widget) {
|
|
// print("widget.actions[<action>] = " + widget.actions[action]);
|
|
// print("widget.parent = " + widget.parent);
|
|
// }
|
|
return widget && (widget.actions[action] || hasAction(widget.parent, action));
|
|
}
|
|
|
|
UI.handleMouseMove = function (event, canStartDrag) {
|
|
// if (canStartDrag === undefined)
|
|
if (arguments.length < 2)
|
|
canStartDrag = true;
|
|
|
|
// print("mouse moved x = " + event.x + ", y = " + event.y);
|
|
var focused = getFocusedWidget(event);
|
|
|
|
if (!ui.draggedWidget && ui.clickedWidget && hasAction(ui.clickedWidget, 'onDragBegin')) {
|
|
ui.draggedWidget = ui.clickedWidget;
|
|
dispatchEvent('onDragBegin', event, ui.draggedWidget);
|
|
} else if (ui.draggedWidget) {
|
|
dispatchEvent('onDragUpdate', event, ui.draggedWidget);
|
|
} else if (focused != ui.focusedWidget) {
|
|
if (ui.focusedWidget)
|
|
dispatchEvent('onMouseExit', event, ui.focusedWidget);
|
|
if (focused)
|
|
dispatchEvent('onMouseOver', event, focused);
|
|
ui.focusedWidget = focused;
|
|
}
|
|
}
|
|
|
|
UI.handleMousePress = function (event) {
|
|
// print("Mouse clicked");
|
|
UI.handleMouseMove(event);
|
|
ui.clickedWidget = ui.focusedWidget;
|
|
if (ui.clickedWidget) {
|
|
dispatchEvent('onMouseDown', event, ui.clickedWidget);
|
|
}
|
|
}
|
|
|
|
UI.handleMouseDoublePress = function (event) {
|
|
var focused = getFocusedWidget(event);
|
|
UI.handleMouseMove(event);
|
|
if (focused) {
|
|
dispatchEvent('onDoubleClick', event, focused);
|
|
}
|
|
}
|
|
|
|
UI.handleMouseRelease = function (event) {
|
|
if (ui.draggedWidget) {
|
|
dispatchEvent('onDragEnd', event, ui.draggedWidget);
|
|
} else {
|
|
var clicked = ui.clickedWidget;
|
|
ui.clickedWidget = null;
|
|
UI.handleMouseMove(event);
|
|
if (ui.focusedWidget) {
|
|
dispatchEvent('onMouseUp', event, ui.focusedWidget);
|
|
|
|
if (clicked == ui.focusedWidget) {
|
|
dispatchEvent('onClick', event, ui.focusedWidget);
|
|
}
|
|
}
|
|
}
|
|
ui.clickedWidget = null;
|
|
ui.draggedWidget = null;
|
|
}
|
|
|
|
UI.teardown = function () {
|
|
print("Teardown");
|
|
ui.widgetList.forEach(function(widget) {
|
|
widget.destroy();
|
|
});
|
|
ui.widgetList = [];
|
|
ui.focusedWidget = null;
|
|
};
|
|
UI.setErrorHandler = function (errorHandler) {
|
|
if (typeof(errorHandler) !== 'function') {
|
|
ui.complain("UI.setErrorHandler -- invalid argument: \"" + errorHandler + "\"");
|
|
} else {
|
|
ui.errorHandler = errorHandler;
|
|
}
|
|
}
|
|
|
|
UI.printWidgets = function () {
|
|
print("widgetlist.length = " + ui.widgetList.length);
|
|
ui.widgetList.forEach(function(widget) {
|
|
print(""+widget + " position=(" + widget.position.x + ", " + widget.position.y + ")" +
|
|
" parent = " + widget.parent + " visible = " + widget.isVisible() +
|
|
" width = " + widget.getWidth() + ", height = " + widget.getHeight() +
|
|
" overlay = " + (widget.getOverlay() && widget.getOverlay().getId()) +
|
|
(widget.border ? " border = " + widget.border.x + ", " + widget.border.y : "") +
|
|
(widget.padding ? " padding = " + widget.padding.x + ", " + widget.padding.y : ""));
|
|
});
|
|
}
|
|
|
|
})();
|