webui-aria2/js/libs/dojox/mobile/Slider.js.uncompressed.js
2012-05-01 19:52:07 +08:00

165 lines
6.8 KiB
JavaScript

//>>built
define("dojox/mobile/Slider", [
"dojo/_base/array",
"dojo/_base/connect",
"dojo/_base/declare",
"dojo/_base/lang",
"dojo/_base/window",
"dojo/dom-class",
"dojo/dom-construct",
"dojo/dom-geometry",
"dojo/dom-style",
"dijit/_WidgetBase",
"dijit/form/_FormValueMixin"
],
function(array, connect, declare, lang, win, domClass, domConstruct, domGeometry, domStyle, WidgetBase, FormValueMixin){
/*=====
WidgetBase = dijit._WidgetBase;
FormValueMixin = dijit.form._FormValueMixin;
=====*/
return declare("dojox.mobile.Slider", [WidgetBase, FormValueMixin], {
// summary:
// A non-templated Slider widget similar to the HTML5 INPUT type=range.
//
// value: [const] Number
// The current slider value.
value: 0,
// min: [const] Number
// The first value the slider can be set to.
min: 0,
// max: [const] Number
// The last value the slider can be set to.
max: 100,
// step: [const] Number
// The delta from 1 value to another.
// This causes the slider handle to snap/jump to the closest possible value.
// A value of 0 means continuous (as much as allowed by pixel resolution).
step: 1,
baseClass: "mblSlider",
// flip: [const] Boolean
// Specifies if the slider should change its default: ascending <--> descending.
flip: false,
// orientation: [const] String
// The slider direction.
// "H": horizontal
// "V": vertical
// "auto": use width/height comparison at instantiation time (default is "H" if width/height are 0)
orientation: "auto",
// halo: Number
// Size of the boundary that extends beyond the edges of the slider
// to make it easier to touch.
halo: "8pt",
buildRendering: function(){
this.focusNode = this.domNode = domConstruct.create("div", {});
this.valueNode = domConstruct.create("input", (this.srcNodeRef && this.srcNodeRef.name) ? { type: "hidden", name: this.srcNodeRef.name } : { type: "hidden" }, this.domNode, "last");
var relativeParent = domConstruct.create("div", { style: { position:"relative", height:"100%", width:"100%" } }, this.domNode, "last");
this.progressBar = domConstruct.create("div", { style:{ position:"absolute" }, "class":"mblSliderProgressBar" }, relativeParent, "last");
this.touchBox = domConstruct.create("div", { style:{ position:"absolute" }, "class":"mblSliderTouchBox" }, relativeParent, "last");
this.handle = domConstruct.create("div", { style:{ position:"absolute" }, "class":"mblSliderHandle" }, relativeParent, "last");
this.inherited(arguments);
},
_setValueAttr: function(/*Number*/ value, /*Boolean?*/ priorityChange){
// summary:
// Hook so set('value', value) works.
var fromPercent = (this.value - this.min) * 100 / (this.max - this.min);
this.valueNode.value = value;
this.inherited(arguments);
if(!this._started){ return; } // don't move images until all the properties are set
this.focusNode.setAttribute("aria-valuenow", value);
var toPercent = (value - this.min) * 100 / (this.max - this.min);
// now perform visual slide
var horizontal = this.orientation != "V";
if(priorityChange === true){
domClass.add(this.handle, "mblSliderTransition");
domClass.add(this.progressBar, "mblSliderTransition");
}else{
domClass.remove(this.handle, "mblSliderTransition");
domClass.remove(this.progressBar, "mblSliderTransition");
}
domStyle.set(this.handle, this._attrs.handleLeft, (this._reversed ? (100-toPercent) : toPercent) + "%");
domStyle.set(this.progressBar, this._attrs.width, toPercent + "%");
},
postCreate: function(){
this.inherited(arguments);
function beginDrag(e){
function getEventData(e){
point = isMouse ? e[this._attrs.pageX] : (e.touches ? e.touches[0][this._attrs.pageX] : e[this._attrs.clientX]);
pixelValue = point - startPixel;
pixelValue = Math.min(Math.max(pixelValue, 0), maxPixels);
var discreteValues = this.step ? ((this.max - this.min) / this.step) : maxPixels;
if(discreteValues <= 1 || discreteValues == Infinity ){ discreteValues = maxPixels; }
var wholeIncrements = Math.round(pixelValue * discreteValues / maxPixels);
value = (this.max - this.min) * wholeIncrements / discreteValues;
value = this._reversed ? (this.max - value) : (this.min + value);
}
function continueDrag(e){
e.preventDefault();
lang.hitch(this, getEventData)(e);
this.set('value', value, false);
}
function endDrag(e){
e.preventDefault();
array.forEach(actionHandles, lang.hitch(this, "disconnect"));
actionHandles = [];
this.set('value', this.value, true);
}
e.preventDefault();
var isMouse = e.type == "mousedown";
var box = domGeometry.position(node, false); // can't use true since the added docScroll and the returned x are body-zoom incompatibile
var bodyZoom = domStyle.get(win.body(), "zoom") || 1;
if(isNaN(bodyZoom)){ bodyZoom = 1; }
var nodeZoom = domStyle.get(node, "zoom") || 1;
if(isNaN(nodeZoom)){ nodeZoom = 1; }
var startPixel = box[this._attrs.x] * nodeZoom * bodyZoom + domGeometry.docScroll()[this._attrs.x];
var maxPixels = box[this._attrs.w] * nodeZoom * bodyZoom;
lang.hitch(this, getEventData)(e);
if(e.target == this.touchBox){
this.set('value', value, true);
}
array.forEach(actionHandles, connect.disconnect);
var root = win.doc.documentElement;
var actionHandles = [
this.connect(root, isMouse ? "onmousemove" : "ontouchmove", continueDrag),
this.connect(root, isMouse ? "onmouseup" : "ontouchend", endDrag)
];
}
var point, pixelValue, value;
var node = this.domNode;
if(this.orientation == "auto"){
this.orientation = node.offsetHeight <= node.offsetWidth ? "H" : "V";
}
// add V or H suffix to baseClass for styling purposes
domClass.add(this.domNode, array.map(this.baseClass.split(" "), lang.hitch(this, function(c){ return c+this.orientation; })));
var horizontal = this.orientation != "V";
var ltr = horizontal ? this.isLeftToRight() : false;
var flip = this.flip;
// _reversed is complicated since you can have flipped right-to-left and vertical is upside down by default
this._reversed = !(horizontal && ((ltr && !flip) || (!ltr && flip))) || (!horizontal && !flip);
this._attrs = horizontal ? { x:'x', w:'w', l:'l', r:'r', pageX:'pageX', clientX:'clientX', handleLeft:"left", left:this._reversed ? "right" : "left", width:"width" } : { x:'y', w:'h', l:'t', r:'b', pageX:'pageY', clientX:'clientY', handleLeft:"top", left:this._reversed ? "bottom" : "top", width:"height" };
this.progressBar.style[this._attrs.left] = "0px";
this.connect(this.touchBox, "touchstart", beginDrag);
this.connect(this.touchBox, "onmousedown", beginDrag); // in case this works
this.connect(this.handle, "touchstart", beginDrag);
this.connect(this.handle, "onmousedown", beginDrag); // in case this works
this.startup();
this.set('value', this.value);
}
});
});