diff --git a/dist/famous-global.js b/dist/famous-global.js new file mode 100644 index 00000000..37a957d1 --- /dev/null +++ b/dist/famous-global.js @@ -0,0 +1,10889 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.famous=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o 0) { + result = nodeStore.pop(); + } else { + result = document.createElement(type); + this.container.appendChild(result); + } + this.nodeCount++; + return result; +}; +ElementAllocator.prototype.deallocate = function deallocate(element) { + var nodeType = element.nodeName.toLowerCase(); + var nodeStore = this.detachedNodes[nodeType]; + nodeStore.push(element); + this.nodeCount--; +}; +ElementAllocator.prototype.getNodeCount = function getNodeCount() { + return this.nodeCount; +}; +module.exports = ElementAllocator; +},{}],3:[function(_dereq_,module,exports){ +var Entity = _dereq_('./Entity'); +var EventHandler = _dereq_('./EventHandler'); +var Transform = _dereq_('./Transform'); +var usePrefix = !('transform' in document.documentElement.style); +var devicePixelRatio = window.devicePixelRatio || 1; +function ElementOutput(element) { + this._matrix = null; + this._opacity = 1; + this._origin = null; + this._size = null; + this._eventOutput = new EventHandler(); + this._eventOutput.bindThis(this); + this.eventForwarder = function eventForwarder(event) { + this._eventOutput.emit(event.type, event); + }.bind(this); + this.id = Entity.register(this); + this._element = null; + this._sizeDirty = false; + this._originDirty = false; + this._transformDirty = false; + this._invisible = false; + if (element) + this.attach(element); +} +ElementOutput.prototype.on = function on(type, fn) { + if (this._element) + this._element.addEventListener(type, this.eventForwarder); + this._eventOutput.on(type, fn); +}; +ElementOutput.prototype.removeListener = function removeListener(type, fn) { + this._eventOutput.removeListener(type, fn); +}; +ElementOutput.prototype.emit = function emit(type, event) { + if (event && !event.origin) + event.origin = this; + var handled = this._eventOutput.emit(type, event); + if (handled && event && event.stopPropagation) + event.stopPropagation(); + return handled; +}; +ElementOutput.prototype.pipe = function pipe(target) { + return this._eventOutput.pipe(target); +}; +ElementOutput.prototype.unpipe = function unpipe(target) { + return this._eventOutput.unpipe(target); +}; +ElementOutput.prototype.render = function render() { + return this.id; +}; +function _addEventListeners(target) { + for (var i in this._eventOutput.listeners) { + target.addEventListener(i, this.eventForwarder); + } +} +function _removeEventListeners(target) { + for (var i in this._eventOutput.listeners) { + target.removeEventListener(i, this.eventForwarder); + } +} +function _formatCSSTransform(m) { + m[12] = Math.round(m[12] * devicePixelRatio) / devicePixelRatio; + m[13] = Math.round(m[13] * devicePixelRatio) / devicePixelRatio; + var result = 'matrix3d('; + for (var i = 0; i < 15; i++) { + result += m[i] < 0.000001 && m[i] > -0.000001 ? '0,' : m[i] + ','; + } + result += m[15] + ')'; + return result; +} +var _setMatrix; +if (usePrefix) { + _setMatrix = function (element, matrix) { + element.style.webkitTransform = _formatCSSTransform(matrix); + }; +} else { + _setMatrix = function (element, matrix) { + element.style.transform = _formatCSSTransform(matrix); + }; +} +function _formatCSSOrigin(origin) { + return 100 * origin[0] + '% ' + 100 * origin[1] + '%'; +} +var _setOrigin = usePrefix ? function (element, origin) { + element.style.webkitTransformOrigin = _formatCSSOrigin(origin); + } : function (element, origin) { + element.style.transformOrigin = _formatCSSOrigin(origin); + }; +var _setInvisible = usePrefix ? function (element) { + element.style.webkitTransform = 'scale3d(0.0001,0.0001,0.0001)'; + element.style.opacity = 0; + } : function (element) { + element.style.transform = 'scale3d(0.0001,0.0001,0.0001)'; + element.style.opacity = 0; + }; +function _xyNotEquals(a, b) { + return a && b ? a[0] !== b[0] || a[1] !== b[1] : a !== b; +} +ElementOutput.prototype.commit = function commit(context) { + var target = this._element; + if (!target) + return; + var matrix = context.transform; + var opacity = context.opacity; + var origin = context.origin; + var size = context.size; + if (!matrix && this._matrix) { + this._matrix = null; + this._opacity = 0; + _setInvisible(target); + return; + } + if (_xyNotEquals(this._origin, origin)) + this._originDirty = true; + if (Transform.notEquals(this._matrix, matrix)) + this._transformDirty = true; + if (this._invisible) { + this._invisible = false; + this._element.style.display = ''; + } + if (this._opacity !== opacity) { + this._opacity = opacity; + target.style.opacity = opacity >= 1 ? '0.999999' : opacity; + } + if (this._transformDirty || this._originDirty || this._sizeDirty) { + if (this._sizeDirty) + this._sizeDirty = false; + if (this._originDirty) { + if (origin) { + if (!this._origin) + this._origin = [ + 0, + 0 + ]; + this._origin[0] = origin[0]; + this._origin[1] = origin[1]; + } else + this._origin = null; + _setOrigin(target, this._origin); + this._originDirty = false; + } + if (!matrix) + matrix = Transform.identity; + this._matrix = matrix; + var aaMatrix = this._size ? Transform.thenMove(matrix, [ + -this._size[0] * origin[0], + -this._size[1] * origin[1], + 0 + ]) : matrix; + _setMatrix(target, aaMatrix); + this._transformDirty = false; + } +}; +ElementOutput.prototype.cleanup = function cleanup() { + if (this._element) { + this._invisible = true; + this._element.style.display = 'none'; + } +}; +ElementOutput.prototype.attach = function attach(target) { + this._element = target; + _addEventListeners.call(this, target); +}; +ElementOutput.prototype.detach = function detach() { + var target = this._element; + if (target) { + _removeEventListeners.call(this, target); + if (this._invisible) { + this._invisible = false; + this._element.style.display = ''; + } + } + this._element = null; + return target; +}; +module.exports = ElementOutput; +},{"./Entity":5,"./EventHandler":7,"./Transform":15}],4:[function(_dereq_,module,exports){ +var Context = _dereq_('./Context'); +var EventHandler = _dereq_('./EventHandler'); +var OptionsManager = _dereq_('./OptionsManager'); +var Engine = {}; +var contexts = []; +var nextTickQueue = []; +var currentFrame = 0; +var nextTickFrame = 0; +var deferQueue = []; +var lastTime = Date.now(); +var frameTime; +var frameTimeLimit; +var loopEnabled = true; +var eventForwarders = {}; +var eventHandler = new EventHandler(); +var options = { + containerType: 'div', + containerClass: 'famous-container', + fpsCap: undefined, + runLoop: true, + appMode: true + }; +var optionsManager = new OptionsManager(options); +var MAX_DEFER_FRAME_TIME = 10; +Engine.step = function step() { + currentFrame++; + nextTickFrame = currentFrame; + var currentTime = Date.now(); + if (frameTimeLimit && currentTime - lastTime < frameTimeLimit) + return; + var i = 0; + frameTime = currentTime - lastTime; + lastTime = currentTime; + eventHandler.emit('prerender'); + var numFunctions = nextTickQueue.length; + while (numFunctions--) + nextTickQueue.shift()(currentFrame); + while (deferQueue.length && Date.now() - currentTime < MAX_DEFER_FRAME_TIME) { + deferQueue.shift().call(this); + } + for (i = 0; i < contexts.length; i++) + contexts[i].update(); + eventHandler.emit('postrender'); +}; +function loop() { + if (options.runLoop) { + Engine.step(); + window.requestAnimationFrame(loop); + } else + loopEnabled = false; +} +window.requestAnimationFrame(loop); +function handleResize(event) { + for (var i = 0; i < contexts.length; i++) { + contexts[i].emit('resize'); + } + eventHandler.emit('resize'); +} +window.addEventListener('resize', handleResize, false); +handleResize(); +function initialize() { + window.addEventListener('touchmove', function (event) { + event.preventDefault(); + }, true); + addRootClasses(); +} +var initialized = false; +function addRootClasses() { + if (!document.body) { + Engine.nextTick(addRootClasses); + return; + } + document.body.classList.add('famous-root'); + document.documentElement.classList.add('famous-root'); +} +Engine.pipe = function pipe(target) { + if (target.subscribe instanceof Function) + return target.subscribe(Engine); + else + return eventHandler.pipe(target); +}; +Engine.unpipe = function unpipe(target) { + if (target.unsubscribe instanceof Function) + return target.unsubscribe(Engine); + else + return eventHandler.unpipe(target); +}; +Engine.on = function on(type, handler) { + if (!(type in eventForwarders)) { + eventForwarders[type] = eventHandler.emit.bind(eventHandler, type); + addEngineListener(type, eventForwarders[type]); + } + return eventHandler.on(type, handler); +}; +function addEngineListener(type, forwarder) { + if (!document.body) { + Engine.nextTick(addEventListener.bind(this, type, forwarder)); + return; + } + document.body.addEventListener(type, forwarder); +} +Engine.emit = function emit(type, event) { + return eventHandler.emit(type, event); +}; +Engine.removeListener = function removeListener(type, handler) { + return eventHandler.removeListener(type, handler); +}; +Engine.getFPS = function getFPS() { + return 1000 / frameTime; +}; +Engine.setFPSCap = function setFPSCap(fps) { + frameTimeLimit = Math.floor(1000 / fps); +}; +Engine.getOptions = function getOptions(key) { + return optionsManager.getOptions(key); +}; +Engine.setOptions = function setOptions(options) { + return optionsManager.setOptions.apply(optionsManager, arguments); +}; +Engine.createContext = function createContext(el) { + if (!initialized && options.appMode) + Engine.nextTick(initialize); + var needMountContainer = false; + if (!el) { + el = document.createElement(options.containerType); + el.classList.add(options.containerClass); + needMountContainer = true; + } + var context = new Context(el); + Engine.registerContext(context); + if (needMountContainer) + mount(context, el); + return context; +}; +function mount(context, el) { + if (!document.body) { + Engine.nextTick(mount.bind(this, context, el)); + return; + } + document.body.appendChild(el); + context.emit('resize'); +} +Engine.registerContext = function registerContext(context) { + contexts.push(context); + return context; +}; +Engine.getContexts = function getContexts() { + return contexts; +}; +Engine.deregisterContext = function deregisterContext(context) { + var i = contexts.indexOf(context); + if (i >= 0) + contexts.splice(i, 1); +}; +Engine.nextTick = function nextTick(fn) { + nextTickQueue.push(fn); +}; +Engine.defer = function defer(fn) { + deferQueue.push(fn); +}; +optionsManager.on('change', function (data) { + if (data.id === 'fpsCap') + Engine.setFPSCap(data.value); + else if (data.id === 'runLoop') { + if (!loopEnabled && data.value) { + loopEnabled = true; + window.requestAnimationFrame(loop); + } + } +}); +module.exports = Engine; +},{"./Context":1,"./EventHandler":7,"./OptionsManager":10}],5:[function(_dereq_,module,exports){ +var entities = []; +function get(id) { + return entities[id]; +} +function set(id, entity) { + entities[id] = entity; +} +function register(entity) { + var id = entities.length; + set(id, entity); + return id; +} +function unregister(id) { + set(id, null); +} +module.exports = { + register: register, + unregister: unregister, + get: get, + set: set +}; +},{}],6:[function(_dereq_,module,exports){ +function EventEmitter() { + this.listeners = {}; + this._owner = this; +} +EventEmitter.prototype.emit = function emit(type, event) { + var handlers = this.listeners[type]; + if (handlers) { + for (var i = 0; i < handlers.length; i++) { + handlers[i].call(this._owner, event); + } + } + return this; +}; +EventEmitter.prototype.on = function on(type, handler) { + if (!(type in this.listeners)) + this.listeners[type] = []; + var index = this.listeners[type].indexOf(handler); + if (index < 0) + this.listeners[type].push(handler); + return this; +}; +EventEmitter.prototype.addListener = EventEmitter.prototype.on; +EventEmitter.prototype.removeListener = function removeListener(type, handler) { + var listener = this.listeners[type]; + if (listener !== undefined) { + var index = listener.indexOf(handler); + if (index >= 0) + listener.splice(index, 1); + } + return this; +}; +EventEmitter.prototype.bindThis = function bindThis(owner) { + this._owner = owner; +}; +module.exports = EventEmitter; +},{}],7:[function(_dereq_,module,exports){ +var EventEmitter = _dereq_('./EventEmitter'); +function EventHandler() { + EventEmitter.apply(this, arguments); + this.downstream = []; + this.downstreamFn = []; + this.upstream = []; + this.upstreamListeners = {}; +} +EventHandler.prototype = Object.create(EventEmitter.prototype); +EventHandler.prototype.constructor = EventHandler; +EventHandler.setInputHandler = function setInputHandler(object, handler) { + object.trigger = handler.trigger.bind(handler); + if (handler.subscribe && handler.unsubscribe) { + object.subscribe = handler.subscribe.bind(handler); + object.unsubscribe = handler.unsubscribe.bind(handler); + } +}; +EventHandler.setOutputHandler = function setOutputHandler(object, handler) { + if (handler instanceof EventHandler) + handler.bindThis(object); + object.pipe = handler.pipe.bind(handler); + object.unpipe = handler.unpipe.bind(handler); + object.on = handler.on.bind(handler); + object.addListener = object.on; + object.removeListener = handler.removeListener.bind(handler); +}; +EventHandler.prototype.emit = function emit(type, event) { + EventEmitter.prototype.emit.apply(this, arguments); + var i = 0; + for (i = 0; i < this.downstream.length; i++) { + if (this.downstream[i].trigger) + this.downstream[i].trigger(type, event); + } + for (i = 0; i < this.downstreamFn.length; i++) { + this.downstreamFn[i](type, event); + } + return this; +}; +EventHandler.prototype.trigger = EventHandler.prototype.emit; +EventHandler.prototype.pipe = function pipe(target) { + if (target.subscribe instanceof Function) + return target.subscribe(this); + var downstreamCtx = target instanceof Function ? this.downstreamFn : this.downstream; + var index = downstreamCtx.indexOf(target); + if (index < 0) + downstreamCtx.push(target); + if (target instanceof Function) + target('pipe', null); + else if (target.trigger) + target.trigger('pipe', null); + return target; +}; +EventHandler.prototype.unpipe = function unpipe(target) { + if (target.unsubscribe instanceof Function) + return target.unsubscribe(this); + var downstreamCtx = target instanceof Function ? this.downstreamFn : this.downstream; + var index = downstreamCtx.indexOf(target); + if (index >= 0) { + downstreamCtx.splice(index, 1); + if (target instanceof Function) + target('unpipe', null); + else if (target.trigger) + target.trigger('unpipe', null); + return target; + } else + return false; +}; +EventHandler.prototype.on = function on(type, handler) { + EventEmitter.prototype.on.apply(this, arguments); + if (!(type in this.upstreamListeners)) { + var upstreamListener = this.trigger.bind(this, type); + this.upstreamListeners[type] = upstreamListener; + for (var i = 0; i < this.upstream.length; i++) { + this.upstream[i].on(type, upstreamListener); + } + } + return this; +}; +EventHandler.prototype.addListener = EventHandler.prototype.on; +EventHandler.prototype.subscribe = function subscribe(source) { + var index = this.upstream.indexOf(source); + if (index < 0) { + this.upstream.push(source); + for (var type in this.upstreamListeners) { + source.on(type, this.upstreamListeners[type]); + } + } + return this; +}; +EventHandler.prototype.unsubscribe = function unsubscribe(source) { + var index = this.upstream.indexOf(source); + if (index >= 0) { + this.upstream.splice(index, 1); + for (var type in this.upstreamListeners) { + source.removeListener(type, this.upstreamListeners[type]); + } + } + return this; +}; +module.exports = EventHandler; +},{"./EventEmitter":6}],8:[function(_dereq_,module,exports){ +var Context = _dereq_('./Context'); +var Transform = _dereq_('./Transform'); +var Surface = _dereq_('./Surface'); +function Group(options) { + Surface.call(this, options); + this._shouldRecalculateSize = false; + this._container = document.createDocumentFragment(); + this.context = new Context(this._container); + this.setContent(this._container); + this._groupSize = [ + undefined, + undefined + ]; +} +Group.SIZE_ZERO = [ + 0, + 0 +]; +Group.prototype = Object.create(Surface.prototype); +Group.prototype.elementType = 'div'; +Group.prototype.elementClass = 'famous-group'; +Group.prototype.add = function add() { + return this.context.add.apply(this.context, arguments); +}; +Group.prototype.render = function render() { + return Surface.prototype.render.call(this); +}; +Group.prototype.deploy = function deploy(target) { + this.context.migrate(target); +}; +Group.prototype.recall = function recall(target) { + this._container = document.createDocumentFragment(); + this.context.migrate(this._container); +}; +Group.prototype.commit = function commit(context) { + var transform = context.transform; + var origin = context.origin; + var opacity = context.opacity; + var size = context.size; + var result = Surface.prototype.commit.call(this, { + allocator: context.allocator, + transform: Transform.thenMove(transform, [ + -origin[0] * size[0], + -origin[1] * size[1], + 0 + ]), + opacity: opacity, + origin: origin, + size: Group.SIZE_ZERO + }); + if (size[0] !== this._groupSize[0] || size[1] !== this._groupSize[1]) { + this._groupSize[0] = size[0]; + this._groupSize[1] = size[1]; + this.context.setSize(size); + } + this.context.update({ + transform: Transform.translate(-origin[0] * size[0], -origin[1] * size[1], 0), + origin: origin, + size: size + }); + return result; +}; +module.exports = Group; +},{"./Context":1,"./Surface":14,"./Transform":15}],9:[function(_dereq_,module,exports){ +var Transform = _dereq_('./Transform'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var TransitionableTransform = _dereq_('../transitions/TransitionableTransform'); +function Modifier(options) { + this._transformGetter = null; + this._opacityGetter = null; + this._originGetter = null; + this._alignGetter = null; + this._sizeGetter = null; + this._proportionGetter = null; + this._legacyStates = {}; + this._output = { + transform: Transform.identity, + opacity: 1, + origin: null, + align: null, + size: null, + proportions: null, + target: null + }; + if (options) { + if (options.transform) + this.transformFrom(options.transform); + if (options.opacity !== undefined) + this.opacityFrom(options.opacity); + if (options.origin) + this.originFrom(options.origin); + if (options.align) + this.alignFrom(options.align); + if (options.size) + this.sizeFrom(options.size); + if (options.proportions) + this.proportionsFrom(options.proportions); + } +} +Modifier.prototype.transformFrom = function transformFrom(transform) { + if (transform instanceof Function) + this._transformGetter = transform; + else if (transform instanceof Object && transform.get) + this._transformGetter = transform.get.bind(transform); + else { + this._transformGetter = null; + this._output.transform = transform; + } + return this; +}; +Modifier.prototype.opacityFrom = function opacityFrom(opacity) { + if (opacity instanceof Function) + this._opacityGetter = opacity; + else if (opacity instanceof Object && opacity.get) + this._opacityGetter = opacity.get.bind(opacity); + else { + this._opacityGetter = null; + this._output.opacity = opacity; + } + return this; +}; +Modifier.prototype.originFrom = function originFrom(origin) { + if (origin instanceof Function) + this._originGetter = origin; + else if (origin instanceof Object && origin.get) + this._originGetter = origin.get.bind(origin); + else { + this._originGetter = null; + this._output.origin = origin; + } + return this; +}; +Modifier.prototype.alignFrom = function alignFrom(align) { + if (align instanceof Function) + this._alignGetter = align; + else if (align instanceof Object && align.get) + this._alignGetter = align.get.bind(align); + else { + this._alignGetter = null; + this._output.align = align; + } + return this; +}; +Modifier.prototype.sizeFrom = function sizeFrom(size) { + if (size instanceof Function) + this._sizeGetter = size; + else if (size instanceof Object && size.get) + this._sizeGetter = size.get.bind(size); + else { + this._sizeGetter = null; + this._output.size = size; + } + return this; +}; +Modifier.prototype.proportionsFrom = function proportionsFrom(proportions) { + if (proportions instanceof Function) + this._proportionGetter = proportions; + else if (proportions instanceof Object && proportions.get) + this._proportionGetter = proportions.get.bind(proportions); + else { + this._proportionGetter = null; + this._output.proportions = proportions; + } + return this; +}; +Modifier.prototype.setTransform = function setTransform(transform, transition, callback) { + if (transition || this._legacyStates.transform) { + if (!this._legacyStates.transform) { + this._legacyStates.transform = new TransitionableTransform(this._output.transform); + } + if (!this._transformGetter) + this.transformFrom(this._legacyStates.transform); + this._legacyStates.transform.set(transform, transition, callback); + return this; + } else + return this.transformFrom(transform); +}; +Modifier.prototype.setOpacity = function setOpacity(opacity, transition, callback) { + if (transition || this._legacyStates.opacity) { + if (!this._legacyStates.opacity) { + this._legacyStates.opacity = new Transitionable(this._output.opacity); + } + if (!this._opacityGetter) + this.opacityFrom(this._legacyStates.opacity); + return this._legacyStates.opacity.set(opacity, transition, callback); + } else + return this.opacityFrom(opacity); +}; +Modifier.prototype.setOrigin = function setOrigin(origin, transition, callback) { + if (transition || this._legacyStates.origin) { + if (!this._legacyStates.origin) { + this._legacyStates.origin = new Transitionable(this._output.origin || [ + 0, + 0 + ]); + } + if (!this._originGetter) + this.originFrom(this._legacyStates.origin); + this._legacyStates.origin.set(origin, transition, callback); + return this; + } else + return this.originFrom(origin); +}; +Modifier.prototype.setAlign = function setAlign(align, transition, callback) { + if (transition || this._legacyStates.align) { + if (!this._legacyStates.align) { + this._legacyStates.align = new Transitionable(this._output.align || [ + 0, + 0 + ]); + } + if (!this._alignGetter) + this.alignFrom(this._legacyStates.align); + this._legacyStates.align.set(align, transition, callback); + return this; + } else + return this.alignFrom(align); +}; +Modifier.prototype.setSize = function setSize(size, transition, callback) { + if (size && (transition || this._legacyStates.size)) { + if (!this._legacyStates.size) { + this._legacyStates.size = new Transitionable(this._output.size || [ + 0, + 0 + ]); + } + if (!this._sizeGetter) + this.sizeFrom(this._legacyStates.size); + this._legacyStates.size.set(size, transition, callback); + return this; + } else + return this.sizeFrom(size); +}; +Modifier.prototype.setProportions = function setProportions(proportions, transition, callback) { + if (proportions && (transition || this._legacyStates.proportions)) { + if (!this._legacyStates.proportions) { + this._legacyStates.proportions = new Transitionable(this._output.proportions || [ + 0, + 0 + ]); + } + if (!this._proportionGetter) + this.proportionsFrom(this._legacyStates.proportions); + this._legacyStates.proportions.set(proportions, transition, callback); + return this; + } else + return this.proportionsFrom(proportions); +}; +Modifier.prototype.halt = function halt() { + if (this._legacyStates.transform) + this._legacyStates.transform.halt(); + if (this._legacyStates.opacity) + this._legacyStates.opacity.halt(); + if (this._legacyStates.origin) + this._legacyStates.origin.halt(); + if (this._legacyStates.align) + this._legacyStates.align.halt(); + if (this._legacyStates.size) + this._legacyStates.size.halt(); + if (this._legacyStates.proportions) + this._legacyStates.proportions.halt(); + this._transformGetter = null; + this._opacityGetter = null; + this._originGetter = null; + this._alignGetter = null; + this._sizeGetter = null; + this._proportionGetter = null; +}; +Modifier.prototype.getTransform = function getTransform() { + return this._transformGetter(); +}; +Modifier.prototype.getFinalTransform = function getFinalTransform() { + return this._legacyStates.transform ? this._legacyStates.transform.getFinal() : this._output.transform; +}; +Modifier.prototype.getOpacity = function getOpacity() { + return this._opacityGetter(); +}; +Modifier.prototype.getOrigin = function getOrigin() { + return this._originGetter(); +}; +Modifier.prototype.getAlign = function getAlign() { + return this._alignGetter(); +}; +Modifier.prototype.getSize = function getSize() { + return this._sizeGetter ? this._sizeGetter() : this._output.size; +}; +Modifier.prototype.getProportions = function getProportions() { + return this._proportionGetter ? this._proportionGetter() : this._output.proportions; +}; +function _update() { + if (this._transformGetter) + this._output.transform = this._transformGetter(); + if (this._opacityGetter) + this._output.opacity = this._opacityGetter(); + if (this._originGetter) + this._output.origin = this._originGetter(); + if (this._alignGetter) + this._output.align = this._alignGetter(); + if (this._sizeGetter) + this._output.size = this._sizeGetter(); + if (this._proportionGetter) + this._output.proportions = this._proportionGetter(); +} +Modifier.prototype.modify = function modify(target) { + _update.call(this); + this._output.target = target; + return this._output; +}; +module.exports = Modifier; +},{"../transitions/Transitionable":88,"../transitions/TransitionableTransform":89,"./Transform":15}],10:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('./EventHandler'); +function OptionsManager(value) { + this._value = value; + this.eventOutput = null; +} +OptionsManager.patch = function patchObject(source, data) { + var manager = new OptionsManager(source); + for (var i = 1; i < arguments.length; i++) + manager.patch(arguments[i]); + return source; +}; +function _createEventOutput() { + this.eventOutput = new EventHandler(); + this.eventOutput.bindThis(this); + EventHandler.setOutputHandler(this, this.eventOutput); +} +OptionsManager.prototype.patch = function patch() { + var myState = this._value; + for (var i = 0; i < arguments.length; i++) { + var data = arguments[i]; + for (var k in data) { + if (k in myState && (data[k] && data[k].constructor === Object) && (myState[k] && myState[k].constructor === Object)) { + if (!myState.hasOwnProperty(k)) + myState[k] = Object.create(myState[k]); + this.key(k).patch(data[k]); + if (this.eventOutput) + this.eventOutput.emit('change', { + id: k, + value: this.key(k).value() + }); + } else + this.set(k, data[k]); + } + } + return this; +}; +OptionsManager.prototype.setOptions = OptionsManager.prototype.patch; +OptionsManager.prototype.key = function key(identifier) { + var result = new OptionsManager(this._value[identifier]); + if (!(result._value instanceof Object) || result._value instanceof Array) + result._value = {}; + return result; +}; +OptionsManager.prototype.get = function get(key) { + return key ? this._value[key] : this._value; +}; +OptionsManager.prototype.getOptions = OptionsManager.prototype.get; +OptionsManager.prototype.set = function set(key, value) { + var originalValue = this.get(key); + this._value[key] = value; + if (this.eventOutput && value !== originalValue) + this.eventOutput.emit('change', { + id: key, + value: value + }); + return this; +}; +OptionsManager.prototype.on = function on() { + _createEventOutput.call(this); + return this.on.apply(this, arguments); +}; +OptionsManager.prototype.removeListener = function removeListener() { + _createEventOutput.call(this); + return this.removeListener.apply(this, arguments); +}; +OptionsManager.prototype.pipe = function pipe() { + _createEventOutput.call(this); + return this.pipe.apply(this, arguments); +}; +OptionsManager.prototype.unpipe = function unpipe() { + _createEventOutput.call(this); + return this.unpipe.apply(this, arguments); +}; +module.exports = OptionsManager; +},{"./EventHandler":7}],11:[function(_dereq_,module,exports){ +var Entity = _dereq_('./Entity'); +var SpecParser = _dereq_('./SpecParser'); +function RenderNode(object) { + this._object = null; + this._child = null; + this._hasMultipleChildren = false; + this._isRenderable = false; + this._isModifier = false; + this._resultCache = {}; + this._prevResults = {}; + this._childResult = null; + if (object) + this.set(object); +} +RenderNode.prototype.add = function add(child) { + var childNode = child instanceof RenderNode ? child : new RenderNode(child); + if (this._child instanceof Array) + this._child.push(childNode); + else if (this._child) { + this._child = [ + this._child, + childNode + ]; + this._hasMultipleChildren = true; + this._childResult = []; + } else + this._child = childNode; + return childNode; +}; +RenderNode.prototype.get = function get() { + return this._object || (this._hasMultipleChildren ? null : this._child ? this._child.get() : null); +}; +RenderNode.prototype.set = function set(child) { + this._childResult = null; + this._hasMultipleChildren = false; + this._isRenderable = child.render ? true : false; + this._isModifier = child.modify ? true : false; + this._object = child; + this._child = null; + if (child instanceof RenderNode) + return child; + else + return this; +}; +RenderNode.prototype.getSize = function getSize() { + var result = null; + var target = this.get(); + if (target && target.getSize) + result = target.getSize(); + if (!result && this._child && this._child.getSize) + result = this._child.getSize(); + return result; +}; +function _applyCommit(spec, context, cacheStorage) { + var result = SpecParser.parse(spec, context); + var keys = Object.keys(result); + for (var i = 0; i < keys.length; i++) { + var id = keys[i]; + var childNode = Entity.get(id); + var commitParams = result[id]; + commitParams.allocator = context.allocator; + var commitResult = childNode.commit(commitParams); + if (commitResult) + _applyCommit(commitResult, context, cacheStorage); + else + cacheStorage[id] = commitParams; + } +} +RenderNode.prototype.commit = function commit(context) { + var prevKeys = Object.keys(this._prevResults); + for (var i = 0; i < prevKeys.length; i++) { + var id = prevKeys[i]; + if (this._resultCache[id] === undefined) { + var object = Entity.get(id); + if (object.cleanup) + object.cleanup(context.allocator); + } + } + this._prevResults = this._resultCache; + this._resultCache = {}; + _applyCommit(this.render(), context, this._resultCache); +}; +RenderNode.prototype.render = function render() { + if (this._isRenderable) + return this._object.render(); + var result = null; + if (this._hasMultipleChildren) { + result = this._childResult; + var children = this._child; + for (var i = 0; i < children.length; i++) { + result[i] = children[i].render(); + } + } else if (this._child) + result = this._child.render(); + return this._isModifier ? this._object.modify(result) : result; +}; +module.exports = RenderNode; +},{"./Entity":5,"./SpecParser":13}],12:[function(_dereq_,module,exports){ +var Transform = _dereq_('./Transform'); +var Modifier = _dereq_('./Modifier'); +var RenderNode = _dereq_('./RenderNode'); +function Scene(definition) { + this.id = null; + this._objects = null; + this.node = new RenderNode(); + this._definition = null; + if (definition) + this.load(definition); +} +var _MATRIX_GENERATORS = { + 'translate': Transform.translate, + 'rotate': Transform.rotate, + 'rotateX': Transform.rotateX, + 'rotateY': Transform.rotateY, + 'rotateZ': Transform.rotateZ, + 'rotateAxis': Transform.rotateAxis, + 'scale': Transform.scale, + 'skew': Transform.skew, + 'matrix3d': function () { + return arguments; + } + }; +Scene.prototype.create = function create() { + return new Scene(this._definition); +}; +function _resolveTransformMatrix(matrixDefinition) { + for (var type in _MATRIX_GENERATORS) { + if (type in matrixDefinition) { + var args = matrixDefinition[type]; + if (!(args instanceof Array)) + args = [args]; + return _MATRIX_GENERATORS[type].apply(this, args); + } + } +} +function _parseTransform(definition) { + var transformDefinition = definition.transform; + var opacity = definition.opacity; + var origin = definition.origin; + var align = definition.align; + var size = definition.size; + var transform = Transform.identity; + if (transformDefinition instanceof Array) { + if (transformDefinition.length === 16 && typeof transformDefinition[0] === 'number') { + transform = transformDefinition; + } else { + for (var i = 0; i < transformDefinition.length; i++) { + transform = Transform.multiply(transform, _resolveTransformMatrix(transformDefinition[i])); + } + } + } else if (transformDefinition instanceof Function) { + transform = transformDefinition; + } else if (transformDefinition instanceof Object) { + transform = _resolveTransformMatrix(transformDefinition); + } + var result = new Modifier({ + transform: transform, + opacity: opacity, + origin: origin, + align: align, + size: size + }); + return result; +} +function _parseArray(definition) { + var result = new RenderNode(); + for (var i = 0; i < definition.length; i++) { + var obj = _parse.call(this, definition[i]); + if (obj) + result.add(obj); + } + return result; +} +function _parse(definition) { + var result; + var id; + if (definition instanceof Array) { + result = _parseArray.call(this, definition); + } else { + id = this._objects.length; + if (definition.render && definition.render instanceof Function) { + result = definition; + } else if (definition.target) { + var targetObj = _parse.call(this, definition.target); + var obj = _parseTransform.call(this, definition); + result = new RenderNode(obj); + result.add(targetObj); + if (definition.id) + this.id[definition.id] = obj; + } else if (definition.id) { + result = new RenderNode(); + this.id[definition.id] = result; + } + } + this._objects[id] = result; + return result; +} +Scene.prototype.load = function load(definition) { + this._definition = definition; + this.id = {}; + this._objects = []; + this.node.set(_parse.call(this, definition)); +}; +Scene.prototype.add = function add() { + return this.node.add.apply(this.node, arguments); +}; +Scene.prototype.render = function render() { + return this.node.render.apply(this.node, arguments); +}; +module.exports = Scene; +},{"./Modifier":9,"./RenderNode":11,"./Transform":15}],13:[function(_dereq_,module,exports){ +var Transform = _dereq_('./Transform'); +function SpecParser() { + this.result = {}; +} +SpecParser._instance = new SpecParser(); +SpecParser.parse = function parse(spec, context) { + return SpecParser._instance.parse(spec, context); +}; +SpecParser.prototype.parse = function parse(spec, context) { + this.reset(); + this._parseSpec(spec, context, Transform.identity); + return this.result; +}; +SpecParser.prototype.reset = function reset() { + this.result = {}; +}; +function _vecInContext(v, m) { + return [ + v[0] * m[0] + v[1] * m[4] + v[2] * m[8], + v[0] * m[1] + v[1] * m[5] + v[2] * m[9], + v[0] * m[2] + v[1] * m[6] + v[2] * m[10] + ]; +} +var _zeroZero = [ + 0, + 0 + ]; +SpecParser.prototype._parseSpec = function _parseSpec(spec, parentContext, sizeContext) { + var id; + var target; + var transform; + var opacity; + var origin; + var align; + var size; + if (typeof spec === 'number') { + id = spec; + transform = parentContext.transform; + align = parentContext.align || _zeroZero; + if (parentContext.size && align && (align[0] || align[1])) { + var alignAdjust = [ + align[0] * parentContext.size[0], + align[1] * parentContext.size[1], + 0 + ]; + transform = Transform.thenMove(transform, _vecInContext(alignAdjust, sizeContext)); + } + this.result[id] = { + transform: transform, + opacity: parentContext.opacity, + origin: parentContext.origin || _zeroZero, + align: parentContext.align || _zeroZero, + size: parentContext.size + }; + } else if (!spec) { + return; + } else if (spec instanceof Array) { + for (var i = 0; i < spec.length; i++) { + this._parseSpec(spec[i], parentContext, sizeContext); + } + } else { + target = spec.target; + transform = parentContext.transform; + opacity = parentContext.opacity; + origin = parentContext.origin; + align = parentContext.align; + size = parentContext.size; + var nextSizeContext = sizeContext; + if (spec.opacity !== undefined) + opacity = parentContext.opacity * spec.opacity; + if (spec.transform) + transform = Transform.multiply(parentContext.transform, spec.transform); + if (spec.origin) { + origin = spec.origin; + nextSizeContext = parentContext.transform; + } + if (spec.align) + align = spec.align; + if (spec.size || spec.proportions) { + var parentSize = size; + size = [ + size[0], + size[1] + ]; + if (spec.size) { + if (spec.size[0] !== undefined) + size[0] = spec.size[0]; + if (spec.size[1] !== undefined) + size[1] = spec.size[1]; + } + if (spec.proportions) { + if (spec.proportions[0] !== undefined) + size[0] = size[0] * spec.proportions[0]; + if (spec.proportions[1] !== undefined) + size[1] = size[1] * spec.proportions[1]; + } + if (parentSize) { + if (align && (align[0] || align[1])) + transform = Transform.thenMove(transform, _vecInContext([ + align[0] * parentSize[0], + align[1] * parentSize[1], + 0 + ], sizeContext)); + if (origin && (origin[0] || origin[1])) + transform = Transform.moveThen([ + -origin[0] * size[0], + -origin[1] * size[1], + 0 + ], transform); + } + nextSizeContext = parentContext.transform; + origin = null; + align = null; + } + this._parseSpec(target, { + transform: transform, + opacity: opacity, + origin: origin, + align: align, + size: size + }, nextSizeContext); + } +}; +module.exports = SpecParser; +},{"./Transform":15}],14:[function(_dereq_,module,exports){ +var ElementOutput = _dereq_('./ElementOutput'); +function Surface(options) { + ElementOutput.call(this); + this.options = {}; + this.properties = {}; + this.attributes = {}; + this.content = ''; + this.classList = []; + this.size = null; + this._classesDirty = true; + this._stylesDirty = true; + this._attributesDirty = true; + this._sizeDirty = true; + this._contentDirty = true; + this._trueSizeCheck = true; + this._dirtyClasses = []; + if (options) + this.setOptions(options); + this._currentTarget = null; +} +Surface.prototype = Object.create(ElementOutput.prototype); +Surface.prototype.constructor = Surface; +Surface.prototype.elementType = 'div'; +Surface.prototype.elementClass = 'famous-surface'; +Surface.prototype.setAttributes = function setAttributes(attributes) { + for (var n in attributes) { + if (n === 'style') + throw new Error('Cannot set styles via "setAttributes" as it will break Famo.us. Use "setProperties" instead.'); + this.attributes[n] = attributes[n]; + } + this._attributesDirty = true; +}; +Surface.prototype.getAttributes = function getAttributes() { + return this.attributes; +}; +Surface.prototype.setProperties = function setProperties(properties) { + for (var n in properties) { + this.properties[n] = properties[n]; + } + this._stylesDirty = true; + return this; +}; +Surface.prototype.getProperties = function getProperties() { + return this.properties; +}; +Surface.prototype.addClass = function addClass(className) { + if (this.classList.indexOf(className) < 0) { + this.classList.push(className); + this._classesDirty = true; + } + return this; +}; +Surface.prototype.removeClass = function removeClass(className) { + var i = this.classList.indexOf(className); + if (i >= 0) { + this._dirtyClasses.push(this.classList.splice(i, 1)[0]); + this._classesDirty = true; + } + return this; +}; +Surface.prototype.toggleClass = function toggleClass(className) { + var i = this.classList.indexOf(className); + if (i >= 0) { + this.removeClass(className); + } else { + this.addClass(className); + } + return this; +}; +Surface.prototype.setClasses = function setClasses(classList) { + var i = 0; + var removal = []; + for (i = 0; i < this.classList.length; i++) { + if (classList.indexOf(this.classList[i]) < 0) + removal.push(this.classList[i]); + } + for (i = 0; i < removal.length; i++) + this.removeClass(removal[i]); + for (i = 0; i < classList.length; i++) + this.addClass(classList[i]); + return this; +}; +Surface.prototype.getClassList = function getClassList() { + return this.classList; +}; +Surface.prototype.setContent = function setContent(content) { + if (this.content !== content) { + this.content = content; + this._contentDirty = true; + } + return this; +}; +Surface.prototype.getContent = function getContent() { + return this.content; +}; +Surface.prototype.setOptions = function setOptions(options) { + if (options.size) + this.setSize(options.size); + if (options.classes) + this.setClasses(options.classes); + if (options.properties) + this.setProperties(options.properties); + if (options.attributes) + this.setAttributes(options.attributes); + if (options.content) + this.setContent(options.content); + return this; +}; +function _cleanupClasses(target) { + for (var i = 0; i < this._dirtyClasses.length; i++) + target.classList.remove(this._dirtyClasses[i]); + this._dirtyClasses = []; +} +function _applyStyles(target) { + for (var n in this.properties) { + target.style[n] = this.properties[n]; + } +} +function _cleanupStyles(target) { + for (var n in this.properties) { + target.style[n] = ''; + } +} +function _applyAttributes(target) { + for (var n in this.attributes) { + target.setAttribute(n, this.attributes[n]); + } +} +function _cleanupAttributes(target) { + for (var n in this.attributes) { + target.removeAttribute(n); + } +} +function _xyNotEquals(a, b) { + return a && b ? a[0] !== b[0] || a[1] !== b[1] : a !== b; +} +Surface.prototype.setup = function setup(allocator) { + var target = allocator.allocate(this.elementType); + if (this.elementClass) { + if (this.elementClass instanceof Array) { + for (var i = 0; i < this.elementClass.length; i++) { + target.classList.add(this.elementClass[i]); + } + } else { + target.classList.add(this.elementClass); + } + } + target.style.display = ''; + this.attach(target); + this._opacity = null; + this._currentTarget = target; + this._stylesDirty = true; + this._classesDirty = true; + this._attributesDirty = true; + this._sizeDirty = true; + this._contentDirty = true; + this._originDirty = true; + this._transformDirty = true; +}; +Surface.prototype.commit = function commit(context) { + if (!this._currentTarget) + this.setup(context.allocator); + var target = this._currentTarget; + var size = context.size; + if (this._classesDirty) { + _cleanupClasses.call(this, target); + var classList = this.getClassList(); + for (var i = 0; i < classList.length; i++) + target.classList.add(classList[i]); + this._classesDirty = false; + this._trueSizeCheck = true; + } + if (this._stylesDirty) { + _applyStyles.call(this, target); + this._stylesDirty = false; + this._trueSizeCheck = true; + } + if (this._attributesDirty) { + _applyAttributes.call(this, target); + this._attributesDirty = false; + this._trueSizeCheck = true; + } + if (this.size) { + var origSize = context.size; + size = [ + this.size[0], + this.size[1] + ]; + if (size[0] === undefined) + size[0] = origSize[0]; + if (size[1] === undefined) + size[1] = origSize[1]; + if (size[0] === true || size[1] === true) { + if (size[0] === true) { + if (this._trueSizeCheck || this._size[0] === 0) { + var width = target.offsetWidth; + if (this._size && this._size[0] !== width) { + this._size[0] = width; + this._sizeDirty = true; + } + size[0] = width; + } else { + if (this._size) + size[0] = this._size[0]; + } + } + if (size[1] === true) { + if (this._trueSizeCheck || this._size[1] === 0) { + var height = target.offsetHeight; + if (this._size && this._size[1] !== height) { + this._size[1] = height; + this._sizeDirty = true; + } + size[1] = height; + } else { + if (this._size) + size[1] = this._size[1]; + } + } + this._trueSizeCheck = false; + } + } + if (_xyNotEquals(this._size, size)) { + if (!this._size) + this._size = [ + 0, + 0 + ]; + this._size[0] = size[0]; + this._size[1] = size[1]; + this._sizeDirty = true; + } + if (this._sizeDirty) { + if (this._size) { + target.style.width = this.size && this.size[0] === true ? '' : this._size[0] + 'px'; + target.style.height = this.size && this.size[1] === true ? '' : this._size[1] + 'px'; + } + this._eventOutput.emit('resize'); + } + if (this._contentDirty) { + this.deploy(target); + this._eventOutput.emit('deploy'); + this._contentDirty = false; + this._trueSizeCheck = true; + } + ElementOutput.prototype.commit.call(this, context); +}; +Surface.prototype.cleanup = function cleanup(allocator) { + var i = 0; + var target = this._currentTarget; + this._eventOutput.emit('recall'); + this.recall(target); + target.style.display = 'none'; + target.style.opacity = ''; + target.style.width = ''; + target.style.height = ''; + _cleanupStyles.call(this, target); + _cleanupAttributes.call(this, target); + var classList = this.getClassList(); + _cleanupClasses.call(this, target); + for (i = 0; i < classList.length; i++) + target.classList.remove(classList[i]); + if (this.elementClass) { + if (this.elementClass instanceof Array) { + for (i = 0; i < this.elementClass.length; i++) { + target.classList.remove(this.elementClass[i]); + } + } else { + target.classList.remove(this.elementClass); + } + } + this.detach(target); + this._currentTarget = null; + allocator.deallocate(target); +}; +Surface.prototype.deploy = function deploy(target) { + var content = this.getContent(); + if (content instanceof Node) { + while (target.hasChildNodes()) + target.removeChild(target.firstChild); + target.appendChild(content); + } else + target.innerHTML = content; +}; +Surface.prototype.recall = function recall(target) { + var df = document.createDocumentFragment(); + while (target.hasChildNodes()) + df.appendChild(target.firstChild); + this.setContent(df); +}; +Surface.prototype.getSize = function getSize() { + return this._size ? this._size : this.size; +}; +Surface.prototype.setSize = function setSize(size) { + this.size = size ? [ + size[0], + size[1] + ] : null; + this._sizeDirty = true; + return this; +}; +module.exports = Surface; +},{"./ElementOutput":3}],15:[function(_dereq_,module,exports){ +var Transform = {}; +Transform.precision = 0.000001; +Transform.identity = [ + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1 +]; +Transform.multiply4x4 = function multiply4x4(a, b) { + return [ + a[0] * b[0] + a[4] * b[1] + a[8] * b[2] + a[12] * b[3], + a[1] * b[0] + a[5] * b[1] + a[9] * b[2] + a[13] * b[3], + a[2] * b[0] + a[6] * b[1] + a[10] * b[2] + a[14] * b[3], + a[3] * b[0] + a[7] * b[1] + a[11] * b[2] + a[15] * b[3], + a[0] * b[4] + a[4] * b[5] + a[8] * b[6] + a[12] * b[7], + a[1] * b[4] + a[5] * b[5] + a[9] * b[6] + a[13] * b[7], + a[2] * b[4] + a[6] * b[5] + a[10] * b[6] + a[14] * b[7], + a[3] * b[4] + a[7] * b[5] + a[11] * b[6] + a[15] * b[7], + a[0] * b[8] + a[4] * b[9] + a[8] * b[10] + a[12] * b[11], + a[1] * b[8] + a[5] * b[9] + a[9] * b[10] + a[13] * b[11], + a[2] * b[8] + a[6] * b[9] + a[10] * b[10] + a[14] * b[11], + a[3] * b[8] + a[7] * b[9] + a[11] * b[10] + a[15] * b[11], + a[0] * b[12] + a[4] * b[13] + a[8] * b[14] + a[12] * b[15], + a[1] * b[12] + a[5] * b[13] + a[9] * b[14] + a[13] * b[15], + a[2] * b[12] + a[6] * b[13] + a[10] * b[14] + a[14] * b[15], + a[3] * b[12] + a[7] * b[13] + a[11] * b[14] + a[15] * b[15] + ]; +}; +Transform.multiply = function multiply(a, b) { + return [ + a[0] * b[0] + a[4] * b[1] + a[8] * b[2], + a[1] * b[0] + a[5] * b[1] + a[9] * b[2], + a[2] * b[0] + a[6] * b[1] + a[10] * b[2], + 0, + a[0] * b[4] + a[4] * b[5] + a[8] * b[6], + a[1] * b[4] + a[5] * b[5] + a[9] * b[6], + a[2] * b[4] + a[6] * b[5] + a[10] * b[6], + 0, + a[0] * b[8] + a[4] * b[9] + a[8] * b[10], + a[1] * b[8] + a[5] * b[9] + a[9] * b[10], + a[2] * b[8] + a[6] * b[9] + a[10] * b[10], + 0, + a[0] * b[12] + a[4] * b[13] + a[8] * b[14] + a[12], + a[1] * b[12] + a[5] * b[13] + a[9] * b[14] + a[13], + a[2] * b[12] + a[6] * b[13] + a[10] * b[14] + a[14], + 1 + ]; +}; +Transform.thenMove = function thenMove(m, t) { + if (!t[2]) + t[2] = 0; + return [ + m[0], + m[1], + m[2], + 0, + m[4], + m[5], + m[6], + 0, + m[8], + m[9], + m[10], + 0, + m[12] + t[0], + m[13] + t[1], + m[14] + t[2], + 1 + ]; +}; +Transform.moveThen = function moveThen(v, m) { + if (!v[2]) + v[2] = 0; + var t0 = v[0] * m[0] + v[1] * m[4] + v[2] * m[8]; + var t1 = v[0] * m[1] + v[1] * m[5] + v[2] * m[9]; + var t2 = v[0] * m[2] + v[1] * m[6] + v[2] * m[10]; + return Transform.thenMove(m, [ + t0, + t1, + t2 + ]); +}; +Transform.translate = function translate(x, y, z) { + if (z === undefined) + z = 0; + return [ + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1, + 0, + x, + y, + z, + 1 + ]; +}; +Transform.thenScale = function thenScale(m, s) { + return [ + s[0] * m[0], + s[1] * m[1], + s[2] * m[2], + 0, + s[0] * m[4], + s[1] * m[5], + s[2] * m[6], + 0, + s[0] * m[8], + s[1] * m[9], + s[2] * m[10], + 0, + s[0] * m[12], + s[1] * m[13], + s[2] * m[14], + 1 + ]; +}; +Transform.scale = function scale(x, y, z) { + if (z === undefined) + z = 1; + if (y === undefined) + y = x; + return [ + x, + 0, + 0, + 0, + 0, + y, + 0, + 0, + 0, + 0, + z, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.rotateX = function rotateX(theta) { + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + return [ + 1, + 0, + 0, + 0, + 0, + cosTheta, + sinTheta, + 0, + 0, + -sinTheta, + cosTheta, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.rotateY = function rotateY(theta) { + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + return [ + cosTheta, + 0, + -sinTheta, + 0, + 0, + 1, + 0, + 0, + sinTheta, + 0, + cosTheta, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.rotateZ = function rotateZ(theta) { + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + return [ + cosTheta, + sinTheta, + 0, + 0, + -sinTheta, + cosTheta, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.rotate = function rotate(phi, theta, psi) { + var cosPhi = Math.cos(phi); + var sinPhi = Math.sin(phi); + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + var cosPsi = Math.cos(psi); + var sinPsi = Math.sin(psi); + var result = [ + cosTheta * cosPsi, + cosPhi * sinPsi + sinPhi * sinTheta * cosPsi, + sinPhi * sinPsi - cosPhi * sinTheta * cosPsi, + 0, + -cosTheta * sinPsi, + cosPhi * cosPsi - sinPhi * sinTheta * sinPsi, + sinPhi * cosPsi + cosPhi * sinTheta * sinPsi, + 0, + sinTheta, + -sinPhi * cosTheta, + cosPhi * cosTheta, + 0, + 0, + 0, + 0, + 1 + ]; + return result; +}; +Transform.rotateAxis = function rotateAxis(v, theta) { + var sinTheta = Math.sin(theta); + var cosTheta = Math.cos(theta); + var verTheta = 1 - cosTheta; + var xxV = v[0] * v[0] * verTheta; + var xyV = v[0] * v[1] * verTheta; + var xzV = v[0] * v[2] * verTheta; + var yyV = v[1] * v[1] * verTheta; + var yzV = v[1] * v[2] * verTheta; + var zzV = v[2] * v[2] * verTheta; + var xs = v[0] * sinTheta; + var ys = v[1] * sinTheta; + var zs = v[2] * sinTheta; + var result = [ + xxV + cosTheta, + xyV + zs, + xzV - ys, + 0, + xyV - zs, + yyV + cosTheta, + yzV + xs, + 0, + xzV + ys, + yzV - xs, + zzV + cosTheta, + 0, + 0, + 0, + 0, + 1 + ]; + return result; +}; +Transform.aboutOrigin = function aboutOrigin(v, m) { + var t0 = v[0] - (v[0] * m[0] + v[1] * m[4] + v[2] * m[8]); + var t1 = v[1] - (v[0] * m[1] + v[1] * m[5] + v[2] * m[9]); + var t2 = v[2] - (v[0] * m[2] + v[1] * m[6] + v[2] * m[10]); + return Transform.thenMove(m, [ + t0, + t1, + t2 + ]); +}; +Transform.skew = function skew(phi, theta, psi) { + return [ + 1, + Math.tan(theta), + 0, + 0, + Math.tan(psi), + 1, + 0, + 0, + 0, + Math.tan(phi), + 1, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.skewX = function skewX(angle) { + return [ + 1, + 0, + 0, + 0, + Math.tan(angle), + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.skewY = function skewY(angle) { + return [ + 1, + Math.tan(angle), + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1 + ]; +}; +Transform.perspective = function perspective(focusZ) { + return [ + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1, + -1 / focusZ, + 0, + 0, + 0, + 1 + ]; +}; +Transform.getTranslate = function getTranslate(m) { + return [ + m[12], + m[13], + m[14] + ]; +}; +Transform.inverse = function inverse(m) { + var c0 = m[5] * m[10] - m[6] * m[9]; + var c1 = m[4] * m[10] - m[6] * m[8]; + var c2 = m[4] * m[9] - m[5] * m[8]; + var c4 = m[1] * m[10] - m[2] * m[9]; + var c5 = m[0] * m[10] - m[2] * m[8]; + var c6 = m[0] * m[9] - m[1] * m[8]; + var c8 = m[1] * m[6] - m[2] * m[5]; + var c9 = m[0] * m[6] - m[2] * m[4]; + var c10 = m[0] * m[5] - m[1] * m[4]; + var detM = m[0] * c0 - m[1] * c1 + m[2] * c2; + var invD = 1 / detM; + var result = [ + invD * c0, + -invD * c4, + invD * c8, + 0, + -invD * c1, + invD * c5, + -invD * c9, + 0, + invD * c2, + -invD * c6, + invD * c10, + 0, + 0, + 0, + 0, + 1 + ]; + result[12] = -m[12] * result[0] - m[13] * result[4] - m[14] * result[8]; + result[13] = -m[12] * result[1] - m[13] * result[5] - m[14] * result[9]; + result[14] = -m[12] * result[2] - m[13] * result[6] - m[14] * result[10]; + return result; +}; +Transform.transpose = function transpose(m) { + return [ + m[0], + m[4], + m[8], + m[12], + m[1], + m[5], + m[9], + m[13], + m[2], + m[6], + m[10], + m[14], + m[3], + m[7], + m[11], + m[15] + ]; +}; +function _normSquared(v) { + return v.length === 2 ? v[0] * v[0] + v[1] * v[1] : v[0] * v[0] + v[1] * v[1] + v[2] * v[2]; +} +function _norm(v) { + return Math.sqrt(_normSquared(v)); +} +function _sign(n) { + return n < 0 ? -1 : 1; +} +Transform.interpret = function interpret(M) { + var x = [ + M[0], + M[1], + M[2] + ]; + var sgn = _sign(x[0]); + var xNorm = _norm(x); + var v = [ + x[0] + sgn * xNorm, + x[1], + x[2] + ]; + var mult = 2 / _normSquared(v); + if (mult >= Infinity) { + return { + translate: Transform.getTranslate(M), + rotate: [ + 0, + 0, + 0 + ], + scale: [ + 0, + 0, + 0 + ], + skew: [ + 0, + 0, + 0 + ] + }; + } + var Q1 = [ + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 1 + ]; + Q1[0] = 1 - mult * v[0] * v[0]; + Q1[5] = 1 - mult * v[1] * v[1]; + Q1[10] = 1 - mult * v[2] * v[2]; + Q1[1] = -mult * v[0] * v[1]; + Q1[2] = -mult * v[0] * v[2]; + Q1[6] = -mult * v[1] * v[2]; + Q1[4] = Q1[1]; + Q1[8] = Q1[2]; + Q1[9] = Q1[6]; + var MQ1 = Transform.multiply(Q1, M); + var x2 = [ + MQ1[5], + MQ1[6] + ]; + var sgn2 = _sign(x2[0]); + var x2Norm = _norm(x2); + var v2 = [ + x2[0] + sgn2 * x2Norm, + x2[1] + ]; + var mult2 = 2 / _normSquared(v2); + var Q2 = [ + 1, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 0, + 1 + ]; + Q2[5] = 1 - mult2 * v2[0] * v2[0]; + Q2[10] = 1 - mult2 * v2[1] * v2[1]; + Q2[6] = -mult2 * v2[0] * v2[1]; + Q2[9] = Q2[6]; + var Q = Transform.multiply(Q2, Q1); + var R = Transform.multiply(Q, M); + var remover = Transform.scale(R[0] < 0 ? -1 : 1, R[5] < 0 ? -1 : 1, R[10] < 0 ? -1 : 1); + R = Transform.multiply(R, remover); + Q = Transform.multiply(remover, Q); + var result = {}; + result.translate = Transform.getTranslate(M); + result.rotate = [ + Math.atan2(-Q[6], Q[10]), + Math.asin(Q[2]), + Math.atan2(-Q[1], Q[0]) + ]; + if (!result.rotate[0]) { + result.rotate[0] = 0; + result.rotate[2] = Math.atan2(Q[4], Q[5]); + } + result.scale = [ + R[0], + R[5], + R[10] + ]; + result.skew = [ + Math.atan2(R[9], result.scale[2]), + Math.atan2(R[8], result.scale[2]), + Math.atan2(R[4], result.scale[0]) + ]; + if (Math.abs(result.rotate[0]) + Math.abs(result.rotate[2]) > 1.5 * Math.PI) { + result.rotate[1] = Math.PI - result.rotate[1]; + if (result.rotate[1] > Math.PI) + result.rotate[1] -= 2 * Math.PI; + if (result.rotate[1] < -Math.PI) + result.rotate[1] += 2 * Math.PI; + if (result.rotate[0] < 0) + result.rotate[0] += Math.PI; + else + result.rotate[0] -= Math.PI; + if (result.rotate[2] < 0) + result.rotate[2] += Math.PI; + else + result.rotate[2] -= Math.PI; + } + return result; +}; +Transform.average = function average(M1, M2, t) { + t = t === undefined ? 0.5 : t; + var specM1 = Transform.interpret(M1); + var specM2 = Transform.interpret(M2); + var specAvg = { + translate: [ + 0, + 0, + 0 + ], + rotate: [ + 0, + 0, + 0 + ], + scale: [ + 0, + 0, + 0 + ], + skew: [ + 0, + 0, + 0 + ] + }; + for (var i = 0; i < 3; i++) { + specAvg.translate[i] = (1 - t) * specM1.translate[i] + t * specM2.translate[i]; + specAvg.rotate[i] = (1 - t) * specM1.rotate[i] + t * specM2.rotate[i]; + specAvg.scale[i] = (1 - t) * specM1.scale[i] + t * specM2.scale[i]; + specAvg.skew[i] = (1 - t) * specM1.skew[i] + t * specM2.skew[i]; + } + return Transform.build(specAvg); +}; +Transform.build = function build(spec) { + var scaleMatrix = Transform.scale(spec.scale[0], spec.scale[1], spec.scale[2]); + var skewMatrix = Transform.skew(spec.skew[0], spec.skew[1], spec.skew[2]); + var rotateMatrix = Transform.rotate(spec.rotate[0], spec.rotate[1], spec.rotate[2]); + return Transform.thenMove(Transform.multiply(Transform.multiply(rotateMatrix, skewMatrix), scaleMatrix), spec.translate); +}; +Transform.equals = function equals(a, b) { + return !Transform.notEquals(a, b); +}; +Transform.notEquals = function notEquals(a, b) { + if (a === b) + return false; + return !(a && b) || a[12] !== b[12] || a[13] !== b[13] || a[14] !== b[14] || a[0] !== b[0] || a[1] !== b[1] || a[2] !== b[2] || a[4] !== b[4] || a[5] !== b[5] || a[6] !== b[6] || a[8] !== b[8] || a[9] !== b[9] || a[10] !== b[10]; +}; +Transform.normalizeRotation = function normalizeRotation(rotation) { + var result = rotation.slice(0); + if (result[0] === Math.PI * 0.5 || result[0] === -Math.PI * 0.5) { + result[0] = -result[0]; + result[1] = Math.PI - result[1]; + result[2] -= Math.PI; + } + if (result[0] > Math.PI * 0.5) { + result[0] = result[0] - Math.PI; + result[1] = Math.PI - result[1]; + result[2] -= Math.PI; + } + if (result[0] < -Math.PI * 0.5) { + result[0] = result[0] + Math.PI; + result[1] = -Math.PI - result[1]; + result[2] -= Math.PI; + } + while (result[1] < -Math.PI) + result[1] += 2 * Math.PI; + while (result[1] >= Math.PI) + result[1] -= 2 * Math.PI; + while (result[2] < -Math.PI) + result[2] += 2 * Math.PI; + while (result[2] >= Math.PI) + result[2] -= 2 * Math.PI; + return result; +}; +Transform.inFront = [ + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0.001, + 1 +]; +Transform.behind = [ + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + 0, + 1, + 0, + 0, + 0, + -0.001, + 1 +]; +module.exports = Transform; +},{}],16:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('./EventHandler'); +var OptionsManager = _dereq_('./OptionsManager'); +var RenderNode = _dereq_('./RenderNode'); +var Utility = _dereq_('../utilities/Utility'); +function View(options) { + this._node = new RenderNode(); + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this.options = Utility.clone(this.constructor.DEFAULT_OPTIONS || View.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); +} +View.DEFAULT_OPTIONS = {}; +View.prototype.getOptions = function getOptions(key) { + return this._optionsManager.getOptions(key); +}; +View.prototype.setOptions = function setOptions(options) { + this._optionsManager.patch(options); +}; +View.prototype.add = function add() { + return this._node.add.apply(this._node, arguments); +}; +View.prototype._add = View.prototype.add; +View.prototype.render = function render() { + return this._node.render(); +}; +View.prototype.getSize = function getSize() { + if (this._node && this._node.getSize) { + return this._node.getSize.apply(this._node, arguments) || this.options.size; + } else + return this.options.size; +}; +module.exports = View; +},{"../utilities/Utility":95,"./EventHandler":7,"./OptionsManager":10,"./RenderNode":11}],17:[function(_dereq_,module,exports){ +function ViewSequence(options) { + if (!options) + options = []; + if (options instanceof Array) + options = { array: options }; + this._ = null; + this.index = options.index || 0; + if (options.array) + this._ = new this.constructor.Backing(options.array); + else if (options._) + this._ = options._; + if (this.index === this._.firstIndex) + this._.firstNode = this; + if (this.index === this._.firstIndex + this._.array.length - 1) + this._.lastNode = this; + if (options.loop !== undefined) + this._.loop = options.loop; + if (options.trackSize !== undefined) + this._.trackSize = options.trackSize; + this._previousNode = null; + this._nextNode = null; +} +ViewSequence.Backing = function Backing(array) { + this.array = array; + this.firstIndex = 0; + this.loop = false; + this.firstNode = null; + this.lastNode = null; + this.cumulativeSizes = [[ + 0, + 0 + ]]; + this.sizeDirty = true; + this.trackSize = false; +}; +ViewSequence.Backing.prototype.getValue = function getValue(i) { + var _i = i - this.firstIndex; + if (_i < 0 || _i >= this.array.length) + return null; + return this.array[_i]; +}; +ViewSequence.Backing.prototype.setValue = function setValue(i, value) { + this.array[i - this.firstIndex] = value; +}; +ViewSequence.Backing.prototype.getSize = function getSize(index) { + return this.cumulativeSizes[index]; +}; +ViewSequence.Backing.prototype.calculateSize = function calculateSize(index) { + index = index || this.array.length; + var size = [ + 0, + 0 + ]; + for (var i = 0; i < index; i++) { + var nodeSize = this.array[i].getSize(); + if (!nodeSize) + return undefined; + if (size[0] !== undefined) { + if (nodeSize[0] === undefined) + size[0] = undefined; + else + size[0] += nodeSize[0]; + } + if (size[1] !== undefined) { + if (nodeSize[1] === undefined) + size[1] = undefined; + else + size[1] += nodeSize[1]; + } + this.cumulativeSizes[i + 1] = size.slice(); + } + this.sizeDirty = false; + return size; +}; +ViewSequence.Backing.prototype.reindex = function reindex(start, removeCount, insertCount) { + if (!this.array[0]) + return; + var i = 0; + var index = this.firstIndex; + var indexShiftAmount = insertCount - removeCount; + var node = this.firstNode; + while (index < start - 1) { + node = node.getNext(); + index++; + } + var spliceStartNode = node; + for (i = 0; i < removeCount; i++) { + node = node.getNext(); + if (node) + node._previousNode = spliceStartNode; + } + var spliceResumeNode = node ? node.getNext() : null; + spliceStartNode._nextNode = null; + node = spliceStartNode; + for (i = 0; i < insertCount; i++) + node = node.getNext(); + index += insertCount; + if (node !== spliceResumeNode) { + node._nextNode = spliceResumeNode; + if (spliceResumeNode) + spliceResumeNode._previousNode = node; + } + if (spliceResumeNode) { + node = spliceResumeNode; + index++; + while (node && index < this.array.length + this.firstIndex) { + if (node._nextNode) + node.index += indexShiftAmount; + else + node.index = index; + node = node.getNext(); + index++; + } + } + if (this.trackSize) + this.sizeDirty = true; +}; +ViewSequence.prototype.getPrevious = function getPrevious() { + var len = this._.array.length; + if (!len) { + this._previousNode = null; + } else if (this.index === this._.firstIndex) { + if (this._.loop) { + this._previousNode = this._.lastNode || new this.constructor({ + _: this._, + index: this._.firstIndex + len - 1 + }); + this._previousNode._nextNode = this; + } else { + this._previousNode = null; + } + } else if (!this._previousNode) { + this._previousNode = new this.constructor({ + _: this._, + index: this.index - 1 + }); + this._previousNode._nextNode = this; + } + return this._previousNode; +}; +ViewSequence.prototype.getNext = function getNext() { + var len = this._.array.length; + if (!len) { + this._nextNode = null; + } else if (this.index === this._.firstIndex + len - 1) { + if (this._.loop) { + this._nextNode = this._.firstNode || new this.constructor({ + _: this._, + index: this._.firstIndex + }); + this._nextNode._previousNode = this; + } else { + this._nextNode = null; + } + } else if (!this._nextNode) { + this._nextNode = new this.constructor({ + _: this._, + index: this.index + 1 + }); + this._nextNode._previousNode = this; + } + return this._nextNode; +}; +ViewSequence.prototype.indexOf = function indexOf(item) { + return this._.array.indexOf(item); +}; +ViewSequence.prototype.getIndex = function getIndex() { + return this.index; +}; +ViewSequence.prototype.toString = function toString() { + return '' + this.index; +}; +ViewSequence.prototype.unshift = function unshift(value) { + this._.array.unshift.apply(this._.array, arguments); + this._.firstIndex -= arguments.length; + if (this._.trackSize) + this._.sizeDirty = true; +}; +ViewSequence.prototype.push = function push(value) { + this._.array.push.apply(this._.array, arguments); + if (this._.trackSize) + this._.sizeDirty = true; +}; +ViewSequence.prototype.splice = function splice(index, howMany) { + var values = Array.prototype.slice.call(arguments, 2); + this._.array.splice.apply(this._.array, [ + index - this._.firstIndex, + howMany + ].concat(values)); + this._.reindex(index, howMany, values.length); +}; +ViewSequence.prototype.swap = function swap(other) { + var otherValue = other.get(); + var myValue = this.get(); + this._.setValue(this.index, otherValue); + this._.setValue(other.index, myValue); + var myPrevious = this._previousNode; + var myNext = this._nextNode; + var myIndex = this.index; + var otherPrevious = other._previousNode; + var otherNext = other._nextNode; + var otherIndex = other.index; + this.index = otherIndex; + this._previousNode = otherPrevious === this ? other : otherPrevious; + if (this._previousNode) + this._previousNode._nextNode = this; + this._nextNode = otherNext === this ? other : otherNext; + if (this._nextNode) + this._nextNode._previousNode = this; + other.index = myIndex; + other._previousNode = myPrevious === other ? this : myPrevious; + if (other._previousNode) + other._previousNode._nextNode = other; + other._nextNode = myNext === other ? this : myNext; + if (other._nextNode) + other._nextNode._previousNode = other; + if (this.index === this._.firstIndex) + this._.firstNode = this; + else if (this.index === this._.firstIndex + this._.array.length - 1) + this._.lastNode = this; + if (other.index === this._.firstIndex) + this._.firstNode = other; + else if (other.index === this._.firstIndex + this._.array.length - 1) + this._.lastNode = other; + if (this._.trackSize) + this._.sizeDirty = true; +}; +ViewSequence.prototype.get = function get() { + return this._.getValue(this.index); +}; +ViewSequence.prototype.getSize = function getSize() { + var target = this.get(); + return target ? target.getSize() : null; +}; +ViewSequence.prototype.render = function render() { + if (this._.trackSize && this._.sizeDirty) + this._.calculateSize(); + var target = this.get(); + return target ? target.render.apply(target, arguments) : null; +}; +module.exports = ViewSequence; +},{}],18:[function(_dereq_,module,exports){ +module.exports = { + Context: _dereq_('./Context'), + ElementAllocator: _dereq_('./ElementAllocator'), + ElementOutput: _dereq_('./ElementOutput'), + Engine: _dereq_('./Engine'), + Entity: _dereq_('./Entity'), + EventEmitter: _dereq_('./EventEmitter'), + EventHandler: _dereq_('./EventHandler'), + Group: _dereq_('./Group'), + Modifier: _dereq_('./Modifier'), + OptionsManager: _dereq_('./OptionsManager'), + RenderNode: _dereq_('./RenderNode'), + Scene: _dereq_('./Scene'), + SpecParser: _dereq_('./SpecParser'), + Surface: _dereq_('./Surface'), + Transform: _dereq_('./Transform'), + View: _dereq_('./View'), + ViewSequence: _dereq_('./ViewSequence') +}; + +},{"./Context":1,"./ElementAllocator":2,"./ElementOutput":3,"./Engine":4,"./Entity":5,"./EventEmitter":6,"./EventHandler":7,"./Group":8,"./Modifier":9,"./OptionsManager":10,"./RenderNode":11,"./Scene":12,"./SpecParser":13,"./Surface":14,"./Transform":15,"./View":16,"./ViewSequence":17}],19:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +function EventArbiter(startMode) { + this.dispatchers = {}; + this.currMode = undefined; + this.setMode(startMode); +} +EventArbiter.prototype.setMode = function setMode(mode) { + if (mode !== this.currMode) { + var startMode = this.currMode; + if (this.dispatchers[this.currMode]) + this.dispatchers[this.currMode].trigger('unpipe'); + this.currMode = mode; + if (this.dispatchers[mode]) + this.dispatchers[mode].emit('pipe'); + this.emit('change', { + from: startMode, + to: mode + }); + } +}; +EventArbiter.prototype.forMode = function forMode(mode) { + if (!this.dispatchers[mode]) + this.dispatchers[mode] = new EventHandler(); + return this.dispatchers[mode]; +}; +EventArbiter.prototype.emit = function emit(eventType, event) { + if (this.currMode === undefined) + return false; + if (!event) + event = {}; + var dispatcher = this.dispatchers[this.currMode]; + if (dispatcher) + return dispatcher.trigger(eventType, event); +}; +module.exports = EventArbiter; +},{"../core/EventHandler":7}],20:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +function EventFilter(condition) { + EventHandler.call(this); + this._condition = condition; +} +EventFilter.prototype = Object.create(EventHandler.prototype); +EventFilter.prototype.constructor = EventFilter; +EventFilter.prototype.emit = function emit(type, data) { + if (this._condition(type, data)) + return EventHandler.prototype.emit.apply(this, arguments); +}; +EventFilter.prototype.trigger = EventFilter.prototype.emit; +module.exports = EventFilter; +},{"../core/EventHandler":7}],21:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +function EventMapper(mappingFunction) { + EventHandler.call(this); + this._mappingFunction = mappingFunction; +} +EventMapper.prototype = Object.create(EventHandler.prototype); +EventMapper.prototype.constructor = EventMapper; +EventMapper.prototype.subscribe = null; +EventMapper.prototype.unsubscribe = null; +EventMapper.prototype.emit = function emit(type, data) { + var target = this._mappingFunction.apply(this, arguments); + if (target && target.emit instanceof Function) + target.emit(type, data); +}; +EventMapper.prototype.trigger = EventMapper.prototype.emit; +module.exports = EventMapper; +},{"../core/EventHandler":7}],22:[function(_dereq_,module,exports){ +module.exports = { + EventArbiter: _dereq_('./EventArbiter'), + EventFilter: _dereq_('./EventFilter'), + EventMapper: _dereq_('./EventMapper') +}; + +},{"./EventArbiter":19,"./EventFilter":20,"./EventMapper":21}],23:[function(_dereq_,module,exports){ +module.exports = { + core: _dereq_('./core'), + events: _dereq_('./events'), + inputs: _dereq_('./inputs'), + math: _dereq_('./math'), + modifiers: _dereq_('./modifiers'), + physics: _dereq_('./physics'), + surfaces: _dereq_('./surfaces'), + transitions: _dereq_('./transitions'), + utilities: _dereq_('./utilities'), + views: _dereq_('./views'), + widgets: _dereq_('./widgets') +}; + +},{"./core":18,"./events":22,"./inputs":36,"./math":42,"./modifiers":47,"./physics":71,"./surfaces":82,"./transitions":92,"./utilities":96,"./views":112,"./widgets":117}],24:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +var Transitionable = _dereq_('../transitions/Transitionable'); +function Accumulator(value, eventName) { + if (eventName === undefined) + eventName = 'update'; + this._state = value && value.get && value.set ? value : new Transitionable(value || 0); + this._eventInput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + this._eventInput.on(eventName, _handleUpdate.bind(this)); +} +function _handleUpdate(data) { + var delta = data.delta; + var state = this.get(); + if (delta.constructor === state.constructor) { + var newState = delta instanceof Array ? [ + state[0] + delta[0], + state[1] + delta[1] + ] : state + delta; + this.set(newState); + } +} +Accumulator.prototype.get = function get() { + return this._state.get(); +}; +Accumulator.prototype.set = function set(value) { + this._state.set(value); +}; +module.exports = Accumulator; +},{"../core/EventHandler":7,"../transitions/Transitionable":88}],25:[function(_dereq_,module,exports){ +var hasTouch = 'ontouchstart' in window; +function kill(type) { + window.addEventListener(type, function (event) { + event.stopPropagation(); + return false; + }, true); +} +if (hasTouch) { + kill('mousedown'); + kill('mousemove'); + kill('mouseup'); + kill('mouseleave'); +} +},{}],26:[function(_dereq_,module,exports){ +(function () { + if (!window.CustomEvent) + return; + var clickThreshold = 300; + var clickWindow = 500; + var potentialClicks = {}; + var recentlyDispatched = {}; + var _now = Date.now; + window.addEventListener('touchstart', function (event) { + var timestamp = _now(); + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + potentialClicks[touch.identifier] = timestamp; + } + }); + window.addEventListener('touchmove', function (event) { + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + delete potentialClicks[touch.identifier]; + } + }); + window.addEventListener('touchend', function (event) { + var currTime = _now(); + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var startTime = potentialClicks[touch.identifier]; + if (startTime && currTime - startTime < clickThreshold) { + var clickEvt = new window.CustomEvent('click', { + 'bubbles': true, + 'detail': touch + }); + recentlyDispatched[currTime] = event; + event.target.dispatchEvent(clickEvt); + } + delete potentialClicks[touch.identifier]; + } + }); + window.addEventListener('click', function (event) { + var currTime = _now(); + for (var i in recentlyDispatched) { + var previousEvent = recentlyDispatched[i]; + if (currTime - i < clickWindow) { + if (event instanceof window.MouseEvent && event.target === previousEvent.target) + event.stopPropagation(); + } else + delete recentlyDispatched[i]; + } + }, true); +}()); +},{}],27:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +function GenericSync(syncs, options) { + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this._syncs = {}; + if (syncs) + this.addSync(syncs); + if (options) + this.setOptions(options); +} +GenericSync.DIRECTION_X = 0; +GenericSync.DIRECTION_Y = 1; +GenericSync.DIRECTION_Z = 2; +var registry = {}; +GenericSync.register = function register(syncObject) { + for (var key in syncObject) { + if (registry[key]) { + if (registry[key] !== syncObject[key]) + throw new Error('Conflicting sync classes for key: ' + key); + } else + registry[key] = syncObject[key]; + } +}; +GenericSync.prototype.setOptions = function (options) { + for (var key in this._syncs) { + this._syncs[key].setOptions(options); + } +}; +GenericSync.prototype.pipeSync = function pipeToSync(key) { + var sync = this._syncs[key]; + this._eventInput.pipe(sync); + sync.pipe(this._eventOutput); +}; +GenericSync.prototype.unpipeSync = function unpipeFromSync(key) { + var sync = this._syncs[key]; + this._eventInput.unpipe(sync); + sync.unpipe(this._eventOutput); +}; +function _addSingleSync(key, options) { + if (!registry[key]) + return; + this._syncs[key] = new registry[key](options); + this.pipeSync(key); +} +GenericSync.prototype.addSync = function addSync(syncs) { + if (syncs instanceof Array) + for (var i = 0; i < syncs.length; i++) + _addSingleSync.call(this, syncs[i]); + else if (syncs instanceof Object) + for (var key in syncs) + _addSingleSync.call(this, key, syncs[key]); +}; +module.exports = GenericSync; +},{"../core/EventHandler":7}],28:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function MouseSync(options) { + this.options = Object.create(MouseSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this._eventInput.on('mousedown', _handleStart.bind(this)); + this._eventInput.on('mousemove', _handleMove.bind(this)); + this._eventInput.on('mouseup', _handleEnd.bind(this)); + if (this.options.propogate) + this._eventInput.on('mouseleave', _handleLeave.bind(this)); + else + this._eventInput.on('mouseleave', _handleEnd.bind(this)); + if (this.options.clickThreshold) { + window.addEventListener('click', function (event) { + if (Math.sqrt(Math.pow(this._displacement[0], 2) + Math.pow(this._displacement[1], 2)) > this.options.clickThreshold) { + event.stopPropagation(); + } + }.bind(this), true); + } + this._payload = { + delta: null, + position: null, + velocity: null, + clientX: 0, + clientY: 0, + offsetX: 0, + offsetY: 0 + }; + this._positionHistory = []; + this._position = null; + this._prevCoord = undefined; + this._prevTime = undefined; + this._down = false; + this._moved = false; + this._displacement = [ + 0, + 0 + ]; + this._documentActive = false; +} +MouseSync.DEFAULT_OPTIONS = { + clickThreshold: undefined, + direction: undefined, + rails: false, + scale: 1, + propogate: true, + velocitySampleLength: 10, + preventDefault: true +}; +MouseSync.DIRECTION_X = 0; +MouseSync.DIRECTION_Y = 1; +var MINIMUM_TICK_TIME = 8; +function _handleStart(event) { + var delta; + var velocity; + if (this.options.preventDefault) + event.preventDefault(); + var x = event.clientX; + var y = event.clientY; + this._prevCoord = [ + x, + y + ]; + this._prevTime = Date.now(); + this._down = true; + this._move = false; + if (this.options.direction !== undefined) { + this._position = 0; + delta = 0; + velocity = 0; + } else { + this._position = [ + 0, + 0 + ]; + delta = [ + 0, + 0 + ]; + velocity = [ + 0, + 0 + ]; + } + if (this.options.clickThreshold) { + this._displacement = [ + 0, + 0 + ]; + } + var payload = this._payload; + payload.delta = delta; + payload.position = this._position; + payload.velocity = velocity; + payload.clientX = x; + payload.clientY = y; + payload.offsetX = event.offsetX; + payload.offsetY = event.offsetY; + this._positionHistory.push({ + position: payload.position.slice ? payload.position.slice(0) : payload.position, + time: this._prevTime + }); + this._eventOutput.emit('start', payload); + this._documentActive = false; +} +function _handleMove(event) { + if (!this._prevCoord) + return; + var prevCoord = this._prevCoord; + var prevTime = this._prevTime; + var x = event.clientX; + var y = event.clientY; + var currTime = Date.now(); + var diffX = x - prevCoord[0]; + var diffY = y - prevCoord[1]; + if (this.options.rails) { + if (Math.abs(diffX) > Math.abs(diffY)) + diffY = 0; + else + diffX = 0; + } + var diffTime = Math.max(currTime - this._positionHistory[0].time, MINIMUM_TICK_TIME); + var scale = this.options.scale; + var nextVel; + var nextDelta; + if (this.options.direction === MouseSync.DIRECTION_X) { + nextDelta = scale * diffX; + this._position += nextDelta; + nextVel = scale * (this._position - this._positionHistory[0].position) / diffTime; + } else if (this.options.direction === MouseSync.DIRECTION_Y) { + nextDelta = scale * diffY; + this._position += nextDelta; + nextVel = scale * (this._position - this._positionHistory[0].position) / diffTime; + } else { + nextDelta = [ + scale * diffX, + scale * diffY + ]; + nextVel = [ + scale * (this._position[0] - this._positionHistory[0].position[0]) / diffTime, + scale * (this._position[1] - this._positionHistory[0].position[1]) / diffTime + ]; + this._position[0] += nextDelta[0]; + this._position[1] += nextDelta[1]; + } + if (this.options.clickThreshold !== false) { + this._displacement[0] += diffX; + this._displacement[1] += diffY; + } + var payload = this._payload; + payload.delta = nextDelta; + payload.position = this._position; + payload.velocity = nextVel; + payload.clientX = x; + payload.clientY = y; + payload.offsetX = event.offsetX; + payload.offsetY = event.offsetY; + if (this._positionHistory.length === this.options.velocitySampleLength) { + this._positionHistory.shift(); + } + this._positionHistory.push({ + position: payload.position.slice ? payload.position.slice(0) : payload.position, + time: currTime + }); + this._eventOutput.emit('update', payload); + this._prevCoord = [ + x, + y + ]; + this._prevTime = currTime; + this._move = true; +} +function _handleEnd(event) { + if (!this._down) + return; + this._eventOutput.emit('end', this._payload); + this._prevCoord = undefined; + this._prevTime = undefined; + this._down = false; + this._move = false; + this._positionHistory = []; +} +function _handleLeave(event) { + if (!this._down || !this._move) + return; + if (!this._documentActive) { + var boundMove = _handleMove.bind(this); + var boundEnd = function (event) { + _handleEnd.call(this, event); + document.removeEventListener('mousemove', boundMove); + document.removeEventListener('mouseup', boundEnd); + }.bind(this, event); + document.addEventListener('mousemove', boundMove); + document.addEventListener('mouseup', boundEnd); + this._documentActive = true; + } +} +MouseSync.prototype.getOptions = function getOptions() { + return this.options; +}; +MouseSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +module.exports = MouseSync; +},{"../core/EventHandler":7,"../core/OptionsManager":10}],29:[function(_dereq_,module,exports){ +var TwoFingerSync = _dereq_('./TwoFingerSync'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function PinchSync(options) { + TwoFingerSync.call(this); + this.options = Object.create(PinchSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._displacement = 0; + this._previousDistance = 0; +} +PinchSync.prototype = Object.create(TwoFingerSync.prototype); +PinchSync.prototype.constructor = PinchSync; +PinchSync.DEFAULT_OPTIONS = { scale: 1 }; +PinchSync.prototype._startUpdate = function _startUpdate(event) { + this._previousDistance = TwoFingerSync.calculateDistance(this.posA, this.posB); + this._displacement = 0; + this._eventOutput.emit('start', { + count: event.touches.length, + touches: [ + this.touchAId, + this.touchBId + ], + distance: this._dist, + center: TwoFingerSync.calculateCenter(this.posA, this.posB) + }); +}; +PinchSync.prototype._moveUpdate = function _moveUpdate(diffTime) { + var currDist = TwoFingerSync.calculateDistance(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + var scale = this.options.scale; + var delta = scale * (currDist - this._previousDistance); + var velocity = delta / diffTime; + this._previousDistance = currDist; + this._displacement += delta; + this._eventOutput.emit('update', { + delta: delta, + velocity: velocity, + distance: currDist, + displacement: this._displacement, + center: center, + touches: [ + this.touchAId, + this.touchBId + ] + }); +}; +PinchSync.prototype.getOptions = function getOptions() { + return this.options; +}; +PinchSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +module.exports = PinchSync; +},{"../core/OptionsManager":10,"./TwoFingerSync":35}],30:[function(_dereq_,module,exports){ +var TwoFingerSync = _dereq_('./TwoFingerSync'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function RotateSync(options) { + TwoFingerSync.call(this); + this.options = Object.create(RotateSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._angle = 0; + this._previousAngle = 0; +} +RotateSync.prototype = Object.create(TwoFingerSync.prototype); +RotateSync.prototype.constructor = RotateSync; +RotateSync.DEFAULT_OPTIONS = { scale: 1 }; +RotateSync.prototype._startUpdate = function _startUpdate(event) { + this._angle = 0; + this._previousAngle = TwoFingerSync.calculateAngle(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + this._eventOutput.emit('start', { + count: event.touches.length, + angle: this._angle, + center: center, + touches: [ + this.touchAId, + this.touchBId + ] + }); +}; +RotateSync.prototype._moveUpdate = function _moveUpdate(diffTime) { + var scale = this.options.scale; + var currAngle = TwoFingerSync.calculateAngle(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + var diffTheta = scale * (currAngle - this._previousAngle); + var velTheta = diffTheta / diffTime; + this._angle += diffTheta; + this._eventOutput.emit('update', { + delta: diffTheta, + velocity: velTheta, + angle: this._angle, + center: center, + touches: [ + this.touchAId, + this.touchBId + ] + }); + this._previousAngle = currAngle; +}; +RotateSync.prototype.getOptions = function getOptions() { + return this.options; +}; +RotateSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +module.exports = RotateSync; +},{"../core/OptionsManager":10,"./TwoFingerSync":35}],31:[function(_dereq_,module,exports){ +var TwoFingerSync = _dereq_('./TwoFingerSync'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function ScaleSync(options) { + TwoFingerSync.call(this); + this.options = Object.create(ScaleSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._scaleFactor = 1; + this._startDist = 0; + this._eventInput.on('pipe', _reset.bind(this)); +} +ScaleSync.prototype = Object.create(TwoFingerSync.prototype); +ScaleSync.prototype.constructor = ScaleSync; +ScaleSync.DEFAULT_OPTIONS = { scale: 1 }; +function _reset() { + this.touchAId = undefined; + this.touchBId = undefined; +} +ScaleSync.prototype._startUpdate = function _startUpdate(event) { + this._scaleFactor = 1; + this._startDist = TwoFingerSync.calculateDistance(this.posA, this.posB); + this._eventOutput.emit('start', { + count: event.touches.length, + touches: [ + this.touchAId, + this.touchBId + ], + distance: this._startDist, + center: TwoFingerSync.calculateCenter(this.posA, this.posB) + }); +}; +ScaleSync.prototype._moveUpdate = function _moveUpdate(diffTime) { + var scale = this.options.scale; + var currDist = TwoFingerSync.calculateDistance(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + var delta = (currDist - this._startDist) / this._startDist; + var newScaleFactor = Math.max(1 + scale * delta, 0); + var veloScale = (newScaleFactor - this._scaleFactor) / diffTime; + this._eventOutput.emit('update', { + delta: delta, + scale: newScaleFactor, + velocity: veloScale, + distance: currDist, + center: center, + touches: [ + this.touchAId, + this.touchBId + ] + }); + this._scaleFactor = newScaleFactor; +}; +ScaleSync.prototype.getOptions = function getOptions() { + return this.options; +}; +ScaleSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +module.exports = ScaleSync; +},{"../core/OptionsManager":10,"./TwoFingerSync":35}],32:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +var Engine = _dereq_('../core/Engine'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function ScrollSync(options) { + this.options = Object.create(ScrollSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._payload = { + delta: null, + position: null, + velocity: null, + slip: true + }; + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this._position = this.options.direction === undefined ? [ + 0, + 0 + ] : 0; + this._prevTime = undefined; + this._prevVel = undefined; + this._eventInput.on('mousewheel', _handleMove.bind(this)); + this._eventInput.on('wheel', _handleMove.bind(this)); + this._inProgress = false; + this._loopBound = false; +} +ScrollSync.DEFAULT_OPTIONS = { + direction: undefined, + minimumEndSpeed: Infinity, + rails: false, + scale: 1, + stallTime: 50, + lineHeight: 40, + preventDefault: true +}; +ScrollSync.DIRECTION_X = 0; +ScrollSync.DIRECTION_Y = 1; +var MINIMUM_TICK_TIME = 8; +var _now = Date.now; +function _newFrame() { + if (this._inProgress && _now() - this._prevTime > this.options.stallTime) { + this._inProgress = false; + var finalVel = Math.abs(this._prevVel) >= this.options.minimumEndSpeed ? this._prevVel : 0; + var payload = this._payload; + payload.position = this._position; + payload.velocity = finalVel; + payload.slip = true; + this._eventOutput.emit('end', payload); + } +} +function _handleMove(event) { + if (this.options.preventDefault) + event.preventDefault(); + if (!this._inProgress) { + this._inProgress = true; + this._position = this.options.direction === undefined ? [ + 0, + 0 + ] : 0; + payload = this._payload; + payload.slip = true; + payload.position = this._position; + payload.clientX = event.clientX; + payload.clientY = event.clientY; + payload.offsetX = event.offsetX; + payload.offsetY = event.offsetY; + this._eventOutput.emit('start', payload); + if (!this._loopBound) { + Engine.on('prerender', _newFrame.bind(this)); + this._loopBound = true; + } + } + var currTime = _now(); + var prevTime = this._prevTime || currTime; + var diffX = event.wheelDeltaX !== undefined ? event.wheelDeltaX : -event.deltaX; + var diffY = event.wheelDeltaY !== undefined ? event.wheelDeltaY : -event.deltaY; + if (event.deltaMode === 1) { + diffX *= this.options.lineHeight; + diffY *= this.options.lineHeight; + } + if (this.options.rails) { + if (Math.abs(diffX) > Math.abs(diffY)) + diffY = 0; + else + diffX = 0; + } + var diffTime = Math.max(currTime - prevTime, MINIMUM_TICK_TIME); + var velX = diffX / diffTime; + var velY = diffY / diffTime; + var scale = this.options.scale; + var nextVel; + var nextDelta; + if (this.options.direction === ScrollSync.DIRECTION_X) { + nextDelta = scale * diffX; + nextVel = scale * velX; + this._position += nextDelta; + } else if (this.options.direction === ScrollSync.DIRECTION_Y) { + nextDelta = scale * diffY; + nextVel = scale * velY; + this._position += nextDelta; + } else { + nextDelta = [ + scale * diffX, + scale * diffY + ]; + nextVel = [ + scale * velX, + scale * velY + ]; + this._position[0] += nextDelta[0]; + this._position[1] += nextDelta[1]; + } + var payload = this._payload; + payload.delta = nextDelta; + payload.velocity = nextVel; + payload.position = this._position; + payload.slip = true; + this._eventOutput.emit('update', payload); + this._prevTime = currTime; + this._prevVel = nextVel; +} +ScrollSync.prototype.getOptions = function getOptions() { + return this.options; +}; +ScrollSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +module.exports = ScrollSync; +},{"../core/Engine":4,"../core/EventHandler":7,"../core/OptionsManager":10}],33:[function(_dereq_,module,exports){ +var TouchTracker = _dereq_('./TouchTracker'); +var EventHandler = _dereq_('../core/EventHandler'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function TouchSync(options) { + this.options = Object.create(TouchSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._eventOutput = new EventHandler(); + this._touchTracker = new TouchTracker({ touchLimit: this.options.touchLimit }); + EventHandler.setOutputHandler(this, this._eventOutput); + EventHandler.setInputHandler(this, this._touchTracker); + this._touchTracker.on('trackstart', _handleStart.bind(this)); + this._touchTracker.on('trackmove', _handleMove.bind(this)); + this._touchTracker.on('trackend', _handleEnd.bind(this)); + this._payload = { + delta: null, + position: null, + velocity: null, + clientX: undefined, + clientY: undefined, + count: 0, + touch: undefined + }; + this._position = null; +} +TouchSync.DEFAULT_OPTIONS = { + direction: undefined, + rails: false, + touchLimit: 1, + velocitySampleLength: 10, + scale: 1 +}; +TouchSync.DIRECTION_X = 0; +TouchSync.DIRECTION_Y = 1; +var MINIMUM_TICK_TIME = 8; +function _handleStart(data) { + var velocity; + var delta; + if (this.options.direction !== undefined) { + this._position = 0; + velocity = 0; + delta = 0; + } else { + this._position = [ + 0, + 0 + ]; + velocity = [ + 0, + 0 + ]; + delta = [ + 0, + 0 + ]; + } + var payload = this._payload; + payload.delta = delta; + payload.position = this._position; + payload.velocity = velocity; + payload.clientX = data.x; + payload.clientY = data.y; + payload.count = data.count; + payload.touch = data.identifier; + this._eventOutput.emit('start', payload); +} +function _handleMove(data) { + var history = data.history; + var currHistory = history[history.length - 1]; + var prevHistory = history[history.length - 2]; + var distantHistory = history[history.length - this.options.velocitySampleLength] ? history[history.length - this.options.velocitySampleLength] : history[history.length - 2]; + var distantTime = distantHistory.timestamp; + var currTime = currHistory.timestamp; + var diffX = currHistory.x - prevHistory.x; + var diffY = currHistory.y - prevHistory.y; + var velDiffX = currHistory.x - distantHistory.x; + var velDiffY = currHistory.y - distantHistory.y; + if (this.options.rails) { + if (Math.abs(diffX) > Math.abs(diffY)) + diffY = 0; + else + diffX = 0; + if (Math.abs(velDiffX) > Math.abs(velDiffY)) + velDiffY = 0; + else + velDiffX = 0; + } + var diffTime = Math.max(currTime - distantTime, MINIMUM_TICK_TIME); + var velX = velDiffX / diffTime; + var velY = velDiffY / diffTime; + var scale = this.options.scale; + var nextVel; + var nextDelta; + if (this.options.direction === TouchSync.DIRECTION_X) { + nextDelta = scale * diffX; + nextVel = scale * velX; + this._position += nextDelta; + } else if (this.options.direction === TouchSync.DIRECTION_Y) { + nextDelta = scale * diffY; + nextVel = scale * velY; + this._position += nextDelta; + } else { + nextDelta = [ + scale * diffX, + scale * diffY + ]; + nextVel = [ + scale * velX, + scale * velY + ]; + this._position[0] += nextDelta[0]; + this._position[1] += nextDelta[1]; + } + var payload = this._payload; + payload.delta = nextDelta; + payload.velocity = nextVel; + payload.position = this._position; + payload.clientX = data.x; + payload.clientY = data.y; + payload.count = data.count; + payload.touch = data.identifier; + this._eventOutput.emit('update', payload); +} +function _handleEnd(data) { + this._payload.count = data.count; + this._eventOutput.emit('end', this._payload); +} +TouchSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +TouchSync.prototype.getOptions = function getOptions() { + return this.options; +}; +module.exports = TouchSync; +},{"../core/EventHandler":7,"../core/OptionsManager":10,"./TouchTracker":34}],34:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +var _now = Date.now; +function _timestampTouch(touch, event, history) { + return { + x: touch.clientX, + y: touch.clientY, + identifier: touch.identifier, + origin: event.origin, + timestamp: _now(), + count: event.touches.length, + history: history + }; +} +function _handleStart(event) { + if (event.touches.length > this.touchLimit) + return; + this.isTouched = true; + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var data = _timestampTouch(touch, event, null); + this.eventOutput.emit('trackstart', data); + if (!this.selective && !this.touchHistory[touch.identifier]) + this.track(data); + } +} +function _handleMove(event) { + if (event.touches.length > this.touchLimit) + return; + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var history = this.touchHistory[touch.identifier]; + if (history) { + var data = _timestampTouch(touch, event, history); + this.touchHistory[touch.identifier].push(data); + this.eventOutput.emit('trackmove', data); + } + } +} +function _handleEnd(event) { + if (!this.isTouched) + return; + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var history = this.touchHistory[touch.identifier]; + if (history) { + var data = _timestampTouch(touch, event, history); + this.eventOutput.emit('trackend', data); + delete this.touchHistory[touch.identifier]; + } + } + this.isTouched = false; +} +function _handleUnpipe() { + for (var i in this.touchHistory) { + var history = this.touchHistory[i]; + this.eventOutput.emit('trackend', { + touch: history[history.length - 1].touch, + timestamp: Date.now(), + count: 0, + history: history + }); + delete this.touchHistory[i]; + } +} +function TouchTracker(options) { + this.selective = options.selective; + this.touchLimit = options.touchLimit || 1; + this.touchHistory = {}; + this.eventInput = new EventHandler(); + this.eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this.eventInput); + EventHandler.setOutputHandler(this, this.eventOutput); + this.eventInput.on('touchstart', _handleStart.bind(this)); + this.eventInput.on('touchmove', _handleMove.bind(this)); + this.eventInput.on('touchend', _handleEnd.bind(this)); + this.eventInput.on('touchcancel', _handleEnd.bind(this)); + this.eventInput.on('unpipe', _handleUnpipe.bind(this)); + this.isTouched = false; +} +TouchTracker.prototype.track = function track(data) { + this.touchHistory[data.identifier] = [data]; +}; +module.exports = TouchTracker; +},{"../core/EventHandler":7}],35:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +function TwoFingerSync() { + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this.touchAEnabled = false; + this.touchAId = 0; + this.posA = null; + this.timestampA = 0; + this.touchBEnabled = false; + this.touchBId = 0; + this.posB = null; + this.timestampB = 0; + this._eventInput.on('touchstart', this.handleStart.bind(this)); + this._eventInput.on('touchmove', this.handleMove.bind(this)); + this._eventInput.on('touchend', this.handleEnd.bind(this)); + this._eventInput.on('touchcancel', this.handleEnd.bind(this)); +} +TwoFingerSync.calculateAngle = function (posA, posB) { + var diffX = posB[0] - posA[0]; + var diffY = posB[1] - posA[1]; + return Math.atan2(diffY, diffX); +}; +TwoFingerSync.calculateDistance = function (posA, posB) { + var diffX = posB[0] - posA[0]; + var diffY = posB[1] - posA[1]; + return Math.sqrt(diffX * diffX + diffY * diffY); +}; +TwoFingerSync.calculateCenter = function (posA, posB) { + return [ + (posA[0] + posB[0]) / 2, + (posA[1] + posB[1]) / 2 + ]; +}; +var _now = Date.now; +TwoFingerSync.prototype.handleStart = function handleStart(event) { + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + if (!this.touchAEnabled) { + this.touchAId = touch.identifier; + this.touchAEnabled = true; + this.posA = [ + touch.pageX, + touch.pageY + ]; + this.timestampA = _now(); + } else if (!this.touchBEnabled) { + this.touchBId = touch.identifier; + this.touchBEnabled = true; + this.posB = [ + touch.pageX, + touch.pageY + ]; + this.timestampB = _now(); + this._startUpdate(event); + } + } +}; +TwoFingerSync.prototype.handleMove = function handleMove(event) { + if (!(this.touchAEnabled && this.touchBEnabled)) + return; + var prevTimeA = this.timestampA; + var prevTimeB = this.timestampB; + var diffTime; + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + if (touch.identifier === this.touchAId) { + this.posA = [ + touch.pageX, + touch.pageY + ]; + this.timestampA = _now(); + diffTime = this.timestampA - prevTimeA; + } else if (touch.identifier === this.touchBId) { + this.posB = [ + touch.pageX, + touch.pageY + ]; + this.timestampB = _now(); + diffTime = this.timestampB - prevTimeB; + } + } + if (diffTime) + this._moveUpdate(diffTime); +}; +TwoFingerSync.prototype.handleEnd = function handleEnd(event) { + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + if (touch.identifier === this.touchAId || touch.identifier === this.touchBId) { + if (this.touchAEnabled && this.touchBEnabled) { + this._eventOutput.emit('end', { + touches: [ + this.touchAId, + this.touchBId + ], + angle: this._angle + }); + } + this.touchAEnabled = false; + this.touchAId = 0; + this.touchBEnabled = false; + this.touchBId = 0; + } + } +}; +module.exports = TwoFingerSync; +},{"../core/EventHandler":7}],36:[function(_dereq_,module,exports){ +module.exports = { + Accumulator: _dereq_('./Accumulator'), + DesktopEmulationMode: _dereq_('./DesktopEmulationMode'), + FastClick: _dereq_('./FastClick'), + GenericSync: _dereq_('./GenericSync'), + MouseSync: _dereq_('./MouseSync'), + PinchSync: _dereq_('./PinchSync'), + RotateSync: _dereq_('./RotateSync'), + ScaleSync: _dereq_('./ScaleSync'), + ScrollSync: _dereq_('./ScrollSync'), + TouchSync: _dereq_('./TouchSync'), + TouchTracker: _dereq_('./TouchTracker'), + TwoFingerSync: _dereq_('./TwoFingerSync') +}; + +},{"./Accumulator":24,"./DesktopEmulationMode":25,"./FastClick":26,"./GenericSync":27,"./MouseSync":28,"./PinchSync":29,"./RotateSync":30,"./ScaleSync":31,"./ScrollSync":32,"./TouchSync":33,"./TouchTracker":34,"./TwoFingerSync":35}],37:[function(_dereq_,module,exports){ +var Vector = _dereq_('./Vector'); +function Matrix(values) { + this.values = values || [ + [ + 1, + 0, + 0 + ], + [ + 0, + 1, + 0 + ], + [ + 0, + 0, + 1 + ] + ]; + return this; +} +var _register = new Matrix(); +var _vectorRegister = new Vector(); +Matrix.prototype.get = function get() { + return this.values; +}; +Matrix.prototype.set = function set(values) { + this.values = values; +}; +Matrix.prototype.vectorMultiply = function vectorMultiply(v) { + var M = this.get(); + var v0 = v.x; + var v1 = v.y; + var v2 = v.z; + var M0 = M[0]; + var M1 = M[1]; + var M2 = M[2]; + var M00 = M0[0]; + var M01 = M0[1]; + var M02 = M0[2]; + var M10 = M1[0]; + var M11 = M1[1]; + var M12 = M1[2]; + var M20 = M2[0]; + var M21 = M2[1]; + var M22 = M2[2]; + return _vectorRegister.setXYZ(M00 * v0 + M01 * v1 + M02 * v2, M10 * v0 + M11 * v1 + M12 * v2, M20 * v0 + M21 * v1 + M22 * v2); +}; +Matrix.prototype.multiply = function multiply(M2) { + var M1 = this.get(); + var result = [[]]; + for (var i = 0; i < 3; i++) { + result[i] = []; + for (var j = 0; j < 3; j++) { + var sum = 0; + for (var k = 0; k < 3; k++) { + sum += M1[i][k] * M2[k][j]; + } + result[i][j] = sum; + } + } + return _register.set(result); +}; +Matrix.prototype.transpose = function transpose() { + var result = [ + [], + [], + [] + ]; + var M = this.get(); + for (var row = 0; row < 3; row++) { + for (var col = 0; col < 3; col++) { + result[row][col] = M[col][row]; + } + } + return _register.set(result); +}; +Matrix.prototype.clone = function clone() { + var values = this.get(); + var M = []; + for (var row = 0; row < 3; row++) + M[row] = values[row].slice(); + return new Matrix(M); +}; +module.exports = Matrix; +},{"./Vector":41}],38:[function(_dereq_,module,exports){ +var Matrix = _dereq_('./Matrix'); +function Quaternion(w, x, y, z) { + if (arguments.length === 1) + this.set(w); + else { + this.w = w !== undefined ? w : 1; + this.x = x !== undefined ? x : 0; + this.y = y !== undefined ? y : 0; + this.z = z !== undefined ? z : 0; + } + return this; +} +var register = new Quaternion(1, 0, 0, 0); +Quaternion.prototype.add = function add(q) { + return register.setWXYZ(this.w + q.w, this.x + q.x, this.y + q.y, this.z + q.z); +}; +Quaternion.prototype.sub = function sub(q) { + return register.setWXYZ(this.w - q.w, this.x - q.x, this.y - q.y, this.z - q.z); +}; +Quaternion.prototype.scalarDivide = function scalarDivide(s) { + return this.scalarMultiply(1 / s); +}; +Quaternion.prototype.scalarMultiply = function scalarMultiply(s) { + return register.setWXYZ(this.w * s, this.x * s, this.y * s, this.z * s); +}; +Quaternion.prototype.multiply = function multiply(q) { + var x1 = this.x; + var y1 = this.y; + var z1 = this.z; + var w1 = this.w; + var x2 = q.x; + var y2 = q.y; + var z2 = q.z; + var w2 = q.w || 0; + return register.setWXYZ(w1 * w2 - x1 * x2 - y1 * y2 - z1 * z2, x1 * w2 + x2 * w1 + y2 * z1 - y1 * z2, y1 * w2 + y2 * w1 + x1 * z2 - x2 * z1, z1 * w2 + z2 * w1 + x2 * y1 - x1 * y2); +}; +var conj = new Quaternion(1, 0, 0, 0); +Quaternion.prototype.rotateVector = function rotateVector(v) { + conj.set(this.conj()); + return register.set(this.multiply(v).multiply(conj)); +}; +Quaternion.prototype.inverse = function inverse() { + return register.set(this.conj().scalarDivide(this.normSquared())); +}; +Quaternion.prototype.negate = function negate() { + return this.scalarMultiply(-1); +}; +Quaternion.prototype.conj = function conj() { + return register.setWXYZ(this.w, -this.x, -this.y, -this.z); +}; +Quaternion.prototype.normalize = function normalize(length) { + length = length === undefined ? 1 : length; + return this.scalarDivide(length * this.norm()); +}; +Quaternion.prototype.makeFromAngleAndAxis = function makeFromAngleAndAxis(angle, v) { + var n = v.normalize(); + var ha = angle * 0.5; + var s = -Math.sin(ha); + this.x = s * n.x; + this.y = s * n.y; + this.z = s * n.z; + this.w = Math.cos(ha); + return this; +}; +Quaternion.prototype.setWXYZ = function setWXYZ(w, x, y, z) { + register.clear(); + this.w = w; + this.x = x; + this.y = y; + this.z = z; + return this; +}; +Quaternion.prototype.set = function set(v) { + if (v instanceof Array) { + this.w = 0; + this.x = v[0]; + this.y = v[1]; + this.z = v[2]; + } else { + this.w = v.w; + this.x = v.x; + this.y = v.y; + this.z = v.z; + } + if (this !== register) + register.clear(); + return this; +}; +Quaternion.prototype.put = function put(q) { + q.set(register); +}; +Quaternion.prototype.clone = function clone() { + return new Quaternion(this); +}; +Quaternion.prototype.clear = function clear() { + this.w = 1; + this.x = 0; + this.y = 0; + this.z = 0; + return this; +}; +Quaternion.prototype.isEqual = function isEqual(q) { + return q.w === this.w && q.x === this.x && q.y === this.y && q.z === this.z; +}; +Quaternion.prototype.dot = function dot(q) { + return this.w * q.w + this.x * q.x + this.y * q.y + this.z * q.z; +}; +Quaternion.prototype.normSquared = function normSquared() { + return this.dot(this); +}; +Quaternion.prototype.norm = function norm() { + return Math.sqrt(this.normSquared()); +}; +Quaternion.prototype.isZero = function isZero() { + return !(this.x || this.y || this.z); +}; +Quaternion.prototype.getTransform = function getTransform() { + var temp = this.normalize(1); + var x = temp.x; + var y = temp.y; + var z = temp.z; + var w = temp.w; + return [ + 1 - 2 * y * y - 2 * z * z, + 2 * x * y - 2 * z * w, + 2 * x * z + 2 * y * w, + 0, + 2 * x * y + 2 * z * w, + 1 - 2 * x * x - 2 * z * z, + 2 * y * z - 2 * x * w, + 0, + 2 * x * z - 2 * y * w, + 2 * y * z + 2 * x * w, + 1 - 2 * x * x - 2 * y * y, + 0, + 0, + 0, + 0, + 1 + ]; +}; +var matrixRegister = new Matrix(); +Quaternion.prototype.getMatrix = function getMatrix() { + var temp = this.normalize(1); + var x = temp.x; + var y = temp.y; + var z = temp.z; + var w = temp.w; + return matrixRegister.set([ + [ + 1 - 2 * y * y - 2 * z * z, + 2 * x * y + 2 * z * w, + 2 * x * z - 2 * y * w + ], + [ + 2 * x * y - 2 * z * w, + 1 - 2 * x * x - 2 * z * z, + 2 * y * z + 2 * x * w + ], + [ + 2 * x * z + 2 * y * w, + 2 * y * z - 2 * x * w, + 1 - 2 * x * x - 2 * y * y + ] + ]); +}; +var epsilon = 0.00001; +Quaternion.prototype.slerp = function slerp(q, t) { + var omega; + var cosomega; + var sinomega; + var scaleFrom; + var scaleTo; + cosomega = this.dot(q); + if (1 - cosomega > epsilon) { + omega = Math.acos(cosomega); + sinomega = Math.sin(omega); + scaleFrom = Math.sin((1 - t) * omega) / sinomega; + scaleTo = Math.sin(t * omega) / sinomega; + } else { + scaleFrom = 1 - t; + scaleTo = t; + } + return register.set(this.scalarMultiply(scaleFrom / scaleTo).add(q).scalarMultiply(scaleTo)); +}; +module.exports = Quaternion; +},{"./Matrix":37}],39:[function(_dereq_,module,exports){ +var RAND = Math.random; +function _randomFloat(min, max) { + return min + RAND() * (max - min); +} +function _randomInteger(min, max) { + return min + (RAND() * (max - min + 1) >> 0); +} +function _range(randomFunction, min, max, dim) { + min = min !== undefined ? min : 0; + max = max !== undefined ? max : 1; + if (dim !== undefined) { + var result = []; + for (var i = 0; i < dim; i++) + result.push(randomFunction(min, max)); + return result; + } else + return randomFunction(min, max); +} +var Random = {}; +Random.integer = function integer(min, max, dim) { + return _range(_randomInteger, min, max, dim); +}; +Random.range = function range(min, max, dim) { + return _range(_randomFloat, min, max, dim); +}; +Random.sign = function sign(prob) { + return Random.bool(prob) ? 1 : -1; +}; +Random.bool = function bool(prob) { + prob = prob !== undefined ? prob : 0.5; + return RAND() < prob; +}; +module.exports = Random; +},{}],40:[function(_dereq_,module,exports){ +var Utilities = {}; +Utilities.clamp = function clamp(value, range) { + return Math.max(Math.min(value, range[1]), range[0]); +}; +Utilities.length = function length(array) { + var distanceSquared = 0; + for (var i = 0; i < array.length; i++) { + distanceSquared += array[i] * array[i]; + } + return Math.sqrt(distanceSquared); +}; +module.exports = Utilities; +},{}],41:[function(_dereq_,module,exports){ +function Vector(x, y, z) { + if (arguments.length === 1 && x !== undefined) + this.set(x); + else { + this.x = x || 0; + this.y = y || 0; + this.z = z || 0; + } + return this; +} +var _register = new Vector(0, 0, 0); +Vector.prototype.add = function add(v) { + return _setXYZ.call(_register, this.x + v.x, this.y + v.y, this.z + v.z); +}; +Vector.prototype.sub = function sub(v) { + return _setXYZ.call(_register, this.x - v.x, this.y - v.y, this.z - v.z); +}; +Vector.prototype.mult = function mult(r) { + return _setXYZ.call(_register, r * this.x, r * this.y, r * this.z); +}; +Vector.prototype.div = function div(r) { + return this.mult(1 / r); +}; +Vector.prototype.cross = function cross(v) { + var x = this.x; + var y = this.y; + var z = this.z; + var vx = v.x; + var vy = v.y; + var vz = v.z; + return _setXYZ.call(_register, z * vy - y * vz, x * vz - z * vx, y * vx - x * vy); +}; +Vector.prototype.equals = function equals(v) { + return v.x === this.x && v.y === this.y && v.z === this.z; +}; +Vector.prototype.rotateX = function rotateX(theta) { + var x = this.x; + var y = this.y; + var z = this.z; + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + return _setXYZ.call(_register, x, y * cosTheta - z * sinTheta, y * sinTheta + z * cosTheta); +}; +Vector.prototype.rotateY = function rotateY(theta) { + var x = this.x; + var y = this.y; + var z = this.z; + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + return _setXYZ.call(_register, z * sinTheta + x * cosTheta, y, z * cosTheta - x * sinTheta); +}; +Vector.prototype.rotateZ = function rotateZ(theta) { + var x = this.x; + var y = this.y; + var z = this.z; + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + return _setXYZ.call(_register, x * cosTheta - y * sinTheta, x * sinTheta + y * cosTheta, z); +}; +Vector.prototype.dot = function dot(v) { + return this.x * v.x + this.y * v.y + this.z * v.z; +}; +Vector.prototype.normSquared = function normSquared() { + return this.dot(this); +}; +Vector.prototype.norm = function norm() { + return Math.sqrt(this.normSquared()); +}; +Vector.prototype.normalize = function normalize(length) { + if (arguments.length === 0) + length = 1; + var norm = this.norm(); + if (norm > 1e-7) + return _setFromVector.call(_register, this.mult(length / norm)); + else + return _setXYZ.call(_register, length, 0, 0); +}; +Vector.prototype.clone = function clone() { + return new Vector(this); +}; +Vector.prototype.isZero = function isZero() { + return !(this.x || this.y || this.z); +}; +function _setXYZ(x, y, z) { + this.x = x; + this.y = y; + this.z = z; + return this; +} +function _setFromArray(v) { + return _setXYZ.call(this, v[0], v[1], v[2] || 0); +} +function _setFromVector(v) { + return _setXYZ.call(this, v.x, v.y, v.z); +} +function _setFromNumber(x) { + return _setXYZ.call(this, x, 0, 0); +} +Vector.prototype.set = function set(v) { + if (v instanceof Array) + return _setFromArray.call(this, v); + if (typeof v === 'number') + return _setFromNumber.call(this, v); + return _setFromVector.call(this, v); +}; +Vector.prototype.setXYZ = function (x, y, z) { + return _setXYZ.apply(this, arguments); +}; +Vector.prototype.set1D = function (x) { + return _setFromNumber.call(this, x); +}; +Vector.prototype.put = function put(v) { + if (this === _register) + _setFromVector.call(v, _register); + else + _setFromVector.call(v, this); +}; +Vector.prototype.clear = function clear() { + return _setXYZ.call(this, 0, 0, 0); +}; +Vector.prototype.cap = function cap(cap) { + if (cap === Infinity) + return _setFromVector.call(_register, this); + var norm = this.norm(); + if (norm > cap) + return _setFromVector.call(_register, this.mult(cap / norm)); + else + return _setFromVector.call(_register, this); +}; +Vector.prototype.project = function project(n) { + return n.mult(this.dot(n)); +}; +Vector.prototype.reflectAcross = function reflectAcross(n) { + n.normalize().put(n); + return _setFromVector(_register, this.sub(this.project(n).mult(2))); +}; +Vector.prototype.get = function get() { + return [ + this.x, + this.y, + this.z + ]; +}; +Vector.prototype.get1D = function () { + return this.x; +}; +module.exports = Vector; +},{}],42:[function(_dereq_,module,exports){ +module.exports = { + Matrix: _dereq_('./Matrix'), + Quaternion: _dereq_('./Quaternion'), + Random: _dereq_('./Random'), + Utilities: _dereq_('./Utilities'), + Vector: _dereq_('./Vector') +}; + +},{"./Matrix":37,"./Quaternion":38,"./Random":39,"./Utilities":40,"./Vector":41}],43:[function(_dereq_,module,exports){ +var Transform = _dereq_('../core/Transform'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var EventHandler = _dereq_('../core/EventHandler'); +var Utilities = _dereq_('../math/Utilities'); +var GenericSync = _dereq_('../inputs/GenericSync'); +var MouseSync = _dereq_('../inputs/MouseSync'); +var TouchSync = _dereq_('../inputs/TouchSync'); +GenericSync.register({ + 'mouse': MouseSync, + 'touch': TouchSync +}); +function Draggable(options) { + this.options = Object.create(Draggable.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this._positionState = new Transitionable([ + 0, + 0 + ]); + this._differential = [ + 0, + 0 + ]; + this._active = true; + this.sync = new GenericSync([ + 'mouse', + 'touch' + ], { scale: this.options.scale }); + this.eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this.sync); + EventHandler.setOutputHandler(this, this.eventOutput); + _bindEvents.call(this); +} +var _direction = { + x: 1, + y: 2 + }; +Draggable.DIRECTION_X = _direction.x; +Draggable.DIRECTION_Y = _direction.y; +var _clamp = Utilities.clamp; +Draggable.DEFAULT_OPTIONS = { + projection: _direction.x | _direction.y, + scale: 1, + xRange: null, + yRange: null, + snapX: 0, + snapY: 0, + transition: { duration: 0 } +}; +function _mapDifferential(differential) { + var opts = this.options; + var projection = opts.projection; + var snapX = opts.snapX; + var snapY = opts.snapY; + var tx = projection & _direction.x ? differential[0] : 0; + var ty = projection & _direction.y ? differential[1] : 0; + if (snapX > 0) + tx -= tx % snapX; + if (snapY > 0) + ty -= ty % snapY; + return [ + tx, + ty + ]; +} +function _handleStart() { + if (!this._active) + return; + if (this._positionState.isActive()) + this._positionState.halt(); + this.eventOutput.emit('start', { position: this.getPosition() }); +} +function _handleMove(event) { + if (!this._active) + return; + var options = this.options; + this._differential = event.position; + var newDifferential = _mapDifferential.call(this, this._differential); + this._differential[0] -= newDifferential[0]; + this._differential[1] -= newDifferential[1]; + var pos = this.getPosition(); + pos[0] += newDifferential[0]; + pos[1] += newDifferential[1]; + if (options.xRange) { + var xRange = [ + options.xRange[0] + 0.5 * options.snapX, + options.xRange[1] - 0.5 * options.snapX + ]; + pos[0] = _clamp(pos[0], xRange); + } + if (options.yRange) { + var yRange = [ + options.yRange[0] + 0.5 * options.snapY, + options.yRange[1] - 0.5 * options.snapY + ]; + pos[1] = _clamp(pos[1], yRange); + } + this.eventOutput.emit('update', { position: pos }); +} +function _handleEnd() { + if (!this._active) + return; + this.eventOutput.emit('end', { position: this.getPosition() }); +} +function _bindEvents() { + this.sync.on('start', _handleStart.bind(this)); + this.sync.on('update', _handleMove.bind(this)); + this.sync.on('end', _handleEnd.bind(this)); +} +Draggable.prototype.setOptions = function setOptions(options) { + var currentOptions = this.options; + if (options.projection !== undefined) { + var proj = options.projection; + this.options.projection = 0; + [ + 'x', + 'y' + ].forEach(function (val) { + if (proj.indexOf(val) !== -1) + currentOptions.projection |= _direction[val]; + }); + } + if (options.scale !== undefined) { + currentOptions.scale = options.scale; + this.sync.setOptions({ scale: options.scale }); + } + if (options.xRange !== undefined) + currentOptions.xRange = options.xRange; + if (options.yRange !== undefined) + currentOptions.yRange = options.yRange; + if (options.snapX !== undefined) + currentOptions.snapX = options.snapX; + if (options.snapY !== undefined) + currentOptions.snapY = options.snapY; +}; +Draggable.prototype.getPosition = function getPosition() { + return this._positionState.get(); +}; +Draggable.prototype.setRelativePosition = function setRelativePosition(position, transition, callback) { + var currPos = this.getPosition(); + var relativePosition = [ + currPos[0] + position[0], + currPos[1] + position[1] + ]; + this.setPosition(relativePosition, transition, callback); +}; +Draggable.prototype.setPosition = function setPosition(position, transition, callback) { + if (this._positionState.isActive()) + this._positionState.halt(); + this._positionState.set(position, transition, callback); +}; +Draggable.prototype.activate = function activate() { + this._active = true; +}; +Draggable.prototype.deactivate = function deactivate() { + this._active = false; +}; +Draggable.prototype.toggle = function toggle() { + this._active = !this._active; +}; +Draggable.prototype.modify = function modify(target) { + var pos = this.getPosition(); + return { + transform: Transform.translate(pos[0], pos[1]), + target: target + }; +}; +module.exports = Draggable; +},{"../core/EventHandler":7,"../core/Transform":15,"../inputs/GenericSync":27,"../inputs/MouseSync":28,"../inputs/TouchSync":33,"../math/Utilities":40,"../transitions/Transitionable":88}],44:[function(_dereq_,module,exports){ +var Transitionable = _dereq_('../transitions/Transitionable'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function Fader(options, startState) { + this.options = Object.create(Fader.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + if (!startState) + startState = 0; + this.transitionHelper = new Transitionable(startState); +} +Fader.DEFAULT_OPTIONS = { + cull: false, + transition: true, + pulseInTransition: true, + pulseOutTransition: true +}; +Fader.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +Fader.prototype.show = function show(transition, callback) { + transition = transition || this.options.transition; + this.set(1, transition, callback); +}; +Fader.prototype.hide = function hide(transition, callback) { + transition = transition || this.options.transition; + this.set(0, transition, callback); +}; +Fader.prototype.set = function set(state, transition, callback) { + this.halt(); + this.transitionHelper.set(state, transition, callback); +}; +Fader.prototype.halt = function halt() { + this.transitionHelper.halt(); +}; +Fader.prototype.isVisible = function isVisible() { + return this.transitionHelper.get() > 0; +}; +Fader.prototype.modify = function modify(target) { + var currOpacity = this.transitionHelper.get(); + if (this.options.cull && !currOpacity) + return undefined; + else + return { + opacity: currOpacity, + target: target + }; +}; +module.exports = Fader; +},{"../core/OptionsManager":10,"../transitions/Transitionable":88}],45:[function(_dereq_,module,exports){ +function ModifierChain() { + this._chain = []; + if (arguments.length) + this.addModifier.apply(this, arguments); +} +ModifierChain.prototype.addModifier = function addModifier(varargs) { + Array.prototype.push.apply(this._chain, arguments); +}; +ModifierChain.prototype.removeModifier = function removeModifier(modifier) { + var index = this._chain.indexOf(modifier); + if (index < 0) + return; + this._chain.splice(index, 1); +}; +ModifierChain.prototype.modify = function modify(input) { + var chain = this._chain; + var result = input; + for (var i = 0; i < chain.length; i++) { + result = chain[i].modify(result); + } + return result; +}; +module.exports = ModifierChain; +},{}],46:[function(_dereq_,module,exports){ +var Modifier = _dereq_('../core/Modifier'); +var Transform = _dereq_('../core/Transform'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var TransitionableTransform = _dereq_('../transitions/TransitionableTransform'); +function StateModifier(options) { + this._transformState = new TransitionableTransform(Transform.identity); + this._opacityState = new Transitionable(1); + this._originState = new Transitionable([ + 0, + 0 + ]); + this._alignState = new Transitionable([ + 0, + 0 + ]); + this._sizeState = new Transitionable([ + 0, + 0 + ]); + this._proportionsState = new Transitionable([ + 0, + 0 + ]); + this._modifier = new Modifier({ + transform: this._transformState, + opacity: this._opacityState, + origin: null, + align: null, + size: null, + proportions: null + }); + this._hasOrigin = false; + this._hasAlign = false; + this._hasSize = false; + this._hasProportions = false; + if (options) { + if (options.transform) + this.setTransform(options.transform); + if (options.opacity !== undefined) + this.setOpacity(options.opacity); + if (options.origin) + this.setOrigin(options.origin); + if (options.align) + this.setAlign(options.align); + if (options.size) + this.setSize(options.size); + if (options.proportions) + this.setProportions(options.proportions); + } +} +StateModifier.prototype.setTransform = function setTransform(transform, transition, callback) { + this._transformState.set(transform, transition, callback); + return this; +}; +StateModifier.prototype.setOpacity = function setOpacity(opacity, transition, callback) { + this._opacityState.set(opacity, transition, callback); + return this; +}; +StateModifier.prototype.setOrigin = function setOrigin(origin, transition, callback) { + if (origin === null) { + if (this._hasOrigin) { + this._modifier.originFrom(null); + this._hasOrigin = false; + } + return this; + } else if (!this._hasOrigin) { + this._hasOrigin = true; + this._modifier.originFrom(this._originState); + } + this._originState.set(origin, transition, callback); + return this; +}; +StateModifier.prototype.setAlign = function setOrigin(align, transition, callback) { + if (align === null) { + if (this._hasAlign) { + this._modifier.alignFrom(null); + this._hasAlign = false; + } + return this; + } else if (!this._hasAlign) { + this._hasAlign = true; + this._modifier.alignFrom(this._alignState); + } + this._alignState.set(align, transition, callback); + return this; +}; +StateModifier.prototype.setSize = function setSize(size, transition, callback) { + if (size === null) { + if (this._hasSize) { + this._modifier.sizeFrom(null); + this._hasSize = false; + } + return this; + } else if (!this._hasSize) { + this._hasSize = true; + this._modifier.sizeFrom(this._sizeState); + } + this._sizeState.set(size, transition, callback); + return this; +}; +StateModifier.prototype.setProportions = function setSize(proportions, transition, callback) { + if (proportions === null) { + if (this._hasProportions) { + this._modifier.proportionsFrom(null); + this._hasProportions = false; + } + return this; + } else if (!this._hasProportions) { + this._hasProportions = true; + this._modifier.proportionsFrom(this._proportionsState); + } + this._proportionsState.set(proportions, transition, callback); + return this; +}; +StateModifier.prototype.halt = function halt() { + this._transformState.halt(); + this._opacityState.halt(); + this._originState.halt(); + this._alignState.halt(); + this._sizeState.halt(); + this._proportionsState.halt(); +}; +StateModifier.prototype.getTransform = function getTransform() { + return this._transformState.get(); +}; +StateModifier.prototype.getFinalTransform = function getFinalTransform() { + return this._transformState.getFinal(); +}; +StateModifier.prototype.getOpacity = function getOpacity() { + return this._opacityState.get(); +}; +StateModifier.prototype.getOrigin = function getOrigin() { + return this._hasOrigin ? this._originState.get() : null; +}; +StateModifier.prototype.getAlign = function getAlign() { + return this._hasAlign ? this._alignState.get() : null; +}; +StateModifier.prototype.getSize = function getSize() { + return this._hasSize ? this._sizeState.get() : null; +}; +StateModifier.prototype.getProportions = function getProportions() { + return this._hasProportions ? this._proportionsState.get() : null; +}; +StateModifier.prototype.modify = function modify(target) { + return this._modifier.modify(target); +}; +module.exports = StateModifier; +},{"../core/Modifier":9,"../core/Transform":15,"../transitions/Transitionable":88,"../transitions/TransitionableTransform":89}],47:[function(_dereq_,module,exports){ +module.exports = { + Draggable: _dereq_('./Draggable'), + Fader: _dereq_('./Fader'), + ModifierChain: _dereq_('./ModifierChain'), + StateModifier: _dereq_('./StateModifier') +}; + +},{"./Draggable":43,"./Fader":44,"./ModifierChain":45,"./StateModifier":46}],48:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../core/EventHandler'); +function PhysicsEngine(options) { + this.options = Object.create(PhysicsEngine.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this._particles = []; + this._bodies = []; + this._agentData = {}; + this._forces = []; + this._constraints = []; + this._buffer = 0; + this._prevTime = now(); + this._isSleeping = false; + this._eventHandler = null; + this._currAgentId = 0; + this._hasBodies = false; + this._eventHandler = null; +} +var TIMESTEP = 17; +var MIN_TIME_STEP = 1000 / 120; +var MAX_TIME_STEP = 17; +var now = Date.now; +var _events = { + start: 'start', + update: 'update', + end: 'end' + }; +PhysicsEngine.DEFAULT_OPTIONS = { + constraintSteps: 1, + sleepTolerance: 1e-7, + velocityCap: undefined, + angularVelocityCap: undefined +}; +PhysicsEngine.prototype.setOptions = function setOptions(opts) { + for (var key in opts) + if (this.options[key]) + this.options[key] = opts[key]; +}; +PhysicsEngine.prototype.addBody = function addBody(body) { + body._engine = this; + if (body.isBody) { + this._bodies.push(body); + this._hasBodies = true; + } else + this._particles.push(body); + body.on('start', this.wake.bind(this)); + return body; +}; +PhysicsEngine.prototype.removeBody = function removeBody(body) { + var array = body.isBody ? this._bodies : this._particles; + var index = array.indexOf(body); + if (index > -1) { + for (var agentKey in this._agentData) { + if (this._agentData.hasOwnProperty(agentKey)) { + this.detachFrom(this._agentData[agentKey].id, body); + } + } + array.splice(index, 1); + } + if (this.getBodies().length === 0) + this._hasBodies = false; +}; +function _mapAgentArray(agent) { + if (agent.applyForce) + return this._forces; + if (agent.applyConstraint) + return this._constraints; +} +function _attachOne(agent, targets, source) { + if (targets === undefined) + targets = this.getParticlesAndBodies(); + if (!(targets instanceof Array)) + targets = [targets]; + agent.on('change', this.wake.bind(this)); + this._agentData[this._currAgentId] = { + agent: agent, + id: this._currAgentId, + targets: targets, + source: source + }; + _mapAgentArray.call(this, agent).push(this._currAgentId); + return this._currAgentId++; +} +PhysicsEngine.prototype.attach = function attach(agents, targets, source) { + this.wake(); + if (agents instanceof Array) { + var agentIDs = []; + for (var i = 0; i < agents.length; i++) + agentIDs[i] = _attachOne.call(this, agents[i], targets, source); + return agentIDs; + } else + return _attachOne.call(this, agents, targets, source); +}; +PhysicsEngine.prototype.attachTo = function attachTo(agentID, target) { + _getAgentData.call(this, agentID).targets.push(target); +}; +PhysicsEngine.prototype.detach = function detach(id) { + var agent = this.getAgent(id); + var agentArray = _mapAgentArray.call(this, agent); + var index = agentArray.indexOf(id); + agentArray.splice(index, 1); + delete this._agentData[id]; +}; +PhysicsEngine.prototype.detachFrom = function detachFrom(id, target) { + var boundAgent = _getAgentData.call(this, id); + if (boundAgent.source === target) + this.detach(id); + else { + var targets = boundAgent.targets; + var index = targets.indexOf(target); + if (index > -1) + targets.splice(index, 1); + } +}; +PhysicsEngine.prototype.detachAll = function detachAll() { + this._agentData = {}; + this._forces = []; + this._constraints = []; + this._currAgentId = 0; +}; +function _getAgentData(id) { + return this._agentData[id]; +} +PhysicsEngine.prototype.getAgent = function getAgent(id) { + return _getAgentData.call(this, id).agent; +}; +PhysicsEngine.prototype.getParticles = function getParticles() { + return this._particles; +}; +PhysicsEngine.prototype.getBodies = function getBodies() { + return this._bodies; +}; +PhysicsEngine.prototype.getParticlesAndBodies = function getParticlesAndBodies() { + return this.getParticles().concat(this.getBodies()); +}; +PhysicsEngine.prototype.forEachParticle = function forEachParticle(fn, dt) { + var particles = this.getParticles(); + for (var index = 0, len = particles.length; index < len; index++) + fn.call(this, particles[index], dt); +}; +PhysicsEngine.prototype.forEachBody = function forEachBody(fn, dt) { + if (!this._hasBodies) + return; + var bodies = this.getBodies(); + for (var index = 0, len = bodies.length; index < len; index++) + fn.call(this, bodies[index], dt); +}; +PhysicsEngine.prototype.forEach = function forEach(fn, dt) { + this.forEachParticle(fn, dt); + this.forEachBody(fn, dt); +}; +function _updateForce(index) { + var boundAgent = _getAgentData.call(this, this._forces[index]); + boundAgent.agent.applyForce(boundAgent.targets, boundAgent.source); +} +function _updateForces() { + for (var index = this._forces.length - 1; index > -1; index--) + _updateForce.call(this, index); +} +function _updateConstraint(index, dt) { + var boundAgent = this._agentData[this._constraints[index]]; + return boundAgent.agent.applyConstraint(boundAgent.targets, boundAgent.source, dt); +} +function _updateConstraints(dt) { + var iteration = 0; + while (iteration < this.options.constraintSteps) { + for (var index = this._constraints.length - 1; index > -1; index--) + _updateConstraint.call(this, index, dt); + iteration++; + } +} +function _updateVelocities(body, dt) { + body.integrateVelocity(dt); + if (this.options.velocityCap) + body.velocity.cap(this.options.velocityCap).put(body.velocity); +} +function _updateAngularVelocities(body, dt) { + body.integrateAngularMomentum(dt); + body.updateAngularVelocity(); + if (this.options.angularVelocityCap) + body.angularVelocity.cap(this.options.angularVelocityCap).put(body.angularVelocity); +} +function _updateOrientations(body, dt) { + body.integrateOrientation(dt); +} +function _updatePositions(body, dt) { + body.integratePosition(dt); + body.emit(_events.update, body); +} +function _integrate(dt) { + _updateForces.call(this, dt); + this.forEach(_updateVelocities, dt); + this.forEachBody(_updateAngularVelocities, dt); + _updateConstraints.call(this, dt); + this.forEachBody(_updateOrientations, dt); + this.forEach(_updatePositions, dt); +} +function _getParticlesEnergy() { + var energy = 0; + var particleEnergy = 0; + this.forEach(function (particle) { + particleEnergy = particle.getEnergy(); + energy += particleEnergy; + }); + return energy; +} +function _getAgentsEnergy() { + var energy = 0; + for (var id in this._agentData) + energy += this.getAgentEnergy(id); + return energy; +} +PhysicsEngine.prototype.getAgentEnergy = function (agentId) { + var agentData = _getAgentData.call(this, agentId); + return agentData.agent.getEnergy(agentData.targets, agentData.source); +}; +PhysicsEngine.prototype.getEnergy = function getEnergy() { + return _getParticlesEnergy.call(this) + _getAgentsEnergy.call(this); +}; +PhysicsEngine.prototype.step = function step() { + if (this.isSleeping()) + return; + var currTime = now(); + var dtFrame = currTime - this._prevTime; + this._prevTime = currTime; + if (dtFrame < MIN_TIME_STEP) + return; + if (dtFrame > MAX_TIME_STEP) + dtFrame = MAX_TIME_STEP; + _integrate.call(this, TIMESTEP); + this.emit(_events.update, this); + if (this.getEnergy() < this.options.sleepTolerance) + this.sleep(); +}; +PhysicsEngine.prototype.isSleeping = function isSleeping() { + return this._isSleeping; +}; +PhysicsEngine.prototype.isActive = function isSleeping() { + return !this._isSleeping; +}; +PhysicsEngine.prototype.sleep = function sleep() { + if (this._isSleeping) + return; + this.forEach(function (body) { + body.sleep(); + }); + this.emit(_events.end, this); + this._isSleeping = true; +}; +PhysicsEngine.prototype.wake = function wake() { + if (!this._isSleeping) + return; + this._prevTime = now(); + this.emit(_events.start, this); + this._isSleeping = false; +}; +PhysicsEngine.prototype.emit = function emit(type, data) { + if (this._eventHandler === null) + return; + this._eventHandler.emit(type, data); +}; +PhysicsEngine.prototype.on = function on(event, fn) { + if (this._eventHandler === null) + this._eventHandler = new EventHandler(); + this._eventHandler.on(event, fn); +}; +module.exports = PhysicsEngine; +},{"../core/EventHandler":7}],49:[function(_dereq_,module,exports){ +var Particle = _dereq_('./Particle'); +var Transform = _dereq_('../../core/Transform'); +var Vector = _dereq_('../../math/Vector'); +var Quaternion = _dereq_('../../math/Quaternion'); +var Matrix = _dereq_('../../math/Matrix'); +var Integrator = _dereq_('../integrators/SymplecticEuler'); +function Body(options) { + Particle.call(this, options); + options = options || {}; + this.orientation = new Quaternion(); + this.angularVelocity = new Vector(); + this.angularMomentum = new Vector(); + this.torque = new Vector(); + if (options.orientation) + this.orientation.set(options.orientation); + if (options.angularVelocity) + this.angularVelocity.set(options.angularVelocity); + if (options.angularMomentum) + this.angularMomentum.set(options.angularMomentum); + if (options.torque) + this.torque.set(options.torque); + this.angularVelocity.w = 0; + this.setMomentsOfInertia(); + this.pWorld = new Vector(); +} +Body.DEFAULT_OPTIONS = Particle.DEFAULT_OPTIONS; +Body.DEFAULT_OPTIONS.orientation = [ + 0, + 0, + 0, + 1 +]; +Body.DEFAULT_OPTIONS.angularVelocity = [ + 0, + 0, + 0 +]; +Body.prototype = Object.create(Particle.prototype); +Body.prototype.constructor = Body; +Body.prototype.isBody = true; +Body.prototype.setMass = function setMass() { + Particle.prototype.setMass.apply(this, arguments); + this.setMomentsOfInertia(); +}; +Body.prototype.setMomentsOfInertia = function setMomentsOfInertia() { + this.inertia = new Matrix(); + this.inverseInertia = new Matrix(); +}; +Body.prototype.updateAngularVelocity = function updateAngularVelocity() { + this.angularVelocity.set(this.inverseInertia.vectorMultiply(this.angularMomentum)); +}; +Body.prototype.toWorldCoordinates = function toWorldCoordinates(localPosition) { + return this.pWorld.set(this.orientation.rotateVector(localPosition)); +}; +Body.prototype.getEnergy = function getEnergy() { + return Particle.prototype.getEnergy.call(this) + 0.5 * this.inertia.vectorMultiply(this.angularVelocity).dot(this.angularVelocity); +}; +Body.prototype.reset = function reset(p, v, q, L) { + Particle.prototype.reset.call(this, p, v); + this.angularVelocity.clear(); + this.setOrientation(q || [ + 1, + 0, + 0, + 0 + ]); + this.setAngularMomentum(L || [ + 0, + 0, + 0 + ]); +}; +Body.prototype.setOrientation = function setOrientation(q) { + this.orientation.set(q); +}; +Body.prototype.setAngularVelocity = function setAngularVelocity(w) { + this.wake(); + this.angularVelocity.set(w); +}; +Body.prototype.setAngularMomentum = function setAngularMomentum(L) { + this.wake(); + this.angularMomentum.set(L); +}; +Body.prototype.applyForce = function applyForce(force, location) { + Particle.prototype.applyForce.call(this, force); + if (location !== undefined) + this.applyTorque(location.cross(force)); +}; +Body.prototype.applyTorque = function applyTorque(torque) { + this.wake(); + this.torque.set(this.torque.add(torque)); +}; +Body.prototype.getTransform = function getTransform() { + return Transform.thenMove(this.orientation.getTransform(), Transform.getTranslate(Particle.prototype.getTransform.call(this))); +}; +Body.prototype._integrate = function _integrate(dt) { + Particle.prototype._integrate.call(this, dt); + this.integrateAngularMomentum(dt); + this.updateAngularVelocity(dt); + this.integrateOrientation(dt); +}; +Body.prototype.integrateAngularMomentum = function integrateAngularMomentum(dt) { + Integrator.integrateAngularMomentum(this, dt); +}; +Body.prototype.integrateOrientation = function integrateOrientation(dt) { + Integrator.integrateOrientation(this, dt); +}; +module.exports = Body; +},{"../../core/Transform":15,"../../math/Matrix":37,"../../math/Quaternion":38,"../../math/Vector":41,"../integrators/SymplecticEuler":72,"./Particle":51}],50:[function(_dereq_,module,exports){ +var Body = _dereq_('./Body'); +var Matrix = _dereq_('../../math/Matrix'); +function Circle(options) { + options = options || {}; + this.setRadius(options.radius || 0); + Body.call(this, options); +} +Circle.prototype = Object.create(Body.prototype); +Circle.prototype.constructor = Circle; +Circle.prototype.setRadius = function setRadius(r) { + this.radius = r; + this.size = [ + 2 * this.radius, + 2 * this.radius + ]; + this.setMomentsOfInertia(); +}; +Circle.prototype.setMomentsOfInertia = function setMomentsOfInertia() { + var m = this.mass; + var r = this.radius; + this.inertia = new Matrix([ + [ + 0.25 * m * r * r, + 0, + 0 + ], + [ + 0, + 0.25 * m * r * r, + 0 + ], + [ + 0, + 0, + 0.5 * m * r * r + ] + ]); + this.inverseInertia = new Matrix([ + [ + 4 / (m * r * r), + 0, + 0 + ], + [ + 0, + 4 / (m * r * r), + 0 + ], + [ + 0, + 0, + 2 / (m * r * r) + ] + ]); +}; +module.exports = Circle; +},{"../../math/Matrix":37,"./Body":49}],51:[function(_dereq_,module,exports){ +var Vector = _dereq_('../../math/Vector'); +var Transform = _dereq_('../../core/Transform'); +var EventHandler = _dereq_('../../core/EventHandler'); +var Integrator = _dereq_('../integrators/SymplecticEuler'); +function Particle(options) { + options = options || {}; + var defaults = Particle.DEFAULT_OPTIONS; + this.position = new Vector(); + this.velocity = new Vector(); + this.force = new Vector(); + this._engine = null; + this._isSleeping = true; + this._eventOutput = null; + this.mass = options.mass !== undefined ? options.mass : defaults.mass; + this.inverseMass = 1 / this.mass; + this.setPosition(options.position || defaults.position); + this.setVelocity(options.velocity || defaults.velocity); + this.force.set(options.force || [ + 0, + 0, + 0 + ]); + this.transform = Transform.identity.slice(); + this._spec = { + size: [ + true, + true + ], + target: { + transform: this.transform, + origin: [ + 0.5, + 0.5 + ], + target: null + } + }; +} +Particle.DEFAULT_OPTIONS = { + position: [ + 0, + 0, + 0 + ], + velocity: [ + 0, + 0, + 0 + ], + mass: 1 +}; +var _events = { + start: 'start', + update: 'update', + end: 'end' + }; +var now = Date.now; +Particle.prototype.isBody = false; +Particle.prototype.isActive = function isActive() { + return !this._isSleeping; +}; +Particle.prototype.sleep = function sleep() { + if (this._isSleeping) + return; + this.emit(_events.end, this); + this._isSleeping = true; +}; +Particle.prototype.wake = function wake() { + if (!this._isSleeping) + return; + this.emit(_events.start, this); + this._isSleeping = false; + this._prevTime = now(); + if (this._engine) + this._engine.wake(); +}; +Particle.prototype.setPosition = function setPosition(position) { + this.position.set(position); +}; +Particle.prototype.setPosition1D = function setPosition1D(x) { + this.position.x = x; +}; +Particle.prototype.getPosition = function getPosition() { + this._engine.step(); + return this.position.get(); +}; +Particle.prototype.getPosition1D = function getPosition1D() { + this._engine.step(); + return this.position.x; +}; +Particle.prototype.setVelocity = function setVelocity(velocity) { + this.velocity.set(velocity); + if (!(velocity[0] === 0 && velocity[1] === 0 && velocity[2] === 0)) + this.wake(); +}; +Particle.prototype.setVelocity1D = function setVelocity1D(x) { + this.velocity.x = x; + if (x !== 0) + this.wake(); +}; +Particle.prototype.getVelocity = function getVelocity() { + return this.velocity.get(); +}; +Particle.prototype.setForce = function setForce(force) { + this.force.set(force); + this.wake(); +}; +Particle.prototype.getVelocity1D = function getVelocity1D() { + return this.velocity.x; +}; +Particle.prototype.setMass = function setMass(mass) { + this.mass = mass; + this.inverseMass = 1 / mass; +}; +Particle.prototype.getMass = function getMass() { + return this.mass; +}; +Particle.prototype.reset = function reset(position, velocity) { + this.setPosition(position || [ + 0, + 0, + 0 + ]); + this.setVelocity(velocity || [ + 0, + 0, + 0 + ]); +}; +Particle.prototype.applyForce = function applyForce(force) { + if (force.isZero()) + return; + this.force.add(force).put(this.force); + this.wake(); +}; +Particle.prototype.applyImpulse = function applyImpulse(impulse) { + if (impulse.isZero()) + return; + var velocity = this.velocity; + velocity.add(impulse.mult(this.inverseMass)).put(velocity); +}; +Particle.prototype.integrateVelocity = function integrateVelocity(dt) { + Integrator.integrateVelocity(this, dt); +}; +Particle.prototype.integratePosition = function integratePosition(dt) { + Integrator.integratePosition(this, dt); +}; +Particle.prototype._integrate = function _integrate(dt) { + this.integrateVelocity(dt); + this.integratePosition(dt); +}; +Particle.prototype.getEnergy = function getEnergy() { + return 0.5 * this.mass * this.velocity.normSquared(); +}; +Particle.prototype.getTransform = function getTransform() { + this._engine.step(); + var position = this.position; + var transform = this.transform; + transform[12] = position.x; + transform[13] = position.y; + transform[14] = position.z; + return transform; +}; +Particle.prototype.modify = function modify(target) { + var _spec = this._spec.target; + _spec.transform = this.getTransform(); + _spec.target = target; + return this._spec; +}; +function _createEventOutput() { + this._eventOutput = new EventHandler(); + this._eventOutput.bindThis(this); + EventHandler.setOutputHandler(this, this._eventOutput); +} +Particle.prototype.emit = function emit(type, data) { + if (!this._eventOutput) + return; + this._eventOutput.emit(type, data); +}; +Particle.prototype.on = function on() { + _createEventOutput.call(this); + return this.on.apply(this, arguments); +}; +Particle.prototype.removeListener = function removeListener() { + _createEventOutput.call(this); + return this.removeListener.apply(this, arguments); +}; +Particle.prototype.pipe = function pipe() { + _createEventOutput.call(this); + return this.pipe.apply(this, arguments); +}; +Particle.prototype.unpipe = function unpipe() { + _createEventOutput.call(this); + return this.unpipe.apply(this, arguments); +}; +module.exports = Particle; +},{"../../core/EventHandler":7,"../../core/Transform":15,"../../math/Vector":41,"../integrators/SymplecticEuler":72}],52:[function(_dereq_,module,exports){ +var Body = _dereq_('./Body'); +var Matrix = _dereq_('../../math/Matrix'); +function Rectangle(options) { + options = options || {}; + this.size = options.size || [ + 0, + 0 + ]; + Body.call(this, options); +} +Rectangle.prototype = Object.create(Body.prototype); +Rectangle.prototype.constructor = Rectangle; +Rectangle.prototype.setSize = function setSize(size) { + this.size = size; + this.setMomentsOfInertia(); +}; +Rectangle.prototype.setMomentsOfInertia = function setMomentsOfInertia() { + var m = this.mass; + var w = this.size[0]; + var h = this.size[1]; + this.inertia = new Matrix([ + [ + m * h * h / 12, + 0, + 0 + ], + [ + 0, + m * w * w / 12, + 0 + ], + [ + 0, + 0, + m * (w * w + h * h) / 12 + ] + ]); + this.inverseInertia = new Matrix([ + [ + 12 / (m * h * h), + 0, + 0 + ], + [ + 0, + 12 / (m * w * w), + 0 + ], + [ + 0, + 0, + 12 / (m * (w * w + h * h)) + ] + ]); +}; +module.exports = Rectangle; +},{"../../math/Matrix":37,"./Body":49}],53:[function(_dereq_,module,exports){ +module.exports = { + Body: _dereq_('./Body'), + Circle: _dereq_('./Circle'), + Particle: _dereq_('./Particle'), + Rectangle: _dereq_('./Rectangle') +}; + +},{"./Body":49,"./Circle":50,"./Particle":51,"./Rectangle":52}],54:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Vector = _dereq_('../../math/Vector'); +function Collision(options) { + this.options = Object.create(Collision.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.normal = new Vector(); + this.pDiff = new Vector(); + this.vDiff = new Vector(); + this.impulse1 = new Vector(); + this.impulse2 = new Vector(); + Constraint.call(this); +} +Collision.prototype = Object.create(Constraint.prototype); +Collision.prototype.constructor = Collision; +Collision.DEFAULT_OPTIONS = { + restitution: 0.5, + drift: 0.5, + slop: 0 +}; +function _normalVelocity(particle1, particle2) { + return particle1.velocity.dot(particle2.velocity); +} +Collision.prototype.setOptions = function setOptions(options) { + for (var key in options) + this.options[key] = options[key]; +}; +Collision.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + if (source === undefined) + return; + var v1 = source.velocity; + var p1 = source.position; + var w1 = source.inverseMass; + var r1 = source.radius; + var options = this.options; + var drift = options.drift; + var slop = -options.slop; + var restitution = options.restitution; + var n = this.normal; + var pDiff = this.pDiff; + var vDiff = this.vDiff; + var impulse1 = this.impulse1; + var impulse2 = this.impulse2; + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + if (target === source) + continue; + var v2 = target.velocity; + var p2 = target.position; + var w2 = target.inverseMass; + var r2 = target.radius; + pDiff.set(p2.sub(p1)); + vDiff.set(v2.sub(v1)); + var dist = pDiff.norm(); + var overlap = dist - (r1 + r2); + var effMass = 1 / (w1 + w2); + var gamma = 0; + if (overlap < 0) { + n.set(pDiff.normalize()); + if (this._eventOutput) { + var collisionData = { + target: target, + source: source, + overlap: overlap, + normal: n + }; + this._eventOutput.emit('preCollision', collisionData); + this._eventOutput.emit('collision', collisionData); + } + var lambda = overlap <= slop ? ((1 + restitution) * n.dot(vDiff) + drift / dt * (overlap - slop)) / (gamma + dt / effMass) : (1 + restitution) * n.dot(vDiff) / (gamma + dt / effMass); + n.mult(dt * lambda).put(impulse1); + impulse1.mult(-1).put(impulse2); + source.applyImpulse(impulse1); + target.applyImpulse(impulse2); + if (this._eventOutput) + this._eventOutput.emit('postCollision', collisionData); + } + } +}; +module.exports = Collision; +},{"../../math/Vector":41,"./Constraint":55}],55:[function(_dereq_,module,exports){ +var EventHandler = _dereq_('../../core/EventHandler'); +function Constraint() { + this.options = this.options || {}; + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); +} +Constraint.prototype.setOptions = function setOptions(options) { + this._eventOutput.emit('change', options); +}; +Constraint.prototype.applyConstraint = function applyConstraint() { +}; +Constraint.prototype.getEnergy = function getEnergy() { + return 0; +}; +module.exports = Constraint; +},{"../../core/EventHandler":7}],56:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Vector = _dereq_('../../math/Vector'); +function Curve(options) { + this.options = Object.create(Curve.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.J = new Vector(); + this.impulse = new Vector(); + Constraint.call(this); +} +Curve.prototype = Object.create(Constraint.prototype); +Curve.prototype.constructor = Curve; +var epsilon = 1e-7; +var pi = Math.PI; +Curve.DEFAULT_OPTIONS = { + equation: function (x, y, z) { + return 0; + }, + plane: function (x, y, z) { + return z; + }, + period: 0, + dampingRatio: 0 +}; +Curve.prototype.setOptions = function setOptions(options) { + for (var key in options) + this.options[key] = options[key]; +}; +Curve.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var options = this.options; + var impulse = this.impulse; + var J = this.J; + var f = options.equation; + var g = options.plane; + var dampingRatio = options.dampingRatio; + var period = options.period; + for (var i = 0; i < targets.length; i++) { + var body = targets[i]; + var v = body.velocity; + var p = body.position; + var m = body.mass; + var gamma; + var beta; + if (period === 0) { + gamma = 0; + beta = 1; + } else { + var c = 4 * m * pi * dampingRatio / period; + var k = 4 * m * pi * pi / (period * period); + gamma = 1 / (c + dt * k); + beta = dt * k / (c + dt * k); + } + var x = p.x; + var y = p.y; + var z = p.z; + var f0 = f(x, y, z); + var dfx = (f(x + epsilon, y, z) - f0) / epsilon; + var dfy = (f(x, y + epsilon, z) - f0) / epsilon; + var dfz = (f(x, y, z + epsilon) - f0) / epsilon; + var g0 = g(x, y, z); + var dgx = (g(x + epsilon, y, z) - g0) / epsilon; + var dgy = (g(x, y + epsilon, z) - g0) / epsilon; + var dgz = (g(x, y, z + epsilon) - g0) / epsilon; + J.setXYZ(dfx + dgx, dfy + dgy, dfz + dgz); + var antiDrift = beta / dt * (f0 + g0); + var lambda = -(J.dot(v) + antiDrift) / (gamma + dt * J.normSquared() / m); + impulse.set(J.mult(dt * lambda)); + body.applyImpulse(impulse); + } +}; +module.exports = Curve; +},{"../../math/Vector":41,"./Constraint":55}],57:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Vector = _dereq_('../../math/Vector'); +function Distance(options) { + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.impulse = new Vector(); + this.normal = new Vector(); + this.diffP = new Vector(); + this.diffV = new Vector(); + Constraint.call(this); +} +Distance.prototype = Object.create(Constraint.prototype); +Distance.prototype.constructor = Distance; +Distance.DEFAULT_OPTIONS = { + anchor: null, + length: 0, + minLength: 0, + period: 0, + dampingRatio: 0 +}; +var pi = Math.PI; +Distance.prototype.setOptions = function setOptions(options) { + if (options.anchor) { + if (options.anchor.position instanceof Vector) + this.options.anchor = options.anchor.position; + if (options.anchor instanceof Vector) + this.options.anchor = options.anchor; + if (options.anchor instanceof Array) + this.options.anchor = new Vector(options.anchor); + } + if (options.length !== undefined) + this.options.length = options.length; + if (options.dampingRatio !== undefined) + this.options.dampingRatio = options.dampingRatio; + if (options.period !== undefined) + this.options.period = options.period; + if (options.minLength !== undefined) + this.options.minLength = options.minLength; +}; +function _calcError(impulse, body) { + return body.mass * impulse.norm(); +} +Distance.prototype.setAnchor = function setAnchor(anchor) { + if (!this.options.anchor) + this.options.anchor = new Vector(); + this.options.anchor.set(anchor); +}; +Distance.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var n = this.normal; + var diffP = this.diffP; + var diffV = this.diffV; + var impulse = this.impulse; + var options = this.options; + var dampingRatio = options.dampingRatio; + var period = options.period; + var minLength = options.minLength; + var p2; + var w2; + if (source) { + var v2 = source.velocity; + p2 = source.position; + w2 = source.inverseMass; + } else { + p2 = this.options.anchor; + w2 = 0; + } + var length = this.options.length; + for (var i = 0; i < targets.length; i++) { + var body = targets[i]; + var v1 = body.velocity; + var p1 = body.position; + var w1 = body.inverseMass; + diffP.set(p1.sub(p2)); + n.set(diffP.normalize()); + var dist = diffP.norm() - length; + if (Math.abs(dist) < minLength) + return; + if (source) + diffV.set(v1.sub(v2)); + else + diffV.set(v1); + var effMass = 1 / (w1 + w2); + var gamma; + var beta; + if (period === 0) { + gamma = 0; + beta = 1; + } else { + var c = 4 * effMass * pi * dampingRatio / period; + var k = 4 * effMass * pi * pi / (period * period); + gamma = 1 / (c + dt * k); + beta = dt * k / (c + dt * k); + } + var antiDrift = beta / dt * dist; + var lambda = -(n.dot(diffV) + antiDrift) / (gamma + dt / effMass); + impulse.set(n.mult(dt * lambda)); + body.applyImpulse(impulse); + if (source) + source.applyImpulse(impulse.mult(-1)); + } +}; +module.exports = Distance; +},{"../../math/Vector":41,"./Constraint":55}],58:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Vector = _dereq_('../../math/Vector'); +function Snap(options) { + Constraint.call(this); + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.pDiff = new Vector(); + this.vDiff = new Vector(); + this.impulse1 = new Vector(); + this.impulse2 = new Vector(); +} +Snap.prototype = Object.create(Constraint.prototype); +Snap.prototype.constructor = Snap; +Snap.DEFAULT_OPTIONS = { + period: 300, + dampingRatio: 0.1, + length: 0, + anchor: undefined +}; +var pi = Math.PI; +Snap.prototype.setOptions = function setOptions(options) { + if (options.anchor !== undefined) { + if (options.anchor instanceof Vector) + this.options.anchor = options.anchor; + if (options.anchor.position instanceof Vector) + this.options.anchor = options.anchor.position; + if (options.anchor instanceof Array) + this.options.anchor = new Vector(options.anchor); + } + if (options.length !== undefined) + this.options.length = options.length; + if (options.dampingRatio !== undefined) + this.options.dampingRatio = options.dampingRatio; + if (options.period !== undefined) + this.options.period = options.period; + Constraint.prototype.setOptions.call(this, options); +}; +Snap.prototype.getEnergy = function getEnergy(targets, source) { + var options = this.options; + var restLength = options.length; + var anchor = options.anchor || source.position; + var strength = Math.pow(2 * pi / options.period, 2); + var energy = 0; + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + var dist = anchor.sub(target.position).norm() - restLength; + energy += 0.5 * strength * dist * dist; + } + return energy; +}; +Snap.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var options = this.options; + var pDiff = this.pDiff; + var vDiff = this.vDiff; + var impulse1 = this.impulse1; + var impulse2 = this.impulse2; + var length = options.length; + var anchor = options.anchor || source.position; + var period = options.period; + var dampingRatio = options.dampingRatio; + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + var p1 = target.position; + var v1 = target.velocity; + var m1 = target.mass; + var w1 = target.inverseMass; + pDiff.set(p1.sub(anchor)); + var dist = pDiff.norm() - length; + var effMass; + if (source) { + var w2 = source.inverseMass; + var v2 = source.velocity; + vDiff.set(v1.sub(v2)); + effMass = 1 / (w1 + w2); + } else { + vDiff.set(v1); + effMass = m1; + } + var gamma; + var beta; + if (this.options.period === 0) { + gamma = 0; + beta = 1; + } else { + var k = 4 * effMass * pi * pi / (period * period); + var c = 4 * effMass * pi * dampingRatio / period; + beta = dt * k / (c + dt * k); + gamma = 1 / (c + dt * k); + } + var antiDrift = beta / dt * dist; + pDiff.normalize(-antiDrift).sub(vDiff).mult(dt / (gamma + dt / effMass)).put(impulse1); + target.applyImpulse(impulse1); + if (source) { + impulse1.mult(-1).put(impulse2); + source.applyImpulse(impulse2); + } + } +}; +module.exports = Snap; +},{"../../math/Vector":41,"./Constraint":55}],59:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Vector = _dereq_('../../math/Vector'); +function Surface(options) { + this.options = Object.create(Surface.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.J = new Vector(); + this.impulse = new Vector(); + Constraint.call(this); +} +Surface.prototype = Object.create(Constraint.prototype); +Surface.prototype.constructor = Surface; +Surface.DEFAULT_OPTIONS = { + equation: undefined, + period: 0, + dampingRatio: 0 +}; +var epsilon = 1e-7; +var pi = Math.PI; +Surface.prototype.setOptions = function setOptions(options) { + for (var key in options) + this.options[key] = options[key]; +}; +Surface.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var impulse = this.impulse; + var J = this.J; + var options = this.options; + var f = options.equation; + var dampingRatio = options.dampingRatio; + var period = options.period; + for (var i = 0; i < targets.length; i++) { + var particle = targets[i]; + var v = particle.velocity; + var p = particle.position; + var m = particle.mass; + var gamma; + var beta; + if (period === 0) { + gamma = 0; + beta = 1; + } else { + var c = 4 * m * pi * dampingRatio / period; + var k = 4 * m * pi * pi / (period * period); + gamma = 1 / (c + dt * k); + beta = dt * k / (c + dt * k); + } + var x = p.x; + var y = p.y; + var z = p.z; + var f0 = f(x, y, z); + var dfx = (f(x + epsilon, p, p) - f0) / epsilon; + var dfy = (f(x, y + epsilon, p) - f0) / epsilon; + var dfz = (f(x, y, p + epsilon) - f0) / epsilon; + J.setXYZ(dfx, dfy, dfz); + var antiDrift = beta / dt * f0; + var lambda = -(J.dot(v) + antiDrift) / (gamma + dt * J.normSquared() / m); + impulse.set(J.mult(dt * lambda)); + particle.applyImpulse(impulse); + } +}; +module.exports = Surface; +},{"../../math/Vector":41,"./Constraint":55}],60:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Vector = _dereq_('../../math/Vector'); +function Wall(options) { + this.options = Object.create(Wall.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.diff = new Vector(); + this.impulse = new Vector(); + Constraint.call(this); +} +Wall.prototype = Object.create(Constraint.prototype); +Wall.prototype.constructor = Wall; +Wall.ON_CONTACT = { + REFLECT: 0, + SILENT: 1 +}; +Wall.DEFAULT_OPTIONS = { + restitution: 0.5, + drift: 0.5, + slop: 0, + normal: [ + 1, + 0, + 0 + ], + distance: 0, + onContact: Wall.ON_CONTACT.REFLECT +}; +Wall.prototype.setOptions = function setOptions(options) { + if (options.normal !== undefined) { + if (options.normal instanceof Vector) + this.options.normal = options.normal.clone(); + if (options.normal instanceof Array) + this.options.normal = new Vector(options.normal); + } + if (options.restitution !== undefined) + this.options.restitution = options.restitution; + if (options.drift !== undefined) + this.options.drift = options.drift; + if (options.slop !== undefined) + this.options.slop = options.slop; + if (options.distance !== undefined) + this.options.distance = options.distance; + if (options.onContact !== undefined) + this.options.onContact = options.onContact; +}; +function _getNormalVelocity(n, v) { + return v.dot(n); +} +function _getDistanceFromOrigin(p) { + var n = this.options.normal; + var d = this.options.distance; + return p.dot(n) + d; +} +function _onEnter(particle, overlap, dt) { + var p = particle.position; + var v = particle.velocity; + var m = particle.mass; + var n = this.options.normal; + var action = this.options.onContact; + var restitution = this.options.restitution; + var impulse = this.impulse; + var drift = this.options.drift; + var slop = -this.options.slop; + var gamma = 0; + if (this._eventOutput) { + var data = { + particle: particle, + wall: this, + overlap: overlap, + normal: n + }; + this._eventOutput.emit('preCollision', data); + this._eventOutput.emit('collision', data); + } + switch (action) { + case Wall.ON_CONTACT.REFLECT: + var lambda = overlap < slop ? -((1 + restitution) * n.dot(v) + drift / dt * (overlap - slop)) / (m * dt + gamma) : -((1 + restitution) * n.dot(v)) / (m * dt + gamma); + impulse.set(n.mult(dt * lambda)); + particle.applyImpulse(impulse); + particle.setPosition(p.add(n.mult(-overlap))); + break; + } + if (this._eventOutput) + this._eventOutput.emit('postCollision', data); +} +function _onExit(particle, overlap, dt) { + var action = this.options.onContact; + var p = particle.position; + var n = this.options.normal; + if (action === Wall.ON_CONTACT.REFLECT) { + particle.setPosition(p.add(n.mult(-overlap))); + } +} +Wall.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var n = this.options.normal; + for (var i = 0; i < targets.length; i++) { + var particle = targets[i]; + var p = particle.position; + var v = particle.velocity; + var r = particle.radius || 0; + var overlap = _getDistanceFromOrigin.call(this, p.add(n.mult(-r))); + var nv = _getNormalVelocity.call(this, n, v); + if (overlap <= 0) { + if (nv < 0) + _onEnter.call(this, particle, overlap, dt); + else + _onExit.call(this, particle, overlap, dt); + } + } +}; +module.exports = Wall; +},{"../../math/Vector":41,"./Constraint":55}],61:[function(_dereq_,module,exports){ +var Constraint = _dereq_('./Constraint'); +var Wall = _dereq_('./Wall'); +var Vector = _dereq_('../../math/Vector'); +function Walls(options) { + this.options = Object.create(Walls.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + _createComponents.call(this, options.sides || this.options.sides); + Constraint.call(this); +} +Walls.prototype = Object.create(Constraint.prototype); +Walls.prototype.constructor = Walls; +Walls.ON_CONTACT = Wall.ON_CONTACT; +Walls.SIDES = { + LEFT: 0, + RIGHT: 1, + TOP: 2, + BOTTOM: 3, + FRONT: 4, + BACK: 5, + TWO_DIMENSIONAL: [ + 0, + 1, + 2, + 3 + ], + THREE_DIMENSIONAL: [ + 0, + 1, + 2, + 3, + 4, + 5 + ] +}; +Walls.DEFAULT_OPTIONS = { + sides: Walls.SIDES.TWO_DIMENSIONAL, + size: [ + window.innerWidth, + window.innerHeight, + 0 + ], + origin: [ + 0.5, + 0.5, + 0.5 + ], + drift: 0.5, + slop: 0, + restitution: 0.5, + onContact: Walls.ON_CONTACT.REFLECT +}; +var _SIDE_NORMALS = { + 0: new Vector(1, 0, 0), + 1: new Vector(-1, 0, 0), + 2: new Vector(0, 1, 0), + 3: new Vector(0, -1, 0), + 4: new Vector(0, 0, 1), + 5: new Vector(0, 0, -1) + }; +function _getDistance(side, size, origin) { + var distance; + var SIDES = Walls.SIDES; + switch (parseInt(side)) { + case SIDES.LEFT: + distance = size[0] * origin[0]; + break; + case SIDES.TOP: + distance = size[1] * origin[1]; + break; + case SIDES.FRONT: + distance = size[2] * origin[2]; + break; + case SIDES.RIGHT: + distance = size[0] * (1 - origin[0]); + break; + case SIDES.BOTTOM: + distance = size[1] * (1 - origin[1]); + break; + case SIDES.BACK: + distance = size[2] * (1 - origin[2]); + break; + } + return distance; +} +Walls.prototype.setOptions = function setOptions(options) { + var resizeFlag = false; + if (options.restitution !== undefined) + _setOptionsForEach.call(this, { restitution: options.restitution }); + if (options.drift !== undefined) + _setOptionsForEach.call(this, { drift: options.drift }); + if (options.slop !== undefined) + _setOptionsForEach.call(this, { slop: options.slop }); + if (options.onContact !== undefined) + _setOptionsForEach.call(this, { onContact: options.onContact }); + if (options.size !== undefined) + resizeFlag = true; + if (options.sides !== undefined) + this.options.sides = options.sides; + if (options.origin !== undefined) + resizeFlag = true; + if (resizeFlag) + this.setSize(options.size, options.origin); +}; +function _createComponents(sides) { + this.components = {}; + var components = this.components; + for (var i = 0; i < sides.length; i++) { + var side = sides[i]; + components[i] = new Wall({ + normal: _SIDE_NORMALS[side].clone(), + distance: _getDistance(side, this.options.size, this.options.origin) + }); + } +} +Walls.prototype.setSize = function setSize(size, origin) { + origin = origin || this.options.origin; + if (origin.length < 3) + origin[2] = 0.5; + this.forEach(function (wall, side) { + var d = _getDistance(side, size, origin); + wall.setOptions({ distance: d }); + }); + this.options.size = size; + this.options.origin = origin; +}; +function _setOptionsForEach(options) { + this.forEach(function (wall) { + wall.setOptions(options); + }); + for (var key in options) + this.options[key] = options[key]; +} +Walls.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + this.forEach(function (wall) { + wall.applyConstraint(targets, source, dt); + }); +}; +Walls.prototype.forEach = function forEach(fn) { + var sides = this.options.sides; + for (var key in this.sides) + fn(sides[key], key); +}; +Walls.prototype.rotateZ = function rotateZ(angle) { + this.forEach(function (wall) { + var n = wall.options.normal; + n.rotateZ(angle).put(n); + }); +}; +Walls.prototype.rotateX = function rotateX(angle) { + this.forEach(function (wall) { + var n = wall.options.normal; + n.rotateX(angle).put(n); + }); +}; +Walls.prototype.rotateY = function rotateY(angle) { + this.forEach(function (wall) { + var n = wall.options.normal; + n.rotateY(angle).put(n); + }); +}; +Walls.prototype.reset = function reset() { + var sides = this.options.sides; + for (var i in sides) { + var component = this.components[i]; + component.options.normal.set(_SIDE_NORMALS[i]); + } +}; +module.exports = Walls; +},{"../../math/Vector":41,"./Constraint":55,"./Wall":60}],62:[function(_dereq_,module,exports){ +module.exports = { + Collision: _dereq_('./Collision'), + Constraint: _dereq_('./Constraint'), + Curve: _dereq_('./Curve'), + Distance: _dereq_('./Distance'), + Snap: _dereq_('./Snap'), + Surface: _dereq_('./Surface'), + Wall: _dereq_('./Wall'), + Walls: _dereq_('./Walls') +}; + +},{"./Collision":54,"./Constraint":55,"./Curve":56,"./Distance":57,"./Snap":58,"./Surface":59,"./Wall":60,"./Walls":61}],63:[function(_dereq_,module,exports){ +var Force = _dereq_('./Force'); +function Drag(options) { + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + Force.call(this); +} +Drag.prototype = Object.create(Force.prototype); +Drag.prototype.constructor = Drag; +Drag.FORCE_FUNCTIONS = { + LINEAR: function (velocity) { + return velocity; + }, + QUADRATIC: function (velocity) { + return velocity.mult(velocity.norm()); + } +}; +Drag.DEFAULT_OPTIONS = { + strength: 0.01, + forceFunction: Drag.FORCE_FUNCTIONS.LINEAR +}; +Drag.prototype.applyForce = function applyForce(targets) { + var strength = this.options.strength; + var forceFunction = this.options.forceFunction; + var force = this.force; + var index; + var particle; + for (index = 0; index < targets.length; index++) { + particle = targets[index]; + forceFunction(particle.velocity).mult(-strength).put(force); + particle.applyForce(force); + } +}; +Drag.prototype.setOptions = function setOptions(options) { + for (var key in options) + this.options[key] = options[key]; +}; +module.exports = Drag; +},{"./Force":64}],64:[function(_dereq_,module,exports){ +var Vector = _dereq_('../../math/Vector'); +var EventHandler = _dereq_('../../core/EventHandler'); +function Force(force) { + this.force = new Vector(force); + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); +} +Force.prototype.setOptions = function setOptions(options) { + this._eventOutput.emit('change', options); +}; +Force.prototype.applyForce = function applyForce(targets) { + var length = targets.length; + while (length--) { + targets[length].applyForce(this.force); + } +}; +Force.prototype.getEnergy = function getEnergy() { + return 0; +}; +module.exports = Force; +},{"../../core/EventHandler":7,"../../math/Vector":41}],65:[function(_dereq_,module,exports){ +var Force = _dereq_('./Force'); +var Vector = _dereq_('../../math/Vector'); +function Repulsion(options) { + this.options = Object.create(Repulsion.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.disp = new Vector(); + Force.call(this); +} +Repulsion.prototype = Object.create(Force.prototype); +Repulsion.prototype.constructor = Repulsion; +Repulsion.DECAY_FUNCTIONS = { + LINEAR: function (r, cutoff) { + return Math.max(1 - 1 / cutoff * r, 0); + }, + MORSE: function (r, cutoff) { + var r0 = cutoff === 0 ? 100 : cutoff; + var rShifted = r + r0 * (1 - Math.log(2)); + return Math.max(1 - Math.pow(1 - Math.exp(rShifted / r0 - 1), 2), 0); + }, + INVERSE: function (r, cutoff) { + return 1 / (1 - cutoff + r); + }, + GRAVITY: function (r, cutoff) { + return 1 / (1 - cutoff + r * r); + } +}; +Repulsion.DEFAULT_OPTIONS = { + strength: 1, + anchor: undefined, + range: [ + 0, + Infinity + ], + cutoff: 0, + cap: Infinity, + decayFunction: Repulsion.DECAY_FUNCTIONS.GRAVITY +}; +Repulsion.prototype.setOptions = function setOptions(options) { + if (options.anchor !== undefined) { + if (options.anchor.position instanceof Vector) + this.options.anchor = options.anchor.position; + if (options.anchor instanceof Array) + this.options.anchor = new Vector(options.anchor); + delete options.anchor; + } + for (var key in options) + this.options[key] = options[key]; +}; +Repulsion.prototype.applyForce = function applyForce(targets, source) { + var options = this.options; + var force = this.force; + var disp = this.disp; + var strength = options.strength; + var anchor = options.anchor || source.position; + var cap = options.cap; + var cutoff = options.cutoff; + var rMin = options.range[0]; + var rMax = options.range[1]; + var decayFn = options.decayFunction; + if (strength === 0) + return; + var length = targets.length; + var particle; + var m1; + var p1; + var r; + while (length--) { + particle = targets[length]; + if (particle === source) + continue; + m1 = particle.mass; + p1 = particle.position; + disp.set(p1.sub(anchor)); + r = disp.norm(); + if (r < rMax && r > rMin) { + force.set(disp.normalize(strength * m1 * decayFn(r, cutoff)).cap(cap)); + particle.applyForce(force); + } + } +}; +module.exports = Repulsion; +},{"../../math/Vector":41,"./Force":64}],66:[function(_dereq_,module,exports){ +var Drag = _dereq_('./Drag'); +function RotationalDrag(options) { + Drag.call(this, options); +} +RotationalDrag.prototype = Object.create(Drag.prototype); +RotationalDrag.prototype.constructor = RotationalDrag; +RotationalDrag.DEFAULT_OPTIONS = Drag.DEFAULT_OPTIONS; +RotationalDrag.FORCE_FUNCTIONS = Drag.FORCE_FUNCTIONS; +RotationalDrag.FORCE_FUNCTIONS = { + LINEAR: function (angularVelocity) { + return angularVelocity; + }, + QUADRATIC: function (angularVelocity) { + return angularVelocity.mult(angularVelocity.norm()); + } +}; +RotationalDrag.prototype.applyForce = function applyForce(targets) { + var strength = this.options.strength; + var forceFunction = this.options.forceFunction; + var force = this.force; + var index; + var particle; + for (index = 0; index < targets.length; index++) { + particle = targets[index]; + forceFunction(particle.angularVelocity).mult(-100 * strength).put(force); + particle.applyTorque(force); + } +}; +RotationalDrag.prototype.setOptions = function setOptions(options) { + for (var key in options) + this.options[key] = options[key]; +}; +module.exports = RotationalDrag; +},{"./Drag":63}],67:[function(_dereq_,module,exports){ +var Force = _dereq_('./Force'); +var Spring = _dereq_('./Spring'); +var Quaternion = _dereq_('../../math/Quaternion'); +function RotationalSpring(options) { + Spring.call(this, options); +} +RotationalSpring.prototype = Object.create(Spring.prototype); +RotationalSpring.prototype.constructor = RotationalSpring; +RotationalSpring.DEFAULT_OPTIONS = Spring.DEFAULT_OPTIONS; +RotationalSpring.FORCE_FUNCTIONS = Spring.FORCE_FUNCTIONS; +var pi = Math.PI; +function _calcStiffness() { + var options = this.options; + options.stiffness = Math.pow(2 * pi / options.period, 2); +} +function _calcDamping() { + var options = this.options; + options.damping = 4 * pi * options.dampingRatio / options.period; +} +function _init() { + _calcStiffness.call(this); + _calcDamping.call(this); +} +RotationalSpring.prototype.setOptions = function setOptions(options) { + if (options.anchor !== undefined) { + if (options.anchor instanceof Quaternion) + this.options.anchor = options.anchor; + if (options.anchor instanceof Array) + this.options.anchor = new Quaternion(options.anchor); + } + if (options.period !== undefined) { + this.options.period = options.period; + } + if (options.dampingRatio !== undefined) + this.options.dampingRatio = options.dampingRatio; + if (options.length !== undefined) + this.options.length = options.length; + if (options.forceFunction !== undefined) + this.options.forceFunction = options.forceFunction; + if (options.maxLength !== undefined) + this.options.maxLength = options.maxLength; + _init.call(this); + Force.prototype.setOptions.call(this, options); +}; +RotationalSpring.prototype.applyForce = function applyForce(targets) { + var force = this.force; + var options = this.options; + var disp = this.disp; + var stiffness = options.stiffness; + var damping = options.damping; + var restLength = options.length; + var anchor = options.anchor; + var forceFunction = options.forceFunction; + var maxLength = options.maxLength; + var i; + var target; + var dist; + var m; + for (i = 0; i < targets.length; i++) { + target = targets[i]; + disp.set(anchor.sub(target.orientation)); + dist = disp.norm() - restLength; + if (dist === 0) + return; + m = target.mass; + stiffness *= m; + damping *= m; + force.set(disp.normalize(stiffness * forceFunction(dist, maxLength))); + if (damping) + force.add(target.angularVelocity.mult(-damping)).put(force); + target.applyTorque(force); + } +}; +RotationalSpring.prototype.getEnergy = function getEnergy(targets) { + var options = this.options; + var restLength = options.length; + var anchor = options.anchor; + var strength = options.stiffness; + var energy = 0; + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + var dist = anchor.sub(target.orientation).norm() - restLength; + energy += 0.5 * strength * dist * dist; + } + return energy; +}; +module.exports = RotationalSpring; +},{"../../math/Quaternion":38,"./Force":64,"./Spring":68}],68:[function(_dereq_,module,exports){ +var Force = _dereq_('./Force'); +var Vector = _dereq_('../../math/Vector'); +function Spring(options) { + Force.call(this); + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.disp = new Vector(0, 0, 0); + _init.call(this); +} +Spring.prototype = Object.create(Force.prototype); +Spring.prototype.constructor = Spring; +var pi = Math.PI; +var MIN_PERIOD = 150; +Spring.FORCE_FUNCTIONS = { + FENE: function (dist, rMax) { + var rMaxSmall = rMax * 0.99; + var r = Math.max(Math.min(dist, rMaxSmall), -rMaxSmall); + return r / (1 - r * r / (rMax * rMax)); + }, + HOOK: function (dist) { + return dist; + } +}; +Spring.DEFAULT_OPTIONS = { + period: 300, + dampingRatio: 0.1, + length: 0, + maxLength: Infinity, + anchor: undefined, + forceFunction: Spring.FORCE_FUNCTIONS.HOOK +}; +function _calcStiffness() { + var options = this.options; + options.stiffness = Math.pow(2 * pi / options.period, 2); +} +function _calcDamping() { + var options = this.options; + options.damping = 4 * pi * options.dampingRatio / options.period; +} +function _init() { + _calcStiffness.call(this); + _calcDamping.call(this); +} +Spring.prototype.setOptions = function setOptions(options) { + if (options.anchor !== undefined) { + if (options.anchor.position instanceof Vector) + this.options.anchor = options.anchor.position; + if (options.anchor instanceof Vector) + this.options.anchor = options.anchor; + if (options.anchor instanceof Array) + this.options.anchor = new Vector(options.anchor); + } + if (options.period !== undefined) { + if (options.period < MIN_PERIOD) { + options.period = MIN_PERIOD; + console.warn('The period of a SpringTransition is capped at ' + MIN_PERIOD + ' ms. Use a SnapTransition for faster transitions'); + } + this.options.period = options.period; + } + if (options.dampingRatio !== undefined) + this.options.dampingRatio = options.dampingRatio; + if (options.length !== undefined) + this.options.length = options.length; + if (options.forceFunction !== undefined) + this.options.forceFunction = options.forceFunction; + if (options.maxLength !== undefined) + this.options.maxLength = options.maxLength; + _init.call(this); + Force.prototype.setOptions.call(this, options); +}; +Spring.prototype.applyForce = function applyForce(targets, source) { + var force = this.force; + var disp = this.disp; + var options = this.options; + var stiffness = options.stiffness; + var damping = options.damping; + var restLength = options.length; + var maxLength = options.maxLength; + var anchor = options.anchor || source.position; + var forceFunction = options.forceFunction; + var i; + var target; + var p2; + var v2; + var dist; + var m; + for (i = 0; i < targets.length; i++) { + target = targets[i]; + p2 = target.position; + v2 = target.velocity; + anchor.sub(p2).put(disp); + dist = disp.norm() - restLength; + if (dist === 0) + return; + m = target.mass; + stiffness *= m; + damping *= m; + disp.normalize(stiffness * forceFunction(dist, maxLength)).put(force); + if (damping) + if (source) + force.add(v2.sub(source.velocity).mult(-damping)).put(force); + else + force.add(v2.mult(-damping)).put(force); + target.applyForce(force); + if (source) + source.applyForce(force.mult(-1)); + } +}; +Spring.prototype.getEnergy = function getEnergy(targets, source) { + var options = this.options; + var restLength = options.length; + var anchor = source ? source.position : options.anchor; + var strength = options.stiffness; + var energy = 0; + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + var dist = anchor.sub(target.position).norm() - restLength; + energy += 0.5 * strength * dist * dist; + } + return energy; +}; +module.exports = Spring; +},{"../../math/Vector":41,"./Force":64}],69:[function(_dereq_,module,exports){ +var Force = _dereq_('./Force'); +var Vector = _dereq_('../../math/Vector'); +function VectorField(options) { + Force.call(this); + this.options = Object.create(VectorField.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this.evaluation = new Vector(); +} +VectorField.prototype = Object.create(Force.prototype); +VectorField.prototype.constructor = VectorField; +VectorField.FIELDS = { + CONSTANT: function (v, options) { + options.direction.put(this.evaluation); + }, + LINEAR: function (v) { + v.put(this.evaluation); + }, + RADIAL: function (v) { + v.mult(-1).put(this.evaluation); + }, + POINT_ATTRACTOR: function (v, options) { + options.position.sub(v).put(this.evaluation); + } +}; +VectorField.DEFAULT_OPTIONS = { + strength: 0.01, + field: VectorField.FIELDS.CONSTANT +}; +VectorField.prototype.setOptions = function setOptions(options) { + if (options.strength !== undefined) + this.options.strength = options.strength; + if (options.direction !== undefined) + this.options.direction = options.direction; + if (options.field !== undefined) { + this.options.field = options.field; + _setFieldOptions.call(this, this.options.field); + } +}; +function _setFieldOptions(field) { + var FIELDS = VectorField.FIELDS; + switch (field) { + case FIELDS.CONSTANT: + if (!this.options.direction) + this.options.direction = new Vector(0, 1, 0); + else if (this.options.direction instanceof Array) + this.options.direction = new Vector(this.options.direction); + break; + case FIELDS.POINT_ATTRACTOR: + if (!this.options.position) + this.options.position = new Vector(0, 0, 0); + else if (this.options.position instanceof Array) + this.options.position = new Vector(this.options.position); + break; + } +} +VectorField.prototype.applyForce = function applyForce(targets) { + var force = this.force; + var strength = this.options.strength; + var field = this.options.field; + var i; + var target; + for (i = 0; i < targets.length; i++) { + target = targets[i]; + field.call(this, target.position, this.options); + this.evaluation.mult(target.mass * strength).put(force); + target.applyForce(force); + } +}; +VectorField.prototype.getEnergy = function getEnergy(targets) { + var field = this.options.field; + var FIELDS = VectorField.FIELDS; + var energy = 0; + var i; + var target; + switch (field) { + case FIELDS.CONSTANT: + energy = targets.length * this.options.direction.norm(); + break; + case FIELDS.RADIAL: + for (i = 0; i < targets.length; i++) { + target = targets[i]; + energy += target.position.norm(); + } + break; + case FIELDS.POINT_ATTRACTOR: + for (i = 0; i < targets.length; i++) { + target = targets[i]; + energy += target.position.sub(this.options.position).norm(); + } + break; + } + energy *= this.options.strength; + return energy; +}; +module.exports = VectorField; +},{"../../math/Vector":41,"./Force":64}],70:[function(_dereq_,module,exports){ +module.exports = { + Drag: _dereq_('./Drag'), + Force: _dereq_('./Force'), + Repulsion: _dereq_('./Repulsion'), + RotationalDrag: _dereq_('./RotationalDrag'), + RotationalSpring: _dereq_('./RotationalSpring'), + Spring: _dereq_('./Spring'), + VectorField: _dereq_('./VectorField') +}; + +},{"./Drag":63,"./Force":64,"./Repulsion":65,"./RotationalDrag":66,"./RotationalSpring":67,"./Spring":68,"./VectorField":69}],71:[function(_dereq_,module,exports){ +module.exports = { + PhysicsEngine: _dereq_('./PhysicsEngine'), + bodies: _dereq_('./bodies'), + constraints: _dereq_('./constraints'), + forces: _dereq_('./forces'), + integrators: _dereq_('./integrators') +}; + +},{"./PhysicsEngine":48,"./bodies":53,"./constraints":62,"./forces":70,"./integrators":73}],72:[function(_dereq_,module,exports){ +var SymplecticEuler = {}; +SymplecticEuler.integrateVelocity = function integrateVelocity(body, dt) { + var v = body.velocity; + var w = body.inverseMass; + var f = body.force; + if (f.isZero()) + return; + v.add(f.mult(dt * w)).put(v); + f.clear(); +}; +SymplecticEuler.integratePosition = function integratePosition(body, dt) { + var p = body.position; + var v = body.velocity; + p.add(v.mult(dt)).put(p); +}; +SymplecticEuler.integrateAngularMomentum = function integrateAngularMomentum(body, dt) { + var L = body.angularMomentum; + var t = body.torque; + if (t.isZero()) + return; + L.add(t.mult(dt)).put(L); + t.clear(); +}; +SymplecticEuler.integrateOrientation = function integrateOrientation(body, dt) { + var q = body.orientation; + var w = body.angularVelocity; + if (w.isZero()) + return; + q.add(q.multiply(w).scalarMultiply(0.5 * dt)).put(q); +}; +module.exports = SymplecticEuler; +},{}],73:[function(_dereq_,module,exports){ +module.exports = { + SymplecticEuler: _dereq_('./SymplecticEuler') +}; + +},{"./SymplecticEuler":72}],74:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +function CanvasSurface(options) { + if (options && options.canvasSize) + this._canvasSize = options.canvasSize; + Surface.apply(this, arguments); + if (!this._canvasSize) + this._canvasSize = this.getSize(); + this._backBuffer = document.createElement('canvas'); + if (this._canvasSize) { + this._backBuffer.width = this._canvasSize[0]; + this._backBuffer.height = this._canvasSize[1]; + } + this._contextId = undefined; +} +CanvasSurface.prototype = Object.create(Surface.prototype); +CanvasSurface.prototype.constructor = CanvasSurface; +CanvasSurface.prototype.elementType = 'canvas'; +CanvasSurface.prototype.elementClass = 'famous-surface'; +CanvasSurface.prototype.setContent = function setContent() { +}; +CanvasSurface.prototype.deploy = function deploy(target) { + if (this._canvasSize) { + target.width = this._canvasSize[0]; + target.height = this._canvasSize[1]; + } + if (this._contextId === '2d') { + target.getContext(this._contextId).drawImage(this._backBuffer, 0, 0); + this._backBuffer.width = 0; + this._backBuffer.height = 0; + } +}; +CanvasSurface.prototype.recall = function recall(target) { + var size = this.getSize(); + this._backBuffer.width = target.width; + this._backBuffer.height = target.height; + if (this._contextId === '2d') { + this._backBuffer.getContext(this._contextId).drawImage(target, 0, 0); + target.width = 0; + target.height = 0; + } +}; +CanvasSurface.prototype.getContext = function getContext(contextId) { + this._contextId = contextId; + return this._currentTarget ? this._currentTarget.getContext(contextId) : this._backBuffer.getContext(contextId); +}; +CanvasSurface.prototype.setSize = function setSize(size, canvasSize) { + Surface.prototype.setSize.apply(this, arguments); + if (canvasSize) + this._canvasSize = [ + canvasSize[0], + canvasSize[1] + ]; + if (this._currentTarget) { + this._currentTarget.width = this._canvasSize[0]; + this._currentTarget.height = this._canvasSize[1]; + } +}; +module.exports = CanvasSurface; +},{"../core/Surface":14}],75:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +var Context = _dereq_('../core/Context'); +function ContainerSurface(options) { + Surface.call(this, options); + this._container = document.createElement('div'); + this._container.classList.add('famous-group'); + this._container.classList.add('famous-container-group'); + this._shouldRecalculateSize = false; + this.context = new Context(this._container); + this.setContent(this._container); +} +ContainerSurface.prototype = Object.create(Surface.prototype); +ContainerSurface.prototype.constructor = ContainerSurface; +ContainerSurface.prototype.elementType = 'div'; +ContainerSurface.prototype.elementClass = 'famous-surface'; +ContainerSurface.prototype.add = function add() { + return this.context.add.apply(this.context, arguments); +}; +ContainerSurface.prototype.render = function render() { + if (this._sizeDirty) + this._shouldRecalculateSize = true; + return Surface.prototype.render.apply(this, arguments); +}; +ContainerSurface.prototype.deploy = function deploy() { + this._shouldRecalculateSize = true; + return Surface.prototype.deploy.apply(this, arguments); +}; +ContainerSurface.prototype.commit = function commit(context, transform, opacity, origin, size) { + var previousSize = this._size ? [ + this._size[0], + this._size[1] + ] : null; + var result = Surface.prototype.commit.apply(this, arguments); + if (this._shouldRecalculateSize || previousSize && (this._size[0] !== previousSize[0] || this._size[1] !== previousSize[1])) { + this.context.setSize(); + this._shouldRecalculateSize = false; + } + this.context.update(); + return result; +}; +module.exports = ContainerSurface; +},{"../core/Context":1,"../core/Surface":14}],76:[function(_dereq_,module,exports){ +var ContainerSurface = _dereq_('./ContainerSurface'); +function FormContainerSurface(options) { + if (options) + this._method = options.method || ''; + ContainerSurface.apply(this, arguments); +} +FormContainerSurface.prototype = Object.create(ContainerSurface.prototype); +FormContainerSurface.prototype.constructor = FormContainerSurface; +FormContainerSurface.prototype.elementType = 'form'; +FormContainerSurface.prototype.deploy = function deploy(target) { + if (this._method) + target.method = this._method; + return ContainerSurface.prototype.deploy.apply(this, arguments); +}; +module.exports = FormContainerSurface; +},{"./ContainerSurface":75}],77:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +function ImageSurface(options) { + this._imageUrl = undefined; + Surface.apply(this, arguments); +} +var urlCache = []; +var countCache = []; +var nodeCache = []; +var cacheEnabled = true; +ImageSurface.enableCache = function enableCache() { + cacheEnabled = true; +}; +ImageSurface.disableCache = function disableCache() { + cacheEnabled = false; +}; +ImageSurface.clearCache = function clearCache() { + urlCache = []; + countCache = []; + nodeCache = []; +}; +ImageSurface.getCache = function getCache() { + return { + urlCache: urlCache, + countCache: countCache, + nodeCache: nodeCache + }; +}; +ImageSurface.prototype = Object.create(Surface.prototype); +ImageSurface.prototype.constructor = ImageSurface; +ImageSurface.prototype.elementType = 'img'; +ImageSurface.prototype.elementClass = 'famous-surface'; +ImageSurface.prototype.setContent = function setContent(imageUrl) { + var urlIndex = urlCache.indexOf(this._imageUrl); + if (urlIndex !== -1) { + if (countCache[urlIndex] === 1) { + urlCache.splice(urlIndex, 1); + countCache.splice(urlIndex, 1); + nodeCache.splice(urlIndex, 1); + } else { + countCache[urlIndex]--; + } + } + urlIndex = urlCache.indexOf(imageUrl); + if (urlIndex === -1) { + urlCache.push(imageUrl); + countCache.push(1); + } else { + countCache[urlIndex]++; + } + this._imageUrl = imageUrl; + this._contentDirty = true; +}; +ImageSurface.prototype.deploy = function deploy(target) { + var urlIndex = urlCache.indexOf(this._imageUrl); + if (nodeCache[urlIndex] === undefined && cacheEnabled) { + var img = new Image(); + img.src = this._imageUrl || ''; + nodeCache[urlIndex] = img; + } + target.src = this._imageUrl || ''; +}; +ImageSurface.prototype.recall = function recall(target) { + target.src = ''; +}; +module.exports = ImageSurface; +},{"../core/Surface":14}],78:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +function InputSurface(options) { + this._placeholder = options.placeholder || ''; + this._value = options.value || ''; + this._type = options.type || 'text'; + this._name = options.name || ''; + Surface.apply(this, arguments); + this.on('click', this.focus.bind(this)); + window.addEventListener('click', function (event) { + if (event.target !== this._currentTarget) + this.blur(); + }.bind(this)); +} +InputSurface.prototype = Object.create(Surface.prototype); +InputSurface.prototype.constructor = InputSurface; +InputSurface.prototype.elementType = 'input'; +InputSurface.prototype.elementClass = 'famous-surface'; +InputSurface.prototype.setPlaceholder = function setPlaceholder(str) { + this._placeholder = str; + this._contentDirty = true; + return this; +}; +InputSurface.prototype.focus = function focus() { + if (this._currentTarget) + this._currentTarget.focus(); + return this; +}; +InputSurface.prototype.blur = function blur() { + if (this._currentTarget) + this._currentTarget.blur(); + return this; +}; +InputSurface.prototype.setValue = function setValue(str) { + this._value = str; + this._contentDirty = true; + return this; +}; +InputSurface.prototype.setType = function setType(str) { + this._type = str; + this._contentDirty = true; + return this; +}; +InputSurface.prototype.getValue = function getValue() { + if (this._currentTarget) { + return this._currentTarget.value; + } else { + return this._value; + } +}; +InputSurface.prototype.setName = function setName(str) { + this._name = str; + this._contentDirty = true; + return this; +}; +InputSurface.prototype.getName = function getName() { + return this._name; +}; +InputSurface.prototype.deploy = function deploy(target) { + if (this._placeholder !== '') + target.placeholder = this._placeholder; + target.value = this._value; + target.type = this._type; + target.name = this._name; +}; +module.exports = InputSurface; +},{"../core/Surface":14}],79:[function(_dereq_,module,exports){ +var InputSurface = _dereq_('./InputSurface'); +function SubmitInputSurface(options) { + InputSurface.apply(this, arguments); + this._type = 'submit'; + if (options && options.onClick) + this.setOnClick(options.onClick); +} +SubmitInputSurface.prototype = Object.create(InputSurface.prototype); +SubmitInputSurface.prototype.constructor = SubmitInputSurface; +SubmitInputSurface.prototype.setOnClick = function (onClick) { + this.onClick = onClick; +}; +SubmitInputSurface.prototype.deploy = function deploy(target) { + if (this.onclick) + target.onClick = this.onClick; + InputSurface.prototype.deploy.apply(this, arguments); +}; +module.exports = SubmitInputSurface; +},{"./InputSurface":78}],80:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +function TextareaSurface(options) { + this._placeholder = options.placeholder || ''; + this._value = options.value || ''; + this._name = options.name || ''; + this._wrap = options.wrap || ''; + this._cols = options.cols || ''; + this._rows = options.rows || ''; + Surface.apply(this, arguments); + this.on('click', this.focus.bind(this)); +} +TextareaSurface.prototype = Object.create(Surface.prototype); +TextareaSurface.prototype.constructor = TextareaSurface; +TextareaSurface.prototype.elementType = 'textarea'; +TextareaSurface.prototype.elementClass = 'famous-surface'; +TextareaSurface.prototype.setPlaceholder = function setPlaceholder(str) { + this._placeholder = str; + this._contentDirty = true; + return this; +}; +TextareaSurface.prototype.focus = function focus() { + if (this._currentTarget) + this._currentTarget.focus(); + return this; +}; +TextareaSurface.prototype.blur = function blur() { + if (this._currentTarget) + this._currentTarget.blur(); + return this; +}; +TextareaSurface.prototype.setValue = function setValue(str) { + this._value = str; + this._contentDirty = true; + return this; +}; +TextareaSurface.prototype.getValue = function getValue() { + if (this._currentTarget) { + return this._currentTarget.value; + } else { + return this._value; + } +}; +TextareaSurface.prototype.setName = function setName(str) { + this._name = str; + this._contentDirty = true; + return this; +}; +TextareaSurface.prototype.getName = function getName() { + return this._name; +}; +TextareaSurface.prototype.setWrap = function setWrap(str) { + this._wrap = str; + this._contentDirty = true; + return this; +}; +TextareaSurface.prototype.setColumns = function setColumns(num) { + this._cols = num; + this._contentDirty = true; + return this; +}; +TextareaSurface.prototype.setRows = function setRows(num) { + this._rows = num; + this._contentDirty = true; + return this; +}; +TextareaSurface.prototype.deploy = function deploy(target) { + if (this._placeholder !== '') + target.placeholder = this._placeholder; + if (this._value !== '') + target.value = this._value; + if (this._name !== '') + target.name = this._name; + if (this._wrap !== '') + target.wrap = this._wrap; + if (this._cols !== '') + target.cols = this._cols; + if (this._rows !== '') + target.rows = this._rows; +}; +module.exports = TextareaSurface; +},{"../core/Surface":14}],81:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +function VideoSurface(options) { + Surface.apply(this, arguments); + this._videoUrl = undefined; + this.options = Object.create(VideoSurface.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); +} +VideoSurface.prototype = Object.create(Surface.prototype); +VideoSurface.prototype.constructor = VideoSurface; +VideoSurface.DEFAULT_OPTIONS = { autoplay: false }; +VideoSurface.prototype.elementType = 'video'; +VideoSurface.prototype.elementClass = 'famous-surface'; +VideoSurface.prototype.setOptions = function setOptions(options) { + if (options.size) + this.setSize(options.size); + if (options.classes) + this.setClasses(options.classes); + if (options.properties) + this.setProperties(options.properties); + if (options.autoplay) + this.options.autoplay = options.autoplay; + if (options.src) { + this._videoUrl = options.src; + this._contentDirty = true; + } +}; +VideoSurface.prototype.setContent = function setContent(videoUrl) { + this._videoUrl = videoUrl; + this._contentDirty = true; +}; +VideoSurface.prototype.deploy = function deploy(target) { + target.src = this._videoUrl; + target.autoplay = this.options.autoplay; +}; +VideoSurface.prototype.recall = function recall(target) { + target.src = ''; +}; +module.exports = VideoSurface; +},{"../core/Surface":14}],82:[function(_dereq_,module,exports){ +module.exports = { + CanvasSurface: _dereq_('./CanvasSurface'), + ContainerSurface: _dereq_('./ContainerSurface'), + FormContainerSurface: _dereq_('./FormContainerSurface'), + ImageSurface: _dereq_('./ImageSurface'), + InputSurface: _dereq_('./InputSurface'), + SubmitInputSurface: _dereq_('./SubmitInputSurface'), + TextareaSurface: _dereq_('./TextareaSurface'), + VideoSurface: _dereq_('./VideoSurface') +}; + +},{"./CanvasSurface":74,"./ContainerSurface":75,"./FormContainerSurface":76,"./ImageSurface":77,"./InputSurface":78,"./SubmitInputSurface":79,"./TextareaSurface":80,"./VideoSurface":81}],83:[function(_dereq_,module,exports){ +function CachedMap(mappingFunction) { + this._map = mappingFunction || null; + this._cachedOutput = null; + this._cachedInput = Number.NaN; +} +CachedMap.create = function create(mappingFunction) { + var instance = new CachedMap(mappingFunction); + return instance.get.bind(instance); +}; +CachedMap.prototype.get = function get(input) { + if (input !== this._cachedInput) { + this._cachedInput = input; + this._cachedOutput = this._map(input); + } + return this._cachedOutput; +}; +module.exports = CachedMap; +},{}],84:[function(_dereq_,module,exports){ +var Easing = { + inQuad: function (t) { + return t * t; + }, + outQuad: function (t) { + return -(t -= 1) * t + 1; + }, + inOutQuad: function (t) { + if ((t /= 0.5) < 1) + return 0.5 * t * t; + return -0.5 * (--t * (t - 2) - 1); + }, + inCubic: function (t) { + return t * t * t; + }, + outCubic: function (t) { + return --t * t * t + 1; + }, + inOutCubic: function (t) { + if ((t /= 0.5) < 1) + return 0.5 * t * t * t; + return 0.5 * ((t -= 2) * t * t + 2); + }, + inQuart: function (t) { + return t * t * t * t; + }, + outQuart: function (t) { + return -(--t * t * t * t - 1); + }, + inOutQuart: function (t) { + if ((t /= 0.5) < 1) + return 0.5 * t * t * t * t; + return -0.5 * ((t -= 2) * t * t * t - 2); + }, + inQuint: function (t) { + return t * t * t * t * t; + }, + outQuint: function (t) { + return --t * t * t * t * t + 1; + }, + inOutQuint: function (t) { + if ((t /= 0.5) < 1) + return 0.5 * t * t * t * t * t; + return 0.5 * ((t -= 2) * t * t * t * t + 2); + }, + inSine: function (t) { + return -1 * Math.cos(t * (Math.PI / 2)) + 1; + }, + outSine: function (t) { + return Math.sin(t * (Math.PI / 2)); + }, + inOutSine: function (t) { + return -0.5 * (Math.cos(Math.PI * t) - 1); + }, + inExpo: function (t) { + return t === 0 ? 0 : Math.pow(2, 10 * (t - 1)); + }, + outExpo: function (t) { + return t === 1 ? 1 : -Math.pow(2, -10 * t) + 1; + }, + inOutExpo: function (t) { + if (t === 0) + return 0; + if (t === 1) + return 1; + if ((t /= 0.5) < 1) + return 0.5 * Math.pow(2, 10 * (t - 1)); + return 0.5 * (-Math.pow(2, -10 * --t) + 2); + }, + inCirc: function (t) { + return -(Math.sqrt(1 - t * t) - 1); + }, + outCirc: function (t) { + return Math.sqrt(1 - --t * t); + }, + inOutCirc: function (t) { + if ((t /= 0.5) < 1) + return -0.5 * (Math.sqrt(1 - t * t) - 1); + return 0.5 * (Math.sqrt(1 - (t -= 2) * t) + 1); + }, + inElastic: function (t) { + var s = 1.70158; + var p = 0; + var a = 1; + if (t === 0) + return 0; + if (t === 1) + return 1; + if (!p) + p = 0.3; + s = p / (2 * Math.PI) * Math.asin(1 / a); + return -(a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t - s) * (2 * Math.PI) / p)); + }, + outElastic: function (t) { + var s = 1.70158; + var p = 0; + var a = 1; + if (t === 0) + return 0; + if (t === 1) + return 1; + if (!p) + p = 0.3; + s = p / (2 * Math.PI) * Math.asin(1 / a); + return a * Math.pow(2, -10 * t) * Math.sin((t - s) * (2 * Math.PI) / p) + 1; + }, + inOutElastic: function (t) { + var s = 1.70158; + var p = 0; + var a = 1; + if (t === 0) + return 0; + if ((t /= 0.5) === 2) + return 1; + if (!p) + p = 0.3 * 1.5; + s = p / (2 * Math.PI) * Math.asin(1 / a); + if (t < 1) + return -0.5 * (a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t - s) * (2 * Math.PI) / p)); + return a * Math.pow(2, -10 * (t -= 1)) * Math.sin((t - s) * (2 * Math.PI) / p) * 0.5 + 1; + }, + inBack: function (t, s) { + if (s === undefined) + s = 1.70158; + return t * t * ((s + 1) * t - s); + }, + outBack: function (t, s) { + if (s === undefined) + s = 1.70158; + return --t * t * ((s + 1) * t + s) + 1; + }, + inOutBack: function (t, s) { + if (s === undefined) + s = 1.70158; + if ((t /= 0.5) < 1) + return 0.5 * (t * t * (((s *= 1.525) + 1) * t - s)); + return 0.5 * ((t -= 2) * t * (((s *= 1.525) + 1) * t + s) + 2); + }, + inBounce: function (t) { + return 1 - Easing.outBounce(1 - t); + }, + outBounce: function (t) { + if (t < 1 / 2.75) { + return 7.5625 * t * t; + } else if (t < 2 / 2.75) { + return 7.5625 * (t -= 1.5 / 2.75) * t + 0.75; + } else if (t < 2.5 / 2.75) { + return 7.5625 * (t -= 2.25 / 2.75) * t + 0.9375; + } else { + return 7.5625 * (t -= 2.625 / 2.75) * t + 0.984375; + } + }, + inOutBounce: function (t) { + if (t < 0.5) + return Easing.inBounce(t * 2) * 0.5; + return Easing.outBounce(t * 2 - 1) * 0.5 + 0.5; + } + }; +module.exports = Easing; +},{}],85:[function(_dereq_,module,exports){ +var Utility = _dereq_('../utilities/Utility'); +function MultipleTransition(method) { + this.method = method; + this._instances = []; + this.state = []; +} +MultipleTransition.SUPPORTS_MULTIPLE = true; +MultipleTransition.prototype.get = function get() { + for (var i = 0; i < this._instances.length; i++) { + this.state[i] = this._instances[i].get(); + } + return this.state; +}; +MultipleTransition.prototype.set = function set(endState, transition, callback) { + var _allCallback = Utility.after(endState.length, callback); + for (var i = 0; i < endState.length; i++) { + if (!this._instances[i]) + this._instances[i] = new this.method(); + this._instances[i].set(endState[i], transition, _allCallback); + } +}; +MultipleTransition.prototype.reset = function reset(startState) { + for (var i = 0; i < startState.length; i++) { + if (!this._instances[i]) + this._instances[i] = new this.method(); + this._instances[i].reset(startState[i]); + } +}; +module.exports = MultipleTransition; +},{"../utilities/Utility":95}],86:[function(_dereq_,module,exports){ +var PE = _dereq_('../physics/PhysicsEngine'); +var Particle = _dereq_('../physics/bodies/Particle'); +var Spring = _dereq_('../physics/constraints/Snap'); +var Vector = _dereq_('../math/Vector'); +function SnapTransition(state) { + state = state || 0; + this.endState = new Vector(state); + this.initState = new Vector(); + this._dimensions = 1; + this._restTolerance = 1e-10; + this._absRestTolerance = this._restTolerance; + this._callback = undefined; + this.PE = new PE(); + this.particle = new Particle(); + this.spring = new Spring({ anchor: this.endState }); + this.PE.addBody(this.particle); + this.PE.attach(this.spring, this.particle); +} +SnapTransition.SUPPORTS_MULTIPLE = 3; +SnapTransition.DEFAULT_OPTIONS = { + period: 100, + dampingRatio: 0.2, + velocity: 0 +}; +function _getEnergy() { + return this.particle.getEnergy() + this.spring.getEnergy([this.particle]); +} +function _setAbsoluteRestTolerance() { + var distance = this.endState.sub(this.initState).normSquared(); + this._absRestTolerance = distance === 0 ? this._restTolerance : this._restTolerance * distance; +} +function _setTarget(target) { + this.endState.set(target); + _setAbsoluteRestTolerance.call(this); +} +function _wake() { + this.PE.wake(); +} +function _sleep() { + this.PE.sleep(); +} +function _setParticlePosition(p) { + this.particle.position.set(p); +} +function _setParticleVelocity(v) { + this.particle.velocity.set(v); +} +function _getParticlePosition() { + return this._dimensions === 0 ? this.particle.getPosition1D() : this.particle.getPosition(); +} +function _getParticleVelocity() { + return this._dimensions === 0 ? this.particle.getVelocity1D() : this.particle.getVelocity(); +} +function _setCallback(callback) { + this._callback = callback; +} +function _setupDefinition(definition) { + var defaults = SnapTransition.DEFAULT_OPTIONS; + if (definition.period === undefined) + definition.period = defaults.period; + if (definition.dampingRatio === undefined) + definition.dampingRatio = defaults.dampingRatio; + if (definition.velocity === undefined) + definition.velocity = defaults.velocity; + this.spring.setOptions({ + period: definition.period, + dampingRatio: definition.dampingRatio + }); + _setParticleVelocity.call(this, definition.velocity); +} +function _update() { + if (this.PE.isSleeping()) { + if (this._callback) { + var cb = this._callback; + this._callback = undefined; + cb(); + } + return; + } + if (_getEnergy.call(this) < this._absRestTolerance) { + _setParticlePosition.call(this, this.endState); + _setParticleVelocity.call(this, [ + 0, + 0, + 0 + ]); + _sleep.call(this); + } +} +SnapTransition.prototype.reset = function reset(state, velocity) { + this._dimensions = state instanceof Array ? state.length : 0; + this.initState.set(state); + _setParticlePosition.call(this, state); + _setTarget.call(this, state); + if (velocity) + _setParticleVelocity.call(this, velocity); + _setCallback.call(this, undefined); +}; +SnapTransition.prototype.getVelocity = function getVelocity() { + return _getParticleVelocity.call(this); +}; +SnapTransition.prototype.setVelocity = function setVelocity(velocity) { + this.call(this, _setParticleVelocity(velocity)); +}; +SnapTransition.prototype.isActive = function isActive() { + return !this.PE.isSleeping(); +}; +SnapTransition.prototype.halt = function halt() { + this.set(this.get()); +}; +SnapTransition.prototype.get = function get() { + _update.call(this); + return _getParticlePosition.call(this); +}; +SnapTransition.prototype.set = function set(state, definition, callback) { + if (!definition) { + this.reset(state); + if (callback) + callback(); + return; + } + this._dimensions = state instanceof Array ? state.length : 0; + _wake.call(this); + _setupDefinition.call(this, definition); + _setTarget.call(this, state); + _setCallback.call(this, callback); +}; +module.exports = SnapTransition; +},{"../math/Vector":41,"../physics/PhysicsEngine":48,"../physics/bodies/Particle":51,"../physics/constraints/Snap":58}],87:[function(_dereq_,module,exports){ +var PE = _dereq_('../physics/PhysicsEngine'); +var Particle = _dereq_('../physics/bodies/Particle'); +var Spring = _dereq_('../physics/forces/Spring'); +var Vector = _dereq_('../math/Vector'); +function SpringTransition(state) { + state = state || 0; + this.endState = new Vector(state); + this.initState = new Vector(); + this._dimensions = undefined; + this._restTolerance = 1e-10; + this._absRestTolerance = this._restTolerance; + this._callback = undefined; + this.PE = new PE(); + this.spring = new Spring({ anchor: this.endState }); + this.particle = new Particle(); + this.PE.addBody(this.particle); + this.PE.attach(this.spring, this.particle); +} +SpringTransition.SUPPORTS_MULTIPLE = 3; +SpringTransition.DEFAULT_OPTIONS = { + period: 300, + dampingRatio: 0.5, + velocity: 0 +}; +function _getEnergy() { + return this.particle.getEnergy() + this.spring.getEnergy([this.particle]); +} +function _setParticlePosition(p) { + this.particle.setPosition(p); +} +function _setParticleVelocity(v) { + this.particle.setVelocity(v); +} +function _getParticlePosition() { + return this._dimensions === 0 ? this.particle.getPosition1D() : this.particle.getPosition(); +} +function _getParticleVelocity() { + return this._dimensions === 0 ? this.particle.getVelocity1D() : this.particle.getVelocity(); +} +function _setCallback(callback) { + this._callback = callback; +} +function _wake() { + this.PE.wake(); +} +function _sleep() { + this.PE.sleep(); +} +function _update() { + if (this.PE.isSleeping()) { + if (this._callback) { + var cb = this._callback; + this._callback = undefined; + cb(); + } + return; + } + if (_getEnergy.call(this) < this._absRestTolerance) { + _setParticlePosition.call(this, this.endState); + _setParticleVelocity.call(this, [ + 0, + 0, + 0 + ]); + _sleep.call(this); + } +} +function _setupDefinition(definition) { + var defaults = SpringTransition.DEFAULT_OPTIONS; + if (definition.period === undefined) + definition.period = defaults.period; + if (definition.dampingRatio === undefined) + definition.dampingRatio = defaults.dampingRatio; + if (definition.velocity === undefined) + definition.velocity = defaults.velocity; + if (definition.period < 150) { + definition.period = 150; + console.warn('The period of a SpringTransition is capped at 150 ms. Use a SnapTransition for faster transitions'); + } + this.spring.setOptions({ + period: definition.period, + dampingRatio: definition.dampingRatio + }); + _setParticleVelocity.call(this, definition.velocity); +} +function _setAbsoluteRestTolerance() { + var distance = this.endState.sub(this.initState).normSquared(); + this._absRestTolerance = distance === 0 ? this._restTolerance : this._restTolerance * distance; +} +function _setTarget(target) { + this.endState.set(target); + _setAbsoluteRestTolerance.call(this); +} +SpringTransition.prototype.reset = function reset(pos, vel) { + this._dimensions = pos instanceof Array ? pos.length : 0; + this.initState.set(pos); + _setParticlePosition.call(this, pos); + _setTarget.call(this, pos); + if (vel) + _setParticleVelocity.call(this, vel); + _setCallback.call(this, undefined); +}; +SpringTransition.prototype.getVelocity = function getVelocity() { + return _getParticleVelocity.call(this); +}; +SpringTransition.prototype.setVelocity = function setVelocity(v) { + this.call(this, _setParticleVelocity(v)); +}; +SpringTransition.prototype.isActive = function isActive() { + return !this.PE.isSleeping(); +}; +SpringTransition.prototype.halt = function halt() { + this.set(this.get()); +}; +SpringTransition.prototype.get = function get() { + _update.call(this); + return _getParticlePosition.call(this); +}; +SpringTransition.prototype.set = function set(endState, definition, callback) { + if (!definition) { + this.reset(endState); + if (callback) + callback(); + return; + } + this._dimensions = endState instanceof Array ? endState.length : 0; + _wake.call(this); + _setupDefinition.call(this, definition); + _setTarget.call(this, endState); + _setCallback.call(this, callback); +}; +module.exports = SpringTransition; +},{"../math/Vector":41,"../physics/PhysicsEngine":48,"../physics/bodies/Particle":51,"../physics/forces/Spring":68}],88:[function(_dereq_,module,exports){ +var MultipleTransition = _dereq_('./MultipleTransition'); +var TweenTransition = _dereq_('./TweenTransition'); +function Transitionable(start) { + this.currentAction = null; + this.actionQueue = []; + this.callbackQueue = []; + this.state = 0; + this.velocity = undefined; + this._callback = undefined; + this._engineInstance = null; + this._currentMethod = null; + this.set(start); +} +var transitionMethods = {}; +Transitionable.register = function register(methods) { + var success = true; + for (var method in methods) { + if (!Transitionable.registerMethod(method, methods[method])) + success = false; + } + return success; +}; +Transitionable.registerMethod = function registerMethod(name, engineClass) { + if (!(name in transitionMethods)) { + transitionMethods[name] = engineClass; + return true; + } else + return false; +}; +Transitionable.unregisterMethod = function unregisterMethod(name) { + if (name in transitionMethods) { + delete transitionMethods[name]; + return true; + } else + return false; +}; +function _loadNext() { + if (this._callback) { + var callback = this._callback; + this._callback = undefined; + callback(); + } + if (this.actionQueue.length <= 0) { + this.set(this.get()); + return; + } + this.currentAction = this.actionQueue.shift(); + this._callback = this.callbackQueue.shift(); + var method = null; + var endValue = this.currentAction[0]; + var transition = this.currentAction[1]; + if (transition instanceof Object && transition.method) { + method = transition.method; + if (typeof method === 'string') + method = transitionMethods[method]; + } else { + method = TweenTransition; + } + if (this._currentMethod !== method) { + if (!(endValue instanceof Object) || method.SUPPORTS_MULTIPLE === true || endValue.length <= method.SUPPORTS_MULTIPLE) { + this._engineInstance = new method(); + } else { + this._engineInstance = new MultipleTransition(method); + } + this._currentMethod = method; + } + this._engineInstance.reset(this.state, this.velocity); + if (this.velocity !== undefined) + transition.velocity = this.velocity; + this._engineInstance.set(endValue, transition, _loadNext.bind(this)); +} +Transitionable.prototype.set = function set(endState, transition, callback) { + if (!transition) { + this.reset(endState); + if (callback) + callback(); + return this; + } + var action = [ + endState, + transition + ]; + this.actionQueue.push(action); + this.callbackQueue.push(callback); + if (!this.currentAction) + _loadNext.call(this); + return this; +}; +Transitionable.prototype.reset = function reset(startState, startVelocity) { + this._currentMethod = null; + this._engineInstance = null; + this._callback = undefined; + this.state = startState; + this.velocity = startVelocity; + this.currentAction = null; + this.actionQueue = []; + this.callbackQueue = []; +}; +Transitionable.prototype.delay = function delay(duration, callback) { + var endValue; + if (this.actionQueue.length) + endValue = this.actionQueue[this.actionQueue.length - 1][0]; + else if (this.currentAction) + endValue = this.currentAction[0]; + else + endValue = this.get(); + return this.set(endValue, { + duration: duration, + curve: function () { + return 0; + } + }, callback); +}; +Transitionable.prototype.get = function get(timestamp) { + if (this._engineInstance) { + if (this._engineInstance.getVelocity) + this.velocity = this._engineInstance.getVelocity(); + this.state = this._engineInstance.get(timestamp); + } + return this.state; +}; +Transitionable.prototype.isActive = function isActive() { + return !!this.currentAction; +}; +Transitionable.prototype.halt = function halt() { + return this.set(this.get()); +}; +module.exports = Transitionable; +},{"./MultipleTransition":85,"./TweenTransition":90}],89:[function(_dereq_,module,exports){ +var Transitionable = _dereq_('./Transitionable'); +var Transform = _dereq_('../core/Transform'); +var Utility = _dereq_('../utilities/Utility'); +function TransitionableTransform(transform) { + this._final = Transform.identity.slice(); + this._finalTranslate = [ + 0, + 0, + 0 + ]; + this._finalRotate = [ + 0, + 0, + 0 + ]; + this._finalSkew = [ + 0, + 0, + 0 + ]; + this._finalScale = [ + 1, + 1, + 1 + ]; + this.translate = new Transitionable(this._finalTranslate); + this.rotate = new Transitionable(this._finalRotate); + this.skew = new Transitionable(this._finalSkew); + this.scale = new Transitionable(this._finalScale); + if (transform) + this.set(transform); +} +function _build() { + return Transform.build({ + translate: this.translate.get(), + rotate: this.rotate.get(), + skew: this.skew.get(), + scale: this.scale.get() + }); +} +function _buildFinal() { + return Transform.build({ + translate: this._finalTranslate, + rotate: this._finalRotate, + skew: this._finalSkew, + scale: this._finalScale + }); +} +TransitionableTransform.prototype.setTranslate = function setTranslate(translate, transition, callback) { + this._finalTranslate = translate; + this._final = _buildFinal.call(this); + this.translate.set(translate, transition, callback); + return this; +}; +TransitionableTransform.prototype.setScale = function setScale(scale, transition, callback) { + this._finalScale = scale; + this._final = _buildFinal.call(this); + this.scale.set(scale, transition, callback); + return this; +}; +TransitionableTransform.prototype.setRotate = function setRotate(eulerAngles, transition, callback) { + this._finalRotate = eulerAngles; + this._final = _buildFinal.call(this); + this.rotate.set(eulerAngles, transition, callback); + return this; +}; +TransitionableTransform.prototype.setSkew = function setSkew(skewAngles, transition, callback) { + this._finalSkew = skewAngles; + this._final = _buildFinal.call(this); + this.skew.set(skewAngles, transition, callback); + return this; +}; +TransitionableTransform.prototype.set = function set(transform, transition, callback) { + var components = Transform.interpret(transform); + this._finalTranslate = components.translate; + this._finalRotate = components.rotate; + this._finalSkew = components.skew; + this._finalScale = components.scale; + this._final = transform; + var _callback = callback ? Utility.after(4, callback) : null; + this.translate.set(components.translate, transition, _callback); + this.rotate.set(components.rotate, transition, _callback); + this.skew.set(components.skew, transition, _callback); + this.scale.set(components.scale, transition, _callback); + return this; +}; +TransitionableTransform.prototype.setDefaultTransition = function setDefaultTransition(transition) { + this.translate.setDefault(transition); + this.rotate.setDefault(transition); + this.skew.setDefault(transition); + this.scale.setDefault(transition); +}; +TransitionableTransform.prototype.get = function get() { + if (this.isActive()) { + return _build.call(this); + } else + return this._final; +}; +TransitionableTransform.prototype.getFinal = function getFinal() { + return this._final; +}; +TransitionableTransform.prototype.isActive = function isActive() { + return this.translate.isActive() || this.rotate.isActive() || this.scale.isActive() || this.skew.isActive(); +}; +TransitionableTransform.prototype.halt = function halt() { + this.translate.halt(); + this.rotate.halt(); + this.skew.halt(); + this.scale.halt(); + this._final = this.get(); + this._finalTranslate = this.translate.get(); + this._finalRotate = this.rotate.get(); + this._finalSkew = this.skew.get(); + this._finalScale = this.scale.get(); + return this; +}; +module.exports = TransitionableTransform; +},{"../core/Transform":15,"../utilities/Utility":95,"./Transitionable":88}],90:[function(_dereq_,module,exports){ +function TweenTransition(options) { + this.options = Object.create(TweenTransition.DEFAULT_OPTIONS); + if (options) + this.setOptions(options); + this._startTime = 0; + this._startValue = 0; + this._updateTime = 0; + this._endValue = 0; + this._curve = undefined; + this._duration = 0; + this._active = false; + this._callback = undefined; + this.state = 0; + this.velocity = undefined; +} +TweenTransition.Curves = { + linear: function (t) { + return t; + }, + easeIn: function (t) { + return t * t; + }, + easeOut: function (t) { + return t * (2 - t); + }, + easeInOut: function (t) { + if (t <= 0.5) + return 2 * t * t; + else + return -2 * t * t + 4 * t - 1; + }, + easeOutBounce: function (t) { + return t * (3 - 2 * t); + }, + spring: function (t) { + return (1 - t) * Math.sin(6 * Math.PI * t) + t; + } +}; +TweenTransition.SUPPORTS_MULTIPLE = true; +TweenTransition.DEFAULT_OPTIONS = { + curve: TweenTransition.Curves.linear, + duration: 500, + speed: 0 +}; +var registeredCurves = {}; +TweenTransition.registerCurve = function registerCurve(curveName, curve) { + if (!registeredCurves[curveName]) { + registeredCurves[curveName] = curve; + return true; + } else { + return false; + } +}; +TweenTransition.unregisterCurve = function unregisterCurve(curveName) { + if (registeredCurves[curveName]) { + delete registeredCurves[curveName]; + return true; + } else { + return false; + } +}; +TweenTransition.getCurve = function getCurve(curveName) { + var curve = registeredCurves[curveName]; + if (curve !== undefined) + return curve; + else + throw new Error('curve not registered'); +}; +TweenTransition.getCurves = function getCurves() { + return registeredCurves; +}; +function _interpolate(a, b, t) { + return (1 - t) * a + t * b; +} +function _clone(obj) { + if (obj instanceof Object) { + if (obj instanceof Array) + return obj.slice(0); + else + return Object.create(obj); + } else + return obj; +} +function _normalize(transition, defaultTransition) { + var result = { curve: defaultTransition.curve }; + if (defaultTransition.duration) + result.duration = defaultTransition.duration; + if (defaultTransition.speed) + result.speed = defaultTransition.speed; + if (transition instanceof Object) { + if (transition.duration !== undefined) + result.duration = transition.duration; + if (transition.curve) + result.curve = transition.curve; + if (transition.speed) + result.speed = transition.speed; + } + if (typeof result.curve === 'string') + result.curve = TweenTransition.getCurve(result.curve); + return result; +} +TweenTransition.prototype.setOptions = function setOptions(options) { + if (options.curve !== undefined) + this.options.curve = options.curve; + if (options.duration !== undefined) + this.options.duration = options.duration; + if (options.speed !== undefined) + this.options.speed = options.speed; +}; +TweenTransition.prototype.set = function set(endValue, transition, callback) { + if (!transition) { + this.reset(endValue); + if (callback) + callback(); + return; + } + this._startValue = _clone(this.get()); + transition = _normalize(transition, this.options); + if (transition.speed) { + var startValue = this._startValue; + if (startValue instanceof Object) { + var variance = 0; + for (var i in startValue) + variance += (endValue[i] - startValue[i]) * (endValue[i] - startValue[i]); + transition.duration = Math.sqrt(variance) / transition.speed; + } else { + transition.duration = Math.abs(endValue - startValue) / transition.speed; + } + } + this._startTime = Date.now(); + this._endValue = _clone(endValue); + this._startVelocity = _clone(transition.velocity); + this._duration = transition.duration; + this._curve = transition.curve; + this._active = true; + this._callback = callback; +}; +TweenTransition.prototype.reset = function reset(startValue, startVelocity) { + if (this._callback) { + var callback = this._callback; + this._callback = undefined; + callback(); + } + this.state = _clone(startValue); + this.velocity = _clone(startVelocity); + this._startTime = 0; + this._duration = 0; + this._updateTime = 0; + this._startValue = this.state; + this._startVelocity = this.velocity; + this._endValue = this.state; + this._active = false; +}; +TweenTransition.prototype.getVelocity = function getVelocity() { + return this.velocity; +}; +TweenTransition.prototype.get = function get(timestamp) { + this.update(timestamp); + return this.state; +}; +function _calculateVelocity(current, start, curve, duration, t) { + var velocity; + var eps = 1e-7; + var speed = (curve(t) - curve(t - eps)) / eps; + if (current instanceof Array) { + velocity = []; + for (var i = 0; i < current.length; i++) { + if (typeof current[i] === 'number') + velocity[i] = speed * (current[i] - start[i]) / duration; + else + velocity[i] = 0; + } + } else + velocity = speed * (current - start) / duration; + return velocity; +} +function _calculateState(start, end, t) { + var state; + if (start instanceof Array) { + state = []; + for (var i = 0; i < start.length; i++) { + if (typeof start[i] === 'number') + state[i] = _interpolate(start[i], end[i], t); + else + state[i] = start[i]; + } + } else + state = _interpolate(start, end, t); + return state; +} +TweenTransition.prototype.update = function update(timestamp) { + if (!this._active) { + if (this._callback) { + var callback = this._callback; + this._callback = undefined; + callback(); + } + return; + } + if (!timestamp) + timestamp = Date.now(); + if (this._updateTime >= timestamp) + return; + this._updateTime = timestamp; + var timeSinceStart = timestamp - this._startTime; + if (timeSinceStart >= this._duration) { + this.state = this._endValue; + this.velocity = _calculateVelocity(this.state, this._startValue, this._curve, this._duration, 1); + this._active = false; + } else if (timeSinceStart < 0) { + this.state = this._startValue; + this.velocity = this._startVelocity; + } else { + var t = timeSinceStart / this._duration; + this.state = _calculateState(this._startValue, this._endValue, this._curve(t)); + this.velocity = _calculateVelocity(this.state, this._startValue, this._curve, this._duration, t); + } +}; +TweenTransition.prototype.isActive = function isActive() { + return this._active; +}; +TweenTransition.prototype.halt = function halt() { + this.reset(this.get()); +}; +TweenTransition.registerCurve('linear', TweenTransition.Curves.linear); +TweenTransition.registerCurve('easeIn', TweenTransition.Curves.easeIn); +TweenTransition.registerCurve('easeOut', TweenTransition.Curves.easeOut); +TweenTransition.registerCurve('easeInOut', TweenTransition.Curves.easeInOut); +TweenTransition.registerCurve('easeOutBounce', TweenTransition.Curves.easeOutBounce); +TweenTransition.registerCurve('spring', TweenTransition.Curves.spring); +TweenTransition.customCurve = function customCurve(v1, v2) { + v1 = v1 || 0; + v2 = v2 || 0; + return function (t) { + return v1 * t + (-2 * v1 - v2 + 3) * t * t + (v1 + v2 - 2) * t * t * t; + }; +}; +module.exports = TweenTransition; +},{}],91:[function(_dereq_,module,exports){ +var PE = _dereq_('../physics/PhysicsEngine'); +var Particle = _dereq_('../physics/bodies/Particle'); +var Spring = _dereq_('../physics/forces/Spring'); +var Wall = _dereq_('../physics/constraints/Wall'); +var Vector = _dereq_('../math/Vector'); +function WallTransition(state) { + state = state || 0; + this.endState = new Vector(state); + this.initState = new Vector(); + this.spring = new Spring({ anchor: this.endState }); + this.wall = new Wall(); + this._restTolerance = 1e-10; + this._dimensions = 1; + this._absRestTolerance = this._restTolerance; + this._callback = undefined; + this.PE = new PE(); + this.particle = new Particle(); + this.PE.addBody(this.particle); + this.PE.attach([ + this.wall, + this.spring + ], this.particle); +} +WallTransition.SUPPORTS_MULTIPLE = 3; +WallTransition.DEFAULT_OPTIONS = { + period: 300, + dampingRatio: 0.5, + velocity: 0, + restitution: 0.5 +}; +function _getEnergy() { + return this.particle.getEnergy() + this.spring.getEnergy([this.particle]); +} +function _setAbsoluteRestTolerance() { + var distance = this.endState.sub(this.initState).normSquared(); + this._absRestTolerance = distance === 0 ? this._restTolerance : this._restTolerance * distance; +} +function _wake() { + this.PE.wake(); +} +function _sleep() { + this.PE.sleep(); +} +function _setTarget(target) { + this.endState.set(target); + var dist = this.endState.sub(this.initState).norm(); + this.wall.setOptions({ + distance: this.endState.norm(), + normal: dist === 0 ? this.particle.velocity.normalize(-1) : this.endState.sub(this.initState).normalize(-1) + }); + _setAbsoluteRestTolerance.call(this); +} +function _setParticlePosition(p) { + this.particle.position.set(p); +} +function _setParticleVelocity(v) { + this.particle.velocity.set(v); +} +function _getParticlePosition() { + return this._dimensions === 0 ? this.particle.getPosition1D() : this.particle.getPosition(); +} +function _getParticleVelocity() { + return this._dimensions === 0 ? this.particle.getVelocity1D() : this.particle.getVelocity(); +} +function _setCallback(callback) { + this._callback = callback; +} +function _update() { + if (this.PE.isSleeping()) { + if (this._callback) { + var cb = this._callback; + this._callback = undefined; + cb(); + } + return; + } + var energy = _getEnergy.call(this); + if (energy < this._absRestTolerance) { + _sleep.call(this); + _setParticlePosition.call(this, this.endState); + _setParticleVelocity.call(this, [ + 0, + 0, + 0 + ]); + } +} +function _setupDefinition(def) { + var defaults = WallTransition.DEFAULT_OPTIONS; + if (def.period === undefined) + def.period = defaults.period; + if (def.dampingRatio === undefined) + def.dampingRatio = defaults.dampingRatio; + if (def.velocity === undefined) + def.velocity = defaults.velocity; + if (def.restitution === undefined) + def.restitution = defaults.restitution; + if (def.drift === undefined) + def.drift = Wall.DEFAULT_OPTIONS.drift; + if (def.slop === undefined) + def.slop = Wall.DEFAULT_OPTIONS.slop; + this.spring.setOptions({ + period: def.period, + dampingRatio: def.dampingRatio + }); + this.wall.setOptions({ + restitution: def.restitution, + drift: def.drift, + slop: def.slop + }); + _setParticleVelocity.call(this, def.velocity); +} +WallTransition.prototype.reset = function reset(state, velocity) { + this._dimensions = state instanceof Array ? state.length : 0; + this.initState.set(state); + _setParticlePosition.call(this, state); + if (velocity) + _setParticleVelocity.call(this, velocity); + _setTarget.call(this, state); + _setCallback.call(this, undefined); +}; +WallTransition.prototype.getVelocity = function getVelocity() { + return _getParticleVelocity.call(this); +}; +WallTransition.prototype.setVelocity = function setVelocity(velocity) { + this.call(this, _setParticleVelocity(velocity)); +}; +WallTransition.prototype.isActive = function isActive() { + return !this.PE.isSleeping(); +}; +WallTransition.prototype.halt = function halt() { + this.set(this.get()); +}; +WallTransition.prototype.get = function get() { + _update.call(this); + return _getParticlePosition.call(this); +}; +WallTransition.prototype.set = function set(state, definition, callback) { + if (!definition) { + this.reset(state); + if (callback) + callback(); + return; + } + this._dimensions = state instanceof Array ? state.length : 0; + _wake.call(this); + _setupDefinition.call(this, definition); + _setTarget.call(this, state); + _setCallback.call(this, callback); +}; +module.exports = WallTransition; +},{"../math/Vector":41,"../physics/PhysicsEngine":48,"../physics/bodies/Particle":51,"../physics/constraints/Wall":60,"../physics/forces/Spring":68}],92:[function(_dereq_,module,exports){ +module.exports = { + CachedMap: _dereq_('./CachedMap'), + Easing: _dereq_('./Easing'), + MultipleTransition: _dereq_('./MultipleTransition'), + SnapTransition: _dereq_('./SnapTransition'), + SpringTransition: _dereq_('./SpringTransition'), + Transitionable: _dereq_('./Transitionable'), + TransitionableTransform: _dereq_('./TransitionableTransform'), + TweenTransition: _dereq_('./TweenTransition'), + WallTransition: _dereq_('./WallTransition') +}; + +},{"./CachedMap":83,"./Easing":84,"./MultipleTransition":85,"./SnapTransition":86,"./SpringTransition":87,"./Transitionable":88,"./TransitionableTransform":89,"./TweenTransition":90,"./WallTransition":91}],93:[function(_dereq_,module,exports){ +var KeyCodes = { + 0: 48, + 1: 49, + 2: 50, + 3: 51, + 4: 52, + 5: 53, + 6: 54, + 7: 55, + 8: 56, + 9: 57, + a: 97, + b: 98, + c: 99, + d: 100, + e: 101, + f: 102, + g: 103, + h: 104, + i: 105, + j: 106, + k: 107, + l: 108, + m: 109, + n: 110, + o: 111, + p: 112, + q: 113, + r: 114, + s: 115, + t: 116, + u: 117, + v: 118, + w: 119, + x: 120, + y: 121, + z: 122, + A: 65, + B: 66, + C: 67, + D: 68, + E: 69, + F: 70, + G: 71, + H: 72, + I: 73, + J: 74, + K: 75, + L: 76, + M: 77, + N: 78, + O: 79, + P: 80, + Q: 81, + R: 82, + S: 83, + T: 84, + U: 85, + V: 86, + W: 87, + X: 88, + Y: 89, + Z: 90, + ENTER: 13, + LEFT_ARROW: 37, + RIGHT_ARROW: 39, + UP_ARROW: 38, + DOWN_ARROW: 40, + SPACE: 32, + SHIFT: 16, + TAB: 9 + }; +module.exports = KeyCodes; +},{}],94:[function(_dereq_,module,exports){ +var FamousEngine = _dereq_('../core/Engine'); +var _event = 'prerender'; +var getTime = window.performance && window.performance.now ? function () { + return window.performance.now(); + } : function () { + return Date.now(); + }; +function addTimerFunction(fn) { + FamousEngine.on(_event, fn); + return fn; +} +function setTimeout(fn, duration) { + var t = getTime(); + var callback = function () { + var t2 = getTime(); + if (t2 - t >= duration) { + fn.apply(this, arguments); + FamousEngine.removeListener(_event, callback); + } + }; + return addTimerFunction(callback); +} +function setInterval(fn, duration) { + var t = getTime(); + var callback = function () { + var t2 = getTime(); + if (t2 - t >= duration) { + fn.apply(this, arguments); + t = getTime(); + } + }; + return addTimerFunction(callback); +} +function after(fn, numTicks) { + if (numTicks === undefined) + return undefined; + var callback = function () { + numTicks--; + if (numTicks <= 0) { + fn.apply(this, arguments); + clear(callback); + } + }; + return addTimerFunction(callback); +} +function every(fn, numTicks) { + numTicks = numTicks || 1; + var initial = numTicks; + var callback = function () { + numTicks--; + if (numTicks <= 0) { + fn.apply(this, arguments); + numTicks = initial; + } + }; + return addTimerFunction(callback); +} +function clear(fn) { + FamousEngine.removeListener(_event, fn); +} +function debounce(func, wait) { + var timeout; + var ctx; + var timestamp; + var result; + var args; + return function () { + ctx = this; + args = arguments; + timestamp = getTime(); + var fn = function () { + var last = getTime - timestamp; + if (last < wait) { + timeout = setTimeout(fn, wait - last); + } else { + timeout = null; + result = func.apply(ctx, args); + } + }; + clear(timeout); + timeout = setTimeout(fn, wait); + return result; + }; +} +module.exports = { + setTimeout: setTimeout, + setInterval: setInterval, + debounce: debounce, + after: after, + every: every, + clear: clear +}; +},{"../core/Engine":4}],95:[function(_dereq_,module,exports){ +var Utility = {}; +Utility.Direction = { + X: 0, + Y: 1, + Z: 2 +}; +Utility.after = function after(count, callback) { + var counter = count; + return function () { + counter--; + if (counter === 0) + callback.apply(this, arguments); + }; +}; +Utility.loadURL = function loadURL(url, callback) { + var xhr = new XMLHttpRequest(); + xhr.onreadystatechange = function onreadystatechange() { + if (this.readyState === 4) { + if (callback) + callback(this.responseText); + } + }; + xhr.open('GET', url); + xhr.send(); +}; +Utility.createDocumentFragmentFromHTML = function createDocumentFragmentFromHTML(html) { + var element = document.createElement('div'); + element.innerHTML = html; + var result = document.createDocumentFragment(); + while (element.hasChildNodes()) + result.appendChild(element.firstChild); + return result; +}; +Utility.clone = function clone(b) { + var a; + if (typeof b === 'object') { + a = b instanceof Array ? [] : {}; + for (var key in b) { + if (typeof b[key] === 'object' && b[key] !== null) { + if (b[key] instanceof Array) { + a[key] = new Array(b[key].length); + for (var i = 0; i < b[key].length; i++) { + a[key][i] = Utility.clone(b[key][i]); + } + } else { + a[key] = Utility.clone(b[key]); + } + } else { + a[key] = b[key]; + } + } + } else { + a = b; + } + return a; +}; +module.exports = Utility; +},{}],96:[function(_dereq_,module,exports){ +module.exports = { + KeyCodes: _dereq_('./KeyCodes'), + Timer: _dereq_('./Timer'), + Utility: _dereq_('./Utility') +}; + +},{"./KeyCodes":93,"./Timer":94,"./Utility":95}],97:[function(_dereq_,module,exports){ +var Entity = _dereq_('../core/Entity'); +var Transform = _dereq_('../core/Transform'); +var EventHandler = _dereq_('../core/EventHandler'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function ContextualView(options) { + this.options = Object.create(this.constructor.DEFAULT_OPTIONS || ContextualView.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this._id = Entity.register(this); +} +ContextualView.DEFAULT_OPTIONS = {}; +ContextualView.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +ContextualView.prototype.getOptions = function getOptions(key) { + return this._optionsManager.getOptions(key); +}; +ContextualView.prototype.render = function render() { + return this._id; +}; +ContextualView.prototype.commit = function commit(context) { +}; +module.exports = ContextualView; +},{"../core/Entity":5,"../core/EventHandler":7,"../core/OptionsManager":10,"../core/Transform":15}],98:[function(_dereq_,module,exports){ +var Transform = _dereq_('../core/Transform'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var Utility = _dereq_('../utilities/Utility'); +var SequentialLayout = _dereq_('./SequentialLayout'); +function Deck(options) { + SequentialLayout.apply(this, arguments); + this.state = new Transitionable(0); + this._isOpen = false; + this.setOutputFunction(function (input, offset, index) { + var state = _getState.call(this); + var positionMatrix = this.options.direction === Utility.Direction.X ? Transform.translate(state * offset, 0, 0.001 * (state - 1) * offset) : Transform.translate(0, state * offset, 0.001 * (state - 1) * offset); + var output = input.render(); + if (this.options.stackRotation) { + var amount = this.options.stackRotation * index * (1 - state); + output = { + transform: Transform.rotateZ(amount), + origin: [ + 0.5, + 0.5 + ], + target: output + }; + } + return { + transform: positionMatrix, + size: input.getSize(), + target: output + }; + }); +} +Deck.prototype = Object.create(SequentialLayout.prototype); +Deck.prototype.constructor = Deck; +Deck.DEFAULT_OPTIONS = OptionsManager.patch(SequentialLayout.DEFAULT_OPTIONS, { + transition: { + curve: 'easeOutBounce', + duration: 500 + }, + stackRotation: 0 +}); +Deck.prototype.getSize = function getSize() { + var originalSize = SequentialLayout.prototype.getSize.apply(this, arguments); + var firstSize = this._items ? this._items.get().getSize() : [ + 0, + 0 + ]; + if (!firstSize) + firstSize = [ + 0, + 0 + ]; + var state = _getState.call(this); + var invState = 1 - state; + return [ + firstSize[0] * invState + originalSize[0] * state, + firstSize[1] * invState + originalSize[1] * state + ]; +}; +function _getState(returnFinal) { + if (returnFinal) + return this._isOpen ? 1 : 0; + else + return this.state.get(); +} +function _setState(pos, transition, callback) { + this.state.halt(); + this.state.set(pos, transition, callback); +} +Deck.prototype.isOpen = function isOpen() { + return this._isOpen; +}; +Deck.prototype.open = function open(callback) { + this._isOpen = true; + _setState.call(this, 1, this.options.transition, callback); +}; +Deck.prototype.close = function close(callback) { + this._isOpen = false; + _setState.call(this, 0, this.options.transition, callback); +}; +Deck.prototype.toggle = function toggle(callback) { + if (this._isOpen) + this.close(callback); + else + this.open(callback); +}; +module.exports = Deck; +},{"../core/OptionsManager":10,"../core/Transform":15,"../transitions/Transitionable":88,"../utilities/Utility":95,"./SequentialLayout":110}],99:[function(_dereq_,module,exports){ +var RenderNode = _dereq_('../core/RenderNode'); +var Transform = _dereq_('../core/Transform'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var EventHandler = _dereq_('../core/EventHandler'); +function DrawerLayout(options) { + this.options = Object.create(DrawerLayout.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._position = new Transitionable(0); + this._direction = _getDirectionFromSide(this.options.side); + this._orientation = _getOrientationFromSide(this.options.side); + this._isOpen = false; + this._cachedLength = 0; + this.drawer = new RenderNode(); + this.content = new RenderNode(); + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + this._eventInput.on('update', _handleUpdate.bind(this)); + this._eventInput.on('end', _handleEnd.bind(this)); +} +var DIRECTION_X = 0; +var DIRECTION_Y = 1; +DrawerLayout.SIDES = { + LEFT: 0, + TOP: 1, + RIGHT: 2, + BOTTOM: 3 +}; +DrawerLayout.DEFAULT_OPTIONS = { + side: DrawerLayout.SIDES.LEFT, + drawerLength: 0, + velocityThreshold: 0, + positionThreshold: 0, + transition: true +}; +function _getDirectionFromSide(side) { + var SIDES = DrawerLayout.SIDES; + return side === SIDES.LEFT || side === SIDES.RIGHT ? DIRECTION_X : DIRECTION_Y; +} +function _getOrientationFromSide(side) { + var SIDES = DrawerLayout.SIDES; + return side === SIDES.LEFT || side === SIDES.TOP ? 1 : -1; +} +function _resolveNodeSize(node) { + var options = this.options; + var size; + if (options.drawerLength) + size = options.drawerLength; + else { + var nodeSize = node.getSize(); + size = nodeSize ? nodeSize[this._direction] : options.drawerLength; + } + return this._orientation * size; +} +function _handleUpdate(data) { + var newPosition = this.getPosition() + data.delta; + var MIN_LENGTH; + var MAX_LENGTH; + this._cachedLength = _resolveNodeSize.call(this, this.drawer); + if (this._orientation === 1) { + MIN_LENGTH = 0; + MAX_LENGTH = this._cachedLength; + } else { + MIN_LENGTH = this._cachedLength; + MAX_LENGTH = 0; + } + if (newPosition > MAX_LENGTH) + newPosition = MAX_LENGTH; + else if (newPosition < MIN_LENGTH) + newPosition = MIN_LENGTH; + this.setPosition(newPosition); +} +function _handleEnd(data) { + var velocity = data.velocity; + var position = this._orientation * this.getPosition(); + var options = this.options; + var MAX_LENGTH = this._orientation * this._cachedLength; + var positionThreshold = options.positionThreshold || MAX_LENGTH / 2; + var velocityThreshold = options.velocityThreshold; + if (options.transition instanceof Object) + options.transition.velocity = data.velocity; + if (position === 0) { + this._isOpen = false; + return; + } + if (position === MAX_LENGTH) { + this._isOpen = true; + return; + } + var shouldToggle = Math.abs(velocity) > velocityThreshold || !this._isOpen && position > positionThreshold || this._isOpen && position < positionThreshold; + if (shouldToggle) + this.toggle(); + else + this.reset(); +} +DrawerLayout.prototype.setOptions = function setOptions(options) { + this._optionsManager.setOptions(options); + if (options.side !== undefined) { + this._direction = _getDirectionFromSide(options.side); + this._orientation = _getOrientationFromSide(options.side); + } +}; +DrawerLayout.prototype.open = function open(transition, callback) { + if (transition instanceof Function) + callback = transition; + if (transition === undefined) + transition = this.options.transition; + this._cachedLength = _resolveNodeSize.call(this, this.drawer); + this.setPosition(this._cachedLength, transition, callback); + if (!this._isOpen) { + this._isOpen = true; + this._eventOutput.emit('open'); + } +}; +DrawerLayout.prototype.close = function close(transition, callback) { + if (transition instanceof Function) + callback = transition; + if (transition === undefined) + transition = this.options.transition; + this.setPosition(0, transition, callback); + if (this._isOpen) { + this._isOpen = false; + this._eventOutput.emit('close'); + } +}; +DrawerLayout.prototype.setPosition = function setPosition(position, transition, callback) { + if (this._position.isActive()) + this._position.halt(); + this._position.set(position, transition, callback); +}; +DrawerLayout.prototype.getPosition = function getPosition() { + return this._position.get(); +}; +DrawerLayout.prototype.setProgress = function setProgress(progress, transition, callback) { + return this._position.set(progress * this._cachedLength, transition, callback); +}; +DrawerLayout.prototype.getProgress = function getProgress() { + return this._position.get() / this._cachedLength; +}; +DrawerLayout.prototype.toggle = function toggle(transition) { + if (this._isOpen) + this.close(transition); + else + this.open(transition); +}; +DrawerLayout.prototype.reset = function reset(transition) { + if (this._isOpen) + this.open(transition); + else + this.close(transition); +}; +DrawerLayout.prototype.isOpen = function isOpen(transition) { + return this._isOpen; +}; +DrawerLayout.prototype.render = function render() { + var position = this.getPosition(); + if (!this._isOpen && (position < 0 && this._orientation === 1) || position > 0 && this._orientation === -1) { + position = 0; + this.setPosition(position); + } + var contentTransform = this._direction === DIRECTION_X ? Transform.translate(position, 0, 0) : Transform.translate(0, position, 0); + return [ + { + transform: Transform.behind, + target: this.drawer.render() + }, + { + transform: contentTransform, + target: this.content.render() + } + ]; +}; +module.exports = DrawerLayout; +},{"../core/EventHandler":7,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15,"../transitions/Transitionable":88}],100:[function(_dereq_,module,exports){ +var CachedMap = _dereq_('../transitions/CachedMap'); +var Entity = _dereq_('../core/Entity'); +var EventHandler = _dereq_('../core/EventHandler'); +var Transform = _dereq_('../core/Transform'); +var RenderController = _dereq_('./RenderController'); +function EdgeSwapper(options) { + this._currentTarget = null; + this._size = [ + undefined, + undefined + ]; + this._controller = new RenderController(options); + this._controller.inTransformFrom(CachedMap.create(_transformMap.bind(this, 0.0001))); + this._controller.outTransformFrom(CachedMap.create(_transformMap.bind(this, -0.0001))); + this._eventInput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + this._entityId = Entity.register(this); + if (options) + this.setOptions(options); +} +function _transformMap(zMax, progress) { + return Transform.translate(this._size[0] * (1 - progress), 0, zMax * (1 - progress)); +} +EdgeSwapper.prototype.show = function show(content) { + if (this._currentTarget) + this._eventInput.unpipe(this._currentTarget); + this._currentTarget = content; + if (this._currentTarget && this._currentTarget.trigger) + this._eventInput.pipe(this._currentTarget); + this._controller.show.apply(this._controller, arguments); +}; +EdgeSwapper.prototype.setOptions = function setOptions(options) { + this._controller.setOptions(options); +}; +EdgeSwapper.prototype.render = function render() { + return this._entityId; +}; +EdgeSwapper.prototype.commit = function commit(context) { + this._size[0] = context.size[0]; + this._size[1] = context.size[1]; + return { + transform: context.transform, + opacity: context.opacity, + origin: context.origin, + size: context.size, + target: this._controller.render() + }; +}; +module.exports = EdgeSwapper; +},{"../core/Entity":5,"../core/EventHandler":7,"../core/Transform":15,"../transitions/CachedMap":83,"./RenderController":106}],101:[function(_dereq_,module,exports){ +var Entity = _dereq_('../core/Entity'); +var Transform = _dereq_('../core/Transform'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var EventHandler = _dereq_('../core/EventHandler'); +var Transitionable = _dereq_('../transitions/Transitionable'); +function FlexibleLayout(options) { + this.options = Object.create(FlexibleLayout.DEFAULT_OPTIONS); + this.optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this.id = Entity.register(this); + this._ratios = new Transitionable(this.options.ratios); + this._nodes = []; + this._size = [ + 0, + 0 + ]; + this._cachedDirection = null; + this._cachedLengths = []; + this._cachedTransforms = null; + this._ratiosDirty = false; + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); +} +FlexibleLayout.DIRECTION_X = 0; +FlexibleLayout.DIRECTION_Y = 1; +FlexibleLayout.DEFAULT_OPTIONS = { + direction: FlexibleLayout.DIRECTION_X, + transition: false, + ratios: [] +}; +function _reflow(ratios, length, direction) { + var currTransform; + var translation = 0; + var flexLength = length; + var ratioSum = 0; + var ratio; + var node; + var i; + this._cachedLengths = []; + this._cachedTransforms = []; + for (i = 0; i < ratios.length; i++) { + ratio = ratios[i]; + node = this._nodes[i]; + if (typeof ratio !== 'number') + flexLength -= node.getSize()[direction] || 0; + else + ratioSum += ratio; + } + for (i = 0; i < ratios.length; i++) { + node = this._nodes[i]; + ratio = ratios[i]; + length = typeof ratio === 'number' ? flexLength * ratio / ratioSum : node.getSize()[direction]; + currTransform = direction === FlexibleLayout.DIRECTION_X ? Transform.translate(translation, 0, 0) : Transform.translate(0, translation, 0); + this._cachedTransforms.push(currTransform); + this._cachedLengths.push(length); + translation += length; + } +} +function _trueSizedDirty(ratios, direction) { + for (var i = 0; i < ratios.length; i++) { + if (typeof ratios[i] !== 'number') { + if (this._nodes[i].getSize()[direction] !== this._cachedLengths[i]) + return true; + } + } + return false; +} +FlexibleLayout.prototype.render = function render() { + return this.id; +}; +FlexibleLayout.prototype.setOptions = function setOptions(options) { + this.optionsManager.setOptions(options); +}; +FlexibleLayout.prototype.sequenceFrom = function sequenceFrom(sequence) { + this._nodes = sequence; + if (this._ratios.get().length === 0) { + var ratios = []; + for (var i = 0; i < this._nodes.length; i++) + ratios.push(1); + this.setRatios(ratios); + } +}; +FlexibleLayout.prototype.setRatios = function setRatios(ratios, transition, callback) { + if (transition === undefined) + transition = this.options.transition; + var currRatios = this._ratios; + if (currRatios.get().length === 0) + transition = undefined; + if (currRatios.isActive()) + currRatios.halt(); + currRatios.set(ratios, transition, callback); + this._ratiosDirty = true; +}; +FlexibleLayout.prototype.getSize = function getSize() { + return this._size; +}; +FlexibleLayout.prototype.commit = function commit(context) { + var parentSize = context.size; + var parentTransform = context.transform; + var parentOrigin = context.origin; + var parentOpacity = context.opacity; + var ratios = this._ratios.get(); + var direction = this.options.direction; + var length = parentSize[direction]; + var size; + if (length !== this._size[direction] || this._ratiosDirty || this._ratios.isActive() || direction !== this._cachedDirection || _trueSizedDirty.call(this, ratios, direction)) { + _reflow.call(this, ratios, length, direction); + if (length !== this._size[direction]) { + this._size[0] = parentSize[0]; + this._size[1] = parentSize[1]; + } + if (direction !== this._cachedDirection) + this._cachedDirection = direction; + if (this._ratiosDirty) + this._ratiosDirty = false; + } + var result = []; + for (var i = 0; i < ratios.length; i++) { + size = [ + undefined, + undefined + ]; + length = this._cachedLengths[i]; + size[direction] = length; + result.push({ + transform: this._cachedTransforms[i], + size: size, + target: this._nodes[i].render() + }); + } + if (parentSize && (parentOrigin[0] !== 0 && parentOrigin[1] !== 0)) + parentTransform = Transform.moveThen([ + -parentSize[0] * parentOrigin[0], + -parentSize[1] * parentOrigin[1], + 0 + ], parentTransform); + return { + transform: parentTransform, + size: parentSize, + opacity: parentOpacity, + target: result + }; +}; +module.exports = FlexibleLayout; +},{"../core/Entity":5,"../core/EventHandler":7,"../core/OptionsManager":10,"../core/Transform":15,"../transitions/Transitionable":88}],102:[function(_dereq_,module,exports){ +var Transform = _dereq_('../core/Transform'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var RenderNode = _dereq_('../core/RenderNode'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function Flipper(options) { + this.options = Object.create(Flipper.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this.angle = new Transitionable(0); + this.frontNode = undefined; + this.backNode = undefined; + this.flipped = false; +} +Flipper.DIRECTION_X = 0; +Flipper.DIRECTION_Y = 1; +var SEPERATION_LENGTH = 1; +Flipper.DEFAULT_OPTIONS = { + transition: true, + direction: Flipper.DIRECTION_X +}; +Flipper.prototype.flip = function flip(transition, callback) { + var angle = this.flipped ? 0 : Math.PI; + this.setAngle(angle, transition, callback); + this.flipped = !this.flipped; +}; +Flipper.prototype.setAngle = function setAngle(angle, transition, callback) { + if (transition === undefined) + transition = this.options.transition; + if (this.angle.isActive()) + this.angle.halt(); + this.angle.set(angle, transition, callback); +}; +Flipper.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +Flipper.prototype.setFront = function setFront(node) { + this.frontNode = node; +}; +Flipper.prototype.setBack = function setBack(node) { + this.backNode = node; +}; +Flipper.prototype.render = function render() { + var angle = this.angle.get(); + var frontTransform; + var backTransform; + if (this.options.direction === Flipper.DIRECTION_X) { + frontTransform = Transform.rotateY(angle); + backTransform = Transform.rotateY(angle + Math.PI); + } else { + frontTransform = Transform.rotateX(angle); + backTransform = Transform.rotateX(angle + Math.PI); + } + var result = []; + if (this.frontNode) { + result.push({ + transform: frontTransform, + target: this.frontNode.render() + }); + } + if (this.backNode) { + result.push({ + transform: Transform.moveThen([ + 0, + 0, + SEPERATION_LENGTH + ], backTransform), + target: this.backNode.render() + }); + } + return result; +}; +module.exports = Flipper; +},{"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15,"../transitions/Transitionable":88}],103:[function(_dereq_,module,exports){ +var Entity = _dereq_('../core/Entity'); +var RenderNode = _dereq_('../core/RenderNode'); +var Transform = _dereq_('../core/Transform'); +var ViewSequence = _dereq_('../core/ViewSequence'); +var EventHandler = _dereq_('../core/EventHandler'); +var Modifier = _dereq_('../core/Modifier'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var TransitionableTransform = _dereq_('../transitions/TransitionableTransform'); +function GridLayout(options) { + this.options = Object.create(GridLayout.DEFAULT_OPTIONS); + this.optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this.id = Entity.register(this); + this._modifiers = []; + this._states = []; + this._contextSizeCache = [ + 0, + 0 + ]; + this._dimensionsCache = [ + 0, + 0 + ]; + this._activeCount = 0; + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); +} +function _reflow(size, cols, rows) { + var usableSize = [ + size[0], + size[1] + ]; + usableSize[0] -= this.options.gutterSize[0] * (cols - 1); + usableSize[1] -= this.options.gutterSize[1] * (rows - 1); + var rowSize = Math.round(usableSize[1] / rows); + var colSize = Math.round(usableSize[0] / cols); + var currY = 0; + var currX; + var currIndex = 0; + for (var i = 0; i < rows; i++) { + currX = 0; + for (var j = 0; j < cols; j++) { + if (this._modifiers[currIndex] === undefined) { + _createModifier.call(this, currIndex, [ + colSize, + rowSize + ], [ + currX, + currY, + 0 + ], 1); + } else { + _animateModifier.call(this, currIndex, [ + colSize, + rowSize + ], [ + currX, + currY, + 0 + ], 1); + } + currIndex++; + currX += colSize + this.options.gutterSize[0]; + } + currY += rowSize + this.options.gutterSize[1]; + } + this._dimensionsCache = [ + this.options.dimensions[0], + this.options.dimensions[1] + ]; + this._contextSizeCache = [ + size[0], + size[1] + ]; + this._activeCount = rows * cols; + for (i = this._activeCount; i < this._modifiers.length; i++) + _animateModifier.call(this, i, [ + Math.round(colSize), + Math.round(rowSize) + ], [ + 0, + 0 + ], 0); + this._eventOutput.emit('reflow'); +} +function _createModifier(index, size, position, opacity) { + var transitionItem = { + transform: new TransitionableTransform(Transform.translate.apply(null, position)), + opacity: new Transitionable(opacity), + size: new Transitionable(size) + }; + var modifier = new Modifier({ + transform: transitionItem.transform, + opacity: transitionItem.opacity, + size: transitionItem.size + }); + this._states[index] = transitionItem; + this._modifiers[index] = modifier; +} +function _animateModifier(index, size, position, opacity) { + var currState = this._states[index]; + var currSize = currState.size; + var currOpacity = currState.opacity; + var currTransform = currState.transform; + var transition = this.options.transition; + currTransform.halt(); + currOpacity.halt(); + currSize.halt(); + currTransform.setTranslate(position, transition); + currSize.set(size, transition); + currOpacity.set(opacity, transition); +} +GridLayout.DEFAULT_OPTIONS = { + dimensions: [ + 1, + 1 + ], + transition: false, + gutterSize: [ + 0, + 0 + ] +}; +GridLayout.prototype.render = function render() { + return this.id; +}; +GridLayout.prototype.setOptions = function setOptions(options) { + return this.optionsManager.setOptions(options); +}; +GridLayout.prototype.sequenceFrom = function sequenceFrom(sequence) { + if (sequence instanceof Array) + sequence = new ViewSequence(sequence); + this.sequence = sequence; +}; +GridLayout.prototype.getSize = function getSize() { + return this._contextSizeCache; +}; +GridLayout.prototype.commit = function commit(context) { + var transform = context.transform; + var opacity = context.opacity; + var origin = context.origin; + var size = context.size; + var cols = this.options.dimensions[0]; + var rows = this.options.dimensions[1]; + if (size[0] !== this._contextSizeCache[0] || size[1] !== this._contextSizeCache[1] || cols !== this._dimensionsCache[0] || rows !== this._dimensionsCache[1]) { + _reflow.call(this, size, cols, rows); + } + var sequence = this.sequence; + var result = []; + var currIndex = 0; + while (sequence && currIndex < this._modifiers.length) { + var item = sequence.get(); + var modifier = this._modifiers[currIndex]; + if (currIndex >= this._activeCount && this._states[currIndex].opacity.isActive()) { + this._modifiers.splice(currIndex, 1); + this._states.splice(currIndex, 1); + } + if (item) { + result.push(modifier.modify({ + origin: origin, + target: item.render() + })); + } + sequence = sequence.getNext(); + currIndex++; + } + if (size) + transform = Transform.moveThen([ + -size[0] * origin[0], + -size[1] * origin[1], + 0 + ], transform); + return { + transform: transform, + opacity: opacity, + size: size, + target: result + }; +}; +module.exports = GridLayout; +},{"../core/Entity":5,"../core/EventHandler":7,"../core/Modifier":9,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15,"../core/ViewSequence":17,"../transitions/Transitionable":88,"../transitions/TransitionableTransform":89}],104:[function(_dereq_,module,exports){ +var Entity = _dereq_('../core/Entity'); +var RenderNode = _dereq_('../core/RenderNode'); +var Transform = _dereq_('../core/Transform'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function HeaderFooterLayout(options) { + this.options = Object.create(HeaderFooterLayout.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._entityId = Entity.register(this); + this.header = new RenderNode(); + this.footer = new RenderNode(); + this.content = new RenderNode(); +} +HeaderFooterLayout.DIRECTION_X = 0; +HeaderFooterLayout.DIRECTION_Y = 1; +HeaderFooterLayout.DEFAULT_OPTIONS = { + direction: HeaderFooterLayout.DIRECTION_Y, + headerSize: undefined, + footerSize: undefined, + defaultHeaderSize: 0, + defaultFooterSize: 0 +}; +HeaderFooterLayout.prototype.render = function render() { + return this._entityId; +}; +HeaderFooterLayout.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +function _resolveNodeSize(node, defaultSize) { + var nodeSize = node.getSize(); + return nodeSize ? nodeSize[this.options.direction] : defaultSize; +} +function _outputTransform(offset) { + if (this.options.direction === HeaderFooterLayout.DIRECTION_X) + return Transform.translate(offset, 0, 0); + else + return Transform.translate(0, offset, 0); +} +function _finalSize(directionSize, size) { + if (this.options.direction === HeaderFooterLayout.DIRECTION_X) + return [ + directionSize, + size[1] + ]; + else + return [ + size[0], + directionSize + ]; +} +HeaderFooterLayout.prototype.commit = function commit(context) { + var transform = context.transform; + var origin = context.origin; + var size = context.size; + var opacity = context.opacity; + var headerSize = this.options.headerSize !== undefined ? this.options.headerSize : _resolveNodeSize.call(this, this.header, this.options.defaultHeaderSize); + var footerSize = this.options.footerSize !== undefined ? this.options.footerSize : _resolveNodeSize.call(this, this.footer, this.options.defaultFooterSize); + var contentSize = size[this.options.direction] - headerSize - footerSize; + if (size) + transform = Transform.moveThen([ + -size[0] * origin[0], + -size[1] * origin[1], + 0 + ], transform); + var result = [ + { + size: _finalSize.call(this, headerSize, size), + target: this.header.render() + }, + { + transform: _outputTransform.call(this, headerSize), + size: _finalSize.call(this, contentSize, size), + target: this.content.render() + }, + { + transform: _outputTransform.call(this, headerSize + contentSize), + size: _finalSize.call(this, footerSize, size), + target: this.footer.render() + } + ]; + return { + transform: transform, + opacity: opacity, + size: size, + target: result + }; +}; +module.exports = HeaderFooterLayout; +},{"../core/Entity":5,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15}],105:[function(_dereq_,module,exports){ +var Transform = _dereq_('../core/Transform'); +var Modifier = _dereq_('../core/Modifier'); +var RenderNode = _dereq_('../core/RenderNode'); +var Utility = _dereq_('../utilities/Utility'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var TransitionableTransform = _dereq_('../transitions/TransitionableTransform'); +function Lightbox(options) { + this.options = Object.create(Lightbox.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this._showing = false; + this.nodes = []; + this.transforms = []; + this.states = []; +} +Lightbox.DEFAULT_OPTIONS = { + inTransform: Transform.scale(0.001, 0.001, 0.001), + inOpacity: 0, + inOrigin: [ + 0.5, + 0.5 + ], + inAlign: [ + 0.5, + 0.5 + ], + outTransform: Transform.scale(0.001, 0.001, 0.001), + outOpacity: 0, + outOrigin: [ + 0.5, + 0.5 + ], + outAlign: [ + 0.5, + 0.5 + ], + showTransform: Transform.identity, + showOpacity: 1, + showOrigin: [ + 0.5, + 0.5 + ], + showAlign: [ + 0.5, + 0.5 + ], + inTransition: true, + outTransition: true, + overlap: false +}; +Lightbox.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +Lightbox.prototype.show = function show(renderable, transition, callback) { + if (!renderable) { + return this.hide(callback); + } + if (transition instanceof Function) { + callback = transition; + transition = undefined; + } + if (this._showing) { + if (this.options.overlap) + this.hide(); + else { + return this.hide(this.show.bind(this, renderable, transition, callback)); + } + } + this._showing = true; + var stateItem = { + transform: new TransitionableTransform(this.options.inTransform), + origin: new Transitionable(this.options.inOrigin), + align: new Transitionable(this.options.inAlign), + opacity: new Transitionable(this.options.inOpacity) + }; + var transform = new Modifier({ + transform: stateItem.transform, + opacity: stateItem.opacity, + origin: stateItem.origin, + align: stateItem.align + }); + var node = new RenderNode(); + node.add(transform).add(renderable); + this.nodes.push(node); + this.states.push(stateItem); + this.transforms.push(transform); + var _cb = callback ? Utility.after(3, callback) : undefined; + if (!transition) + transition = this.options.inTransition; + stateItem.transform.set(this.options.showTransform, transition, _cb); + stateItem.opacity.set(this.options.showOpacity, transition, _cb); + stateItem.origin.set(this.options.showOrigin, transition, _cb); + stateItem.align.set(this.options.showAlign, transition, _cb); +}; +Lightbox.prototype.hide = function hide(transition, callback) { + if (!this._showing) + return; + this._showing = false; + if (transition instanceof Function) { + callback = transition; + transition = undefined; + } + var node = this.nodes[this.nodes.length - 1]; + var transform = this.transforms[this.transforms.length - 1]; + var stateItem = this.states[this.states.length - 1]; + var _cb = Utility.after(3, function () { + this.nodes.splice(this.nodes.indexOf(node), 1); + this.states.splice(this.states.indexOf(stateItem), 1); + this.transforms.splice(this.transforms.indexOf(transform), 1); + if (callback) + callback.call(this); + }.bind(this)); + if (!transition) + transition = this.options.outTransition; + stateItem.transform.set(this.options.outTransform, transition, _cb); + stateItem.opacity.set(this.options.outOpacity, transition, _cb); + stateItem.origin.set(this.options.outOrigin, transition, _cb); + stateItem.align.set(this.options.outAlign, transition, _cb); +}; +Lightbox.prototype.render = function render() { + var result = []; + for (var i = 0; i < this.nodes.length; i++) { + result.push(this.nodes[i].render()); + } + return result; +}; +module.exports = Lightbox; +},{"../core/Modifier":9,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15,"../transitions/Transitionable":88,"../transitions/TransitionableTransform":89,"../utilities/Utility":95}],106:[function(_dereq_,module,exports){ +var Modifier = _dereq_('../core/Modifier'); +var RenderNode = _dereq_('../core/RenderNode'); +var Transform = _dereq_('../core/Transform'); +var Transitionable = _dereq_('../transitions/Transitionable'); +var View = _dereq_('../core/View'); +function RenderController(options) { + View.apply(this, arguments); + this._showing = -1; + this._outgoingRenderables = []; + this._nextRenderable = null; + this._renderables = []; + this._nodes = []; + this._modifiers = []; + this._states = []; + this.inTransformMap = RenderController.DefaultMap.transform; + this.inOpacityMap = RenderController.DefaultMap.opacity; + this.inOriginMap = RenderController.DefaultMap.origin; + this.inAlignMap = RenderController.DefaultMap.align; + this.outTransformMap = RenderController.DefaultMap.transform; + this.outOpacityMap = RenderController.DefaultMap.opacity; + this.outOriginMap = RenderController.DefaultMap.origin; + this.outAlignMap = RenderController.DefaultMap.align; + this._output = []; +} +RenderController.prototype = Object.create(View.prototype); +RenderController.prototype.constructor = RenderController; +RenderController.DEFAULT_OPTIONS = { + inTransition: true, + outTransition: true, + overlap: true +}; +RenderController.DefaultMap = { + transform: function () { + return Transform.identity; + }, + opacity: function (progress) { + return progress; + }, + origin: null, + align: null +}; +function _mappedState(map, state) { + return map(state.get()); +} +RenderController.prototype.inTransformFrom = function inTransformFrom(transform) { + if (transform instanceof Function) + this.inTransformMap = transform; + else if (transform && transform.get) + this.inTransformMap = transform.get.bind(transform); + else + throw new Error('inTransformFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.inOpacityFrom = function inOpacityFrom(opacity) { + if (opacity instanceof Function) + this.inOpacityMap = opacity; + else if (opacity && opacity.get) + this.inOpacityMap = opacity.get.bind(opacity); + else + throw new Error('inOpacityFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.inOriginFrom = function inOriginFrom(origin) { + if (origin instanceof Function) + this.inOriginMap = origin; + else if (origin && origin.get) + this.inOriginMap = origin.get.bind(origin); + else + throw new Error('inOriginFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.inAlignFrom = function inAlignFrom(align) { + if (align instanceof Function) + this.inAlignMap = align; + else if (align && align.get) + this.inAlignMap = align.get.bind(align); + else + throw new Error('inAlignFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.outTransformFrom = function outTransformFrom(transform) { + if (transform instanceof Function) + this.outTransformMap = transform; + else if (transform && transform.get) + this.outTransformMap = transform.get.bind(transform); + else + throw new Error('outTransformFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.outOpacityFrom = function outOpacityFrom(opacity) { + if (opacity instanceof Function) + this.outOpacityMap = opacity; + else if (opacity && opacity.get) + this.outOpacityMap = opacity.get.bind(opacity); + else + throw new Error('outOpacityFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.outOriginFrom = function outOriginFrom(origin) { + if (origin instanceof Function) + this.outOriginMap = origin; + else if (origin && origin.get) + this.outOriginMap = origin.get.bind(origin); + else + throw new Error('outOriginFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.outAlignFrom = function outAlignFrom(align) { + if (align instanceof Function) + this.outAlignMap = align; + else if (align && align.get) + this.outAlignMap = align.get.bind(align); + else + throw new Error('outAlignFrom takes only function or getter object'); + return this; +}; +RenderController.prototype.show = function show(renderable, transition, callback) { + if (!renderable) { + return this.hide(callback); + } + if (transition instanceof Function) { + callback = transition; + transition = null; + } + if (this._showing >= 0) { + if (this.options.overlap) + this.hide(); + else { + if (this._nextRenderable) { + this._nextRenderable = renderable; + } else { + this._nextRenderable = renderable; + this.hide(function () { + if (this._nextRenderable === renderable) + this.show(this._nextRenderable, callback); + this._nextRenderable = null; + }); + } + return undefined; + } + } + var state = null; + var renderableIndex = this._renderables.indexOf(renderable); + if (renderableIndex >= 0) { + this._showing = renderableIndex; + state = this._states[renderableIndex]; + state.halt(); + var outgoingIndex = this._outgoingRenderables.indexOf(renderable); + if (outgoingIndex >= 0) + this._outgoingRenderables.splice(outgoingIndex, 1); + } else { + state = new Transitionable(0); + var modifier = new Modifier({ + transform: this.inTransformMap ? _mappedState.bind(this, this.inTransformMap, state) : null, + opacity: this.inOpacityMap ? _mappedState.bind(this, this.inOpacityMap, state) : null, + origin: this.inOriginMap ? _mappedState.bind(this, this.inOriginMap, state) : null, + align: this.inAlignMap ? _mappedState.bind(this, this.inAlignMap, state) : null + }); + var node = new RenderNode(); + node.add(modifier).add(renderable); + this._showing = this._nodes.length; + this._nodes.push(node); + this._modifiers.push(modifier); + this._states.push(state); + this._renderables.push(renderable); + } + if (!transition) + transition = this.options.inTransition; + state.set(1, transition, callback); +}; +RenderController.prototype.hide = function hide(transition, callback) { + if (this._showing < 0) + return; + var index = this._showing; + this._showing = -1; + if (transition instanceof Function) { + callback = transition; + transition = undefined; + } + var node = this._nodes[index]; + var modifier = this._modifiers[index]; + var state = this._states[index]; + var renderable = this._renderables[index]; + modifier.transformFrom(this.outTransformMap ? _mappedState.bind(this, this.outTransformMap, state) : null); + modifier.opacityFrom(this.outOpacityMap ? _mappedState.bind(this, this.outOpacityMap, state) : null); + modifier.originFrom(this.outOriginMap ? _mappedState.bind(this, this.outOriginMap, state) : null); + modifier.alignFrom(this.outAlignMap ? _mappedState.bind(this, this.outAlignMap, state) : null); + if (this._outgoingRenderables.indexOf(renderable) < 0) + this._outgoingRenderables.push(renderable); + if (!transition) + transition = this.options.outTransition; + state.halt(); + state.set(0, transition, function (node, modifier, state, renderable) { + if (this._outgoingRenderables.indexOf(renderable) >= 0) { + var index = this._nodes.indexOf(node); + this._nodes.splice(index, 1); + this._modifiers.splice(index, 1); + this._states.splice(index, 1); + this._renderables.splice(index, 1); + this._outgoingRenderables.splice(this._outgoingRenderables.indexOf(renderable), 1); + if (this._showing >= index) + this._showing--; + } + if (callback) + callback.call(this); + }.bind(this, node, modifier, state, renderable)); +}; +RenderController.prototype.render = function render() { + var result = this._output; + if (result.length > this._nodes.length) + result.splice(this._nodes.length); + for (var i = 0; i < this._nodes.length; i++) { + result[i] = this._nodes[i].render(); + } + return result; +}; +module.exports = RenderController; +},{"../core/Modifier":9,"../core/RenderNode":11,"../core/Transform":15,"../core/View":16,"../transitions/Transitionable":88}],107:[function(_dereq_,module,exports){ +var ContainerSurface = _dereq_('../surfaces/ContainerSurface'); +var EventHandler = _dereq_('../core/EventHandler'); +var Scrollview = _dereq_('./Scrollview'); +var Utility = _dereq_('../utilities/Utility'); +var OptionsManager = _dereq_('../core/OptionsManager'); +function ScrollContainer(options) { + this.options = Object.create(ScrollContainer.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this.container = new ContainerSurface(this.options.container); + this.scrollview = new Scrollview(this.options.scrollview); + this.container.add(this.scrollview); + this._eventInput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + this._eventInput.pipe(this.scrollview); + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); + this.container.pipe(this._eventOutput); + this.scrollview.pipe(this._eventOutput); +} +ScrollContainer.DEFAULT_OPTIONS = { + container: { properties: { overflow: 'hidden' } }, + scrollview: {} +}; +ScrollContainer.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); +}; +ScrollContainer.prototype.sequenceFrom = function sequenceFrom() { + return this.scrollview.sequenceFrom.apply(this.scrollview, arguments); +}; +ScrollContainer.prototype.getSize = function getSize() { + return this.container.getSize.apply(this.container, arguments); +}; +ScrollContainer.prototype.render = function render() { + return this.container.render(); +}; +module.exports = ScrollContainer; +},{"../core/EventHandler":7,"../core/OptionsManager":10,"../surfaces/ContainerSurface":75,"../utilities/Utility":95,"./Scrollview":109}],108:[function(_dereq_,module,exports){ +var Entity = _dereq_('../core/Entity'); +var Group = _dereq_('../core/Group'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var Transform = _dereq_('../core/Transform'); +var Utility = _dereq_('../utilities/Utility'); +var ViewSequence = _dereq_('../core/ViewSequence'); +var EventHandler = _dereq_('../core/EventHandler'); +function Scroller(options) { + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) + this._optionsManager.setOptions(options); + this._node = null; + this._position = 0; + this._positionOffset = 0; + this._positionGetter = null; + this._outputFunction = null; + this._masterOutputFunction = null; + this.outputFrom(); + this._onEdge = 0; + this.group = new Group(); + this.group.add({ render: _innerRender.bind(this) }); + this._entityId = Entity.register(this); + this._size = [ + undefined, + undefined + ]; + this._contextSize = [ + undefined, + undefined + ]; + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); +} +Scroller.DEFAULT_OPTIONS = { + direction: Utility.Direction.Y, + margin: 0, + clipSize: undefined, + groupScroll: false +}; +var EDGE_TOLERANCE = 0; +function _sizeForDir(size) { + if (!size) + size = this._contextSize; + var dimension = this.options.direction; + return size[dimension] === undefined ? this._contextSize[dimension] : size[dimension]; +} +function _output(node, offset, target) { + var size = node.getSize ? node.getSize() : this._contextSize; + var transform = this._outputFunction(offset); + target.push({ + transform: transform, + target: node.render() + }); + return _sizeForDir.call(this, size); +} +function _getClipSize() { + if (this.options.clipSize !== undefined) + return this.options.clipSize; + if (this._contextSize[this.options.direction] > this.getCumulativeSize()[this.options.direction]) { + return _sizeForDir.call(this, this.getCumulativeSize()); + } else { + return _sizeForDir.call(this, this._contextSize); + } +} +Scroller.prototype.getCumulativeSize = function (index) { + if (index === undefined) + index = this._node._.cumulativeSizes.length - 1; + return this._node._.getSize(index); +}; +Scroller.prototype.setOptions = function setOptions(options) { + if (options.groupScroll !== this.options.groupScroll) { + if (options.groupScroll) + this.group.pipe(this._eventOutput); + else + this.group.unpipe(this._eventOutput); + } + this._optionsManager.setOptions(options); +}; +Scroller.prototype.onEdge = function onEdge() { + return this._onEdge; +}; +Scroller.prototype.outputFrom = function outputFrom(fn, masterFn) { + if (!fn) { + fn = function (offset) { + return this.options.direction === Utility.Direction.X ? Transform.translate(offset, 0) : Transform.translate(0, offset); + }.bind(this); + if (!masterFn) + masterFn = fn; + } + this._outputFunction = fn; + this._masterOutputFunction = masterFn ? masterFn : function (offset) { + return Transform.inverse(fn(-offset)); + }; +}; +Scroller.prototype.positionFrom = function positionFrom(position) { + if (position instanceof Function) + this._positionGetter = position; + else if (position && position.get) + this._positionGetter = position.get.bind(position); + else { + this._positionGetter = null; + this._position = position; + } + if (this._positionGetter) + this._position = this._positionGetter.call(this); +}; +Scroller.prototype.sequenceFrom = function sequenceFrom(node) { + if (node instanceof Array) + node = new ViewSequence({ array: node }); + this._node = node; + this._positionOffset = 0; +}; +Scroller.prototype.getSize = function getSize(actual) { + return actual ? this._contextSize : this._size; +}; +Scroller.prototype.render = function render() { + if (!this._node) + return null; + if (this._positionGetter) + this._position = this._positionGetter.call(this); + return this._entityId; +}; +Scroller.prototype.commit = function commit(context) { + var transform = context.transform; + var opacity = context.opacity; + var origin = context.origin; + var size = context.size; + if (!this.options.clipSize && (size[0] !== this._contextSize[0] || size[1] !== this._contextSize[1])) { + this._onEdge = 0; + this._contextSize[0] = size[0]; + this._contextSize[1] = size[1]; + if (this.options.direction === Utility.Direction.X) { + this._size[0] = _getClipSize.call(this); + this._size[1] = undefined; + } else { + this._size[0] = undefined; + this._size[1] = _getClipSize.call(this); + } + } + var scrollTransform = this._masterOutputFunction(-this._position); + return { + transform: Transform.multiply(transform, scrollTransform), + size: size, + opacity: opacity, + origin: origin, + target: this.group.render() + }; +}; +function _innerRender() { + var size = null; + var position = this._position; + var result = []; + var offset = -this._positionOffset; + var clipSize = _getClipSize.call(this); + var currNode = this._node; + while (currNode && offset - position < clipSize + this.options.margin) { + offset += _output.call(this, currNode, offset, result); + currNode = currNode.getNext ? currNode.getNext() : null; + } + var sizeNode = this._node; + var nodesSize = _sizeForDir.call(this, sizeNode.getSize()); + if (offset < clipSize) { + while (sizeNode && nodesSize < clipSize) { + sizeNode = sizeNode.getPrevious(); + if (sizeNode) + nodesSize += _sizeForDir.call(this, sizeNode.getSize()); + } + sizeNode = this._node; + while (sizeNode && nodesSize < clipSize) { + sizeNode = sizeNode.getNext(); + if (sizeNode) + nodesSize += _sizeForDir.call(this, sizeNode.getSize()); + } + } + if (!currNode && offset - position < clipSize - EDGE_TOLERANCE) { + if (this._onEdge !== 1) { + this._onEdge = 1; + this._eventOutput.emit('onEdge', { position: offset - clipSize }); + } + } else if (!this._node.getPrevious() && position < -EDGE_TOLERANCE) { + if (this._onEdge !== -1) { + this._onEdge = -1; + this._eventOutput.emit('onEdge', { position: 0 }); + } + } else { + if (this._onEdge !== 0) { + this._onEdge = 0; + this._eventOutput.emit('offEdge'); + } + } + currNode = this._node && this._node.getPrevious ? this._node.getPrevious() : null; + offset = -this._positionOffset; + if (currNode) { + size = currNode.getSize ? currNode.getSize() : this._contextSize; + offset -= _sizeForDir.call(this, size); + } + while (currNode && offset - position > -(clipSize + this.options.margin)) { + _output.call(this, currNode, offset, result); + currNode = currNode.getPrevious ? currNode.getPrevious() : null; + if (currNode) { + size = currNode.getSize ? currNode.getSize() : this._contextSize; + offset -= _sizeForDir.call(this, size); + } + } + return result; +} +module.exports = Scroller; +},{"../core/Entity":5,"../core/EventHandler":7,"../core/Group":8,"../core/OptionsManager":10,"../core/Transform":15,"../core/ViewSequence":17,"../utilities/Utility":95}],109:[function(_dereq_,module,exports){ +var PhysicsEngine = _dereq_('../physics/PhysicsEngine'); +var Particle = _dereq_('../physics/bodies/Particle'); +var Drag = _dereq_('../physics/forces/Drag'); +var Spring = _dereq_('../physics/forces/Spring'); +var EventHandler = _dereq_('../core/EventHandler'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var ViewSequence = _dereq_('../core/ViewSequence'); +var Scroller = _dereq_('../views/Scroller'); +var Utility = _dereq_('../utilities/Utility'); +var GenericSync = _dereq_('../inputs/GenericSync'); +var ScrollSync = _dereq_('../inputs/ScrollSync'); +var TouchSync = _dereq_('../inputs/TouchSync'); +GenericSync.register({ + scroll: ScrollSync, + touch: TouchSync +}); +var TOLERANCE = 0.5; +var SpringStates = { + NONE: 0, + EDGE: 1, + PAGE: 2 + }; +var EdgeStates = { + TOP: -1, + NONE: 0, + BOTTOM: 1 + }; +function Scrollview(options) { + this.options = Object.create(Scrollview.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + this._scroller = new Scroller(this.options); + this.sync = new GenericSync([ + 'scroll', + 'touch' + ], { + direction: this.options.direction, + scale: this.options.syncScale, + rails: this.options.rails, + preventDefault: this.options.preventDefault !== undefined ? this.options.preventDefault : this.options.direction !== Utility.Direction.Y + }); + this._physicsEngine = new PhysicsEngine(); + this._particle = new Particle(); + this._physicsEngine.addBody(this._particle); + this.spring = new Spring({ + anchor: [ + 0, + 0, + 0 + ], + period: this.options.edgePeriod, + dampingRatio: this.options.edgeDamp + }); + this.drag = new Drag({ + forceFunction: Drag.FORCE_FUNCTIONS.QUADRATIC, + strength: this.options.drag + }); + this.friction = new Drag({ + forceFunction: Drag.FORCE_FUNCTIONS.LINEAR, + strength: this.options.friction + }); + this._node = null; + this._touchCount = 0; + this._springState = SpringStates.NONE; + this._onEdge = EdgeStates.NONE; + this._pageSpringPosition = 0; + this._edgeSpringPosition = 0; + this._touchVelocity = 0; + this._earlyEnd = false; + this._needsPaginationCheck = false; + this._displacement = 0; + this._totalShift = 0; + this._cachedIndex = 0; + this._scroller.positionFrom(this.getPosition.bind(this)); + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + this._eventInput.pipe(this.sync); + this.sync.pipe(this._eventInput); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + _bindEvents.call(this); + if (options) + this.setOptions(options); +} +Scrollview.DEFAULT_OPTIONS = { + direction: Utility.Direction.Y, + rails: true, + friction: 0.005, + drag: 0.0001, + edgeGrip: 0.2, + edgePeriod: 300, + edgeDamp: 1, + margin: 1000, + paginated: false, + pagePeriod: 500, + pageDamp: 0.8, + pageStopSpeed: 10, + pageSwitchSpeed: 0.5, + speedLimit: 5, + groupScroll: false, + syncScale: 1 +}; +function _handleStart(event) { + this._touchCount = event.count; + if (event.count === undefined) + this._touchCount = 1; + _detachAgents.call(this); + this.setVelocity(0); + this._touchVelocity = 0; + this._earlyEnd = false; +} +function _handleMove(event) { + var velocity = -event.velocity; + var delta = -event.delta; + if (this._onEdge !== EdgeStates.NONE && event.slip) { + if (velocity < 0 && this._onEdge === EdgeStates.TOP || velocity > 0 && this._onEdge === EdgeStates.BOTTOM) { + if (!this._earlyEnd) { + _handleEnd.call(this, event); + this._earlyEnd = true; + } + } else if (this._earlyEnd && Math.abs(velocity) > Math.abs(this.getVelocity())) { + _handleStart.call(this, event); + } + } + if (this._earlyEnd) + return; + this._touchVelocity = velocity; + if (event.slip) { + var speedLimit = this.options.speedLimit; + if (velocity < -speedLimit) + velocity = -speedLimit; + else if (velocity > speedLimit) + velocity = speedLimit; + this.setVelocity(velocity); + var deltaLimit = speedLimit * 16; + if (delta > deltaLimit) + delta = deltaLimit; + else if (delta < -deltaLimit) + delta = -deltaLimit; + } + this.setPosition(this.getPosition() + delta); + this._displacement += delta; + if (this._springState === SpringStates.NONE) + _normalizeState.call(this); +} +function _handleEnd(event) { + this._touchCount = event.count || 0; + if (!this._touchCount) { + _detachAgents.call(this); + if (this._onEdge !== EdgeStates.NONE) + _setSpring.call(this, this._edgeSpringPosition, SpringStates.EDGE); + _attachAgents.call(this); + var velocity = -event.velocity; + var speedLimit = this.options.speedLimit; + if (event.slip) + speedLimit *= this.options.edgeGrip; + if (velocity < -speedLimit) + velocity = -speedLimit; + else if (velocity > speedLimit) + velocity = speedLimit; + this.setVelocity(velocity); + this._touchVelocity = 0; + this._needsPaginationCheck = true; + } +} +function _bindEvents() { + this._eventInput.bindThis(this); + this._eventInput.on('start', _handleStart); + this._eventInput.on('update', _handleMove); + this._eventInput.on('end', _handleEnd); + this._eventInput.on('resize', function () { + this._node._.calculateSize(); + }.bind(this)); + this._scroller.on('onEdge', function (data) { + this._edgeSpringPosition = data.position; + _handleEdge.call(this, this._scroller.onEdge()); + this._eventOutput.emit('onEdge'); + }.bind(this)); + this._scroller.on('offEdge', function () { + this.sync.setOptions({ scale: this.options.syncScale }); + this._onEdge = this._scroller.onEdge(); + this._eventOutput.emit('offEdge'); + }.bind(this)); + this._particle.on('update', function (particle) { + if (this._springState === SpringStates.NONE) + _normalizeState.call(this); + this._displacement = particle.position.x - this._totalShift; + }.bind(this)); + this._particle.on('end', function () { + if (!this.options.paginated || this.options.paginated && this._springState !== SpringStates.NONE) + this._eventOutput.emit('settle'); + }.bind(this)); +} +function _attachAgents() { + if (this._springState) + this._physicsEngine.attach([this.spring], this._particle); + else + this._physicsEngine.attach([ + this.drag, + this.friction + ], this._particle); +} +function _detachAgents() { + this._springState = SpringStates.NONE; + this._physicsEngine.detachAll(); +} +function _nodeSizeForDirection(node) { + var direction = this.options.direction; + var nodeSize = node.getSize(); + return !nodeSize ? this._scroller.getSize()[direction] : nodeSize[direction]; +} +function _handleEdge(edge) { + this.sync.setOptions({ scale: this.options.edgeGrip }); + this._onEdge = edge; + if (!this._touchCount && this._springState !== SpringStates.EDGE) { + _setSpring.call(this, this._edgeSpringPosition, SpringStates.EDGE); + } + if (this._springState && Math.abs(this.getVelocity()) < 0.001) { + _detachAgents.call(this); + _attachAgents.call(this); + } +} +function _handlePagination() { + if (this._touchCount) + return; + if (this._springState === SpringStates.EDGE) + return; + var velocity = this.getVelocity(); + if (Math.abs(velocity) >= this.options.pageStopSpeed) + return; + var position = this.getPosition(); + var velocitySwitch = Math.abs(velocity) > this.options.pageSwitchSpeed; + var nodeSize = _nodeSizeForDirection.call(this, this._node); + var positionNext = position > 0.5 * nodeSize; + var positionPrev = position < 0.5 * nodeSize; + var velocityNext = velocity > 0; + var velocityPrev = velocity < 0; + this._needsPaginationCheck = false; + if (positionNext && !velocitySwitch || velocitySwitch && velocityNext) { + this.goToNextPage(); + } else if (velocitySwitch && velocityPrev) { + this.goToPreviousPage(); + } else + _setSpring.call(this, 0, SpringStates.PAGE); +} +function _setSpring(position, springState) { + var springOptions; + if (springState === SpringStates.EDGE) { + this._edgeSpringPosition = position; + springOptions = { + anchor: [ + this._edgeSpringPosition, + 0, + 0 + ], + period: this.options.edgePeriod, + dampingRatio: this.options.edgeDamp + }; + } else if (springState === SpringStates.PAGE) { + this._pageSpringPosition = position; + springOptions = { + anchor: [ + this._pageSpringPosition, + 0, + 0 + ], + period: this.options.pagePeriod, + dampingRatio: this.options.pageDamp + }; + } + this.spring.setOptions(springOptions); + if (springState && !this._springState) { + _detachAgents.call(this); + this._springState = springState; + _attachAgents.call(this); + } + this._springState = springState; +} +function _normalizeState() { + var offset = 0; + var position = this.getPosition(); + position += (position < 0 ? -0.5 : 0.5) >> 0; + var nodeSize = _nodeSizeForDirection.call(this, this._node); + var nextNode = this._node.getNext(); + while (offset + position >= nodeSize && nextNode) { + offset -= nodeSize; + this._scroller.sequenceFrom(nextNode); + this._node = nextNode; + nextNode = this._node.getNext(); + nodeSize = _nodeSizeForDirection.call(this, this._node); + } + var previousNode = this._node.getPrevious(); + var previousNodeSize; + while (offset + position <= 0 && previousNode) { + previousNodeSize = _nodeSizeForDirection.call(this, previousNode); + this._scroller.sequenceFrom(previousNode); + this._node = previousNode; + offset += previousNodeSize; + previousNode = this._node.getPrevious(); + } + if (offset) + _shiftOrigin.call(this, offset); + if (this._node) { + if (this._node.index !== this._cachedIndex) { + if (this.getPosition() < 0.5 * nodeSize) { + this._cachedIndex = this._node.index; + this._eventOutput.emit('pageChange', { + direction: -1, + index: this._cachedIndex + }); + } + } else { + if (this.getPosition() > 0.5 * nodeSize) { + this._cachedIndex = this._node.index + 1; + this._eventOutput.emit('pageChange', { + direction: 1, + index: this._cachedIndex + }); + } + } + } +} +function _shiftOrigin(amount) { + this._edgeSpringPosition += amount; + this._pageSpringPosition += amount; + this.setPosition(this.getPosition() + amount); + this._totalShift += amount; + if (this._springState === SpringStates.EDGE) { + this.spring.setOptions({ + anchor: [ + this._edgeSpringPosition, + 0, + 0 + ] + }); + } else if (this._springState === SpringStates.PAGE) { + this.spring.setOptions({ + anchor: [ + this._pageSpringPosition, + 0, + 0 + ] + }); + } +} +Scrollview.prototype.getCurrentIndex = function getCurrentIndex() { + return this._node.index; +}; +Scrollview.prototype.goToPreviousPage = function goToPreviousPage() { + if (!this._node || this._onEdge === EdgeStates.TOP) + return null; + if (this.getPosition() > 1 && this._springState === SpringStates.NONE) { + _setSpring.call(this, 0, SpringStates.PAGE); + return this._node; + } + var previousNode = this._node.getPrevious(); + if (previousNode) { + var previousNodeSize = _nodeSizeForDirection.call(this, previousNode); + this._scroller.sequenceFrom(previousNode); + this._node = previousNode; + _shiftOrigin.call(this, previousNodeSize); + _setSpring.call(this, 0, SpringStates.PAGE); + } + return previousNode; +}; +Scrollview.prototype.goToNextPage = function goToNextPage() { + if (!this._node || this._onEdge === EdgeStates.BOTTOM) + return null; + var nextNode = this._node.getNext(); + if (nextNode) { + var currentNodeSize = _nodeSizeForDirection.call(this, this._node); + this._scroller.sequenceFrom(nextNode); + this._node = nextNode; + _shiftOrigin.call(this, -currentNodeSize); + _setSpring.call(this, 0, SpringStates.PAGE); + } + return nextNode; +}; +Scrollview.prototype.goToPage = function goToPage(index) { + var currentIndex = this.getCurrentIndex(); + var i; + if (currentIndex > index) { + for (i = 0; i < currentIndex - index; i++) + this.goToPreviousPage(); + } + if (currentIndex < index) { + for (i = 0; i < index - currentIndex; i++) + this.goToNextPage(); + } +}; +Scrollview.prototype.outputFrom = function outputFrom() { + return this._scroller.outputFrom.apply(this._scroller, arguments); +}; +Scrollview.prototype.getPosition = function getPosition() { + return this._particle.getPosition1D(); +}; +Scrollview.prototype.getAbsolutePosition = function getAbsolutePosition() { + return this._scroller.getCumulativeSize(this.getCurrentIndex())[this.options.direction] + this.getPosition(); +}; +Scrollview.prototype.getOffset = Scrollview.prototype.getPosition; +Scrollview.prototype.setPosition = function setPosition(x) { + this._particle.setPosition1D(x); +}; +Scrollview.prototype.setOffset = Scrollview.prototype.setPosition; +Scrollview.prototype.getVelocity = function getVelocity() { + return this._touchCount ? this._touchVelocity : this._particle.getVelocity1D(); +}; +Scrollview.prototype.setVelocity = function setVelocity(v) { + this._particle.setVelocity1D(v); +}; +Scrollview.prototype.setOptions = function setOptions(options) { + if (options.direction !== undefined) { + if (options.direction === 'x') + options.direction = Utility.Direction.X; + else if (options.direction === 'y') + options.direction = Utility.Direction.Y; + } + if (options.groupScroll !== this.options.groupScroll) { + if (options.groupScroll) + this.subscribe(this._scroller); + else + this.unsubscribe(this._scroller); + } + this._optionsManager.setOptions(options); + this._scroller.setOptions(options); + if (options.drag !== undefined) + this.drag.setOptions({ strength: this.options.drag }); + if (options.friction !== undefined) + this.friction.setOptions({ strength: this.options.friction }); + if (options.edgePeriod !== undefined || options.edgeDamp !== undefined) { + this.spring.setOptions({ + period: this.options.edgePeriod, + dampingRatio: this.options.edgeDamp + }); + } + if (options.rails || options.direction !== undefined || options.syncScale !== undefined || options.preventDefault) { + this.sync.setOptions({ + rails: this.options.rails, + direction: this.options.direction === Utility.Direction.X ? GenericSync.DIRECTION_X : GenericSync.DIRECTION_Y, + scale: this.options.syncScale, + preventDefault: this.options.preventDefault + }); + } +}; +Scrollview.prototype.sequenceFrom = function sequenceFrom(node) { + if (node instanceof Array) + node = new ViewSequence({ + array: node, + trackSize: true + }); + this._node = node; + return this._scroller.sequenceFrom(node); +}; +Scrollview.prototype.getSize = function getSize() { + return this._scroller.getSize.apply(this._scroller, arguments); +}; +Scrollview.prototype.render = function render() { + if (this.options.paginated && this._needsPaginationCheck) + _handlePagination.call(this); + return this._scroller.render(); +}; +module.exports = Scrollview; +},{"../core/EventHandler":7,"../core/OptionsManager":10,"../core/ViewSequence":17,"../inputs/GenericSync":27,"../inputs/ScrollSync":32,"../inputs/TouchSync":33,"../physics/PhysicsEngine":48,"../physics/bodies/Particle":51,"../physics/forces/Drag":63,"../physics/forces/Spring":68,"../utilities/Utility":95,"../views/Scroller":108}],110:[function(_dereq_,module,exports){ +var OptionsManager = _dereq_('../core/OptionsManager'); +var Entity = _dereq_('../core/Entity'); +var Transform = _dereq_('../core/Transform'); +var ViewSequence = _dereq_('../core/ViewSequence'); +var Utility = _dereq_('../utilities/Utility'); +function SequentialLayout(options) { + this._items = null; + this._size = null; + this._outputFunction = SequentialLayout.DEFAULT_OUTPUT_FUNCTION; + this.options = Utility.clone(this.constructor.DEFAULT_OPTIONS || SequentialLayout.DEFAULT_OPTIONS); + this.optionsManager = new OptionsManager(this.options); + this.id = Entity.register(this); + this.cachedSize = [ + undefined, + undefined + ]; + if (options) + this.setOptions(options); +} +SequentialLayout.DEFAULT_OPTIONS = { + direction: Utility.Direction.Y, + itemSpacing: 0 +}; +SequentialLayout.DEFAULT_OUTPUT_FUNCTION = function DEFAULT_OUTPUT_FUNCTION(input, offset, index) { + var transform = this.options.direction === Utility.Direction.X ? Transform.translate(offset, 0) : Transform.translate(0, offset); + return { + size: this.cachedSize, + transform: transform, + target: input.render() + }; +}; +SequentialLayout.prototype.getSize = function getSize() { + if (!this._size) + this.render(); + return this._size; +}; +SequentialLayout.prototype.sequenceFrom = function sequenceFrom(items) { + if (items instanceof Array) + items = new ViewSequence(items); + this._items = items; + return this; +}; +SequentialLayout.prototype.setOptions = function setOptions(options) { + this.optionsManager.setOptions.apply(this.optionsManager, arguments); + return this; +}; +SequentialLayout.prototype.setOutputFunction = function setOutputFunction(outputFunction) { + this._outputFunction = outputFunction; + return this; +}; +SequentialLayout.prototype.render = function render() { + return this.id; +}; +SequentialLayout.prototype.commit = function commit(parentSpec) { + var length = 0; + var secondaryDirection = this.options.direction ^ 1; + var currentNode = this._items; + var item = null; + var itemSize = []; + var output = {}; + var result = []; + var i = 0; + this._size = [ + 0, + 0 + ]; + this.cachedSize = parentSpec.size; + while (currentNode) { + item = currentNode.get(); + if (!item) + break; + if (item.getSize) + itemSize = item.getSize(); + output = this._outputFunction.call(this, item, length, i++); + result.push(output); + if (itemSize) { + if (itemSize[this.options.direction]) + length += itemSize[this.options.direction]; + if (itemSize[secondaryDirection] > this._size[secondaryDirection]) + this._size[secondaryDirection] = itemSize[secondaryDirection]; + if (itemSize[secondaryDirection] === 0) + this._size[secondaryDirection] = undefined; + } + currentNode = currentNode.getNext(); + if (this.options.itemSpacing && currentNode) + length += this.options.itemSpacing; + } + this._size[this.options.direction] = length; + return { + transform: parentSpec.transform, + origin: parentSpec.origin, + opacity: parentSpec.opacity, + size: this.getSize(), + target: result + }; +}; +module.exports = SequentialLayout; +},{"../core/Entity":5,"../core/OptionsManager":10,"../core/Transform":15,"../core/ViewSequence":17,"../utilities/Utility":95}],111:[function(_dereq_,module,exports){ +var View = _dereq_('../core/View'); +var Entity = _dereq_('../core/Entity'); +var Transform = _dereq_('../core/Transform'); +function SizeAwareView() { + View.apply(this, arguments); + this._id = Entity.register(this); + this._parentSize = []; +} +SizeAwareView.prototype = Object.create(View.prototype); +SizeAwareView.prototype.constructor = SizeAwareView; +SizeAwareView.prototype.commit = function commit(context) { + var transform = context.transform; + var opacity = context.opacity; + var origin = context.origin; + if (!this._parentSize || this._parentSize[0] !== context.size[0] || this._parentSize[1] !== context.size[1]) { + this._parentSize[0] = context.size[0]; + this._parentSize[1] = context.size[1]; + this._eventInput.emit('parentResize', this._parentSize); + if (this.onResize) + this.onResize(this._parentSize); + } + if (this._parentSize) { + transform = Transform.moveThen([ + -this._parentSize[0] * origin[0], + -this._parentSize[1] * origin[1], + 0 + ], transform); + } + return { + transform: transform, + opacity: opacity, + size: this._parentSize, + target: this._node.render() + }; +}; +SizeAwareView.prototype.getParentSize = function getParentSize() { + return this._parentSize; +}; +SizeAwareView.prototype.render = function render() { + return this._id; +}; +module.exports = SizeAwareView; +},{"../core/Entity":5,"../core/Transform":15,"../core/View":16}],112:[function(_dereq_,module,exports){ +module.exports = { + ContextualView: _dereq_('./ContextualView'), + Deck: _dereq_('./Deck'), + DrawerLayout: _dereq_('./DrawerLayout'), + EdgeSwapper: _dereq_('./EdgeSwapper'), + FlexibleLayout: _dereq_('./FlexibleLayout'), + Flipper: _dereq_('./Flipper'), + GridLayout: _dereq_('./GridLayout'), + HeaderFooterLayout: _dereq_('./HeaderFooterLayout'), + Lightbox: _dereq_('./Lightbox'), + RenderController: _dereq_('./RenderController'), + ScrollContainer: _dereq_('./ScrollContainer'), + Scroller: _dereq_('./Scroller'), + Scrollview: _dereq_('./Scrollview'), + SequentialLayout: _dereq_('./SequentialLayout'), + SizeAwareView: _dereq_('./SizeAwareView') +}; + +},{"./ContextualView":97,"./Deck":98,"./DrawerLayout":99,"./EdgeSwapper":100,"./FlexibleLayout":101,"./Flipper":102,"./GridLayout":103,"./HeaderFooterLayout":104,"./Lightbox":105,"./RenderController":106,"./ScrollContainer":107,"./Scroller":108,"./Scrollview":109,"./SequentialLayout":110,"./SizeAwareView":111}],113:[function(_dereq_,module,exports){ +var Scene = _dereq_('../core/Scene'); +var Surface = _dereq_('../core/Surface'); +var Transform = _dereq_('../core/Transform'); +var View = _dereq_('../core/View'); +function NavigationBar(options) { + View.apply(this, arguments); + this.title = new Surface({ + classes: this.options.classes, + content: this.options.content + }); + this.back = new Surface({ + size: [ + this.options.size[1], + this.options.size[1] + ], + classes: this.options.classes, + content: this.options.backContent + }); + this.back.on('click', function () { + this._eventOutput.emit('back', {}); + }.bind(this)); + this.more = new Surface({ + size: [ + this.options.size[1], + this.options.size[1] + ], + classes: this.options.classes, + content: this.options.moreContent + }); + this.more.on('click', function () { + this._eventOutput.emit('more', {}); + }.bind(this)); + this.layout = new Scene({ + id: 'master', + size: this.options.size, + target: [ + { + transform: Transform.inFront, + origin: [ + 0, + 0.5 + ], + align: [ + 0, + 0.5 + ], + target: this.back + }, + { + origin: [ + 0.5, + 0.5 + ], + align: [ + 0.5, + 0.5 + ], + target: this.title + }, + { + transform: Transform.inFront, + origin: [ + 1, + 0.5 + ], + align: [ + 1, + 0.5 + ], + target: this.more + } + ] + }); + this._add(this.layout); + this._optionsManager.on('change', function (event) { + var key = event.id; + var data = event.value; + if (key === 'size') { + this.layout.id.master.setSize(data); + this.title.setSize(data); + this.back.setSize([ + data[1], + data[1] + ]); + this.more.setSize([ + data[1], + data[1] + ]); + } else if (key === 'backClasses') { + this.back.setOptions({ classes: this.options.classes.concat(this.options.backClasses) }); + } else if (key === 'backContent') { + this.back.setContent(this.options.backContent); + } else if (key === 'classes') { + this.title.setOptions({ classes: this.options.classes }); + this.back.setOptions({ classes: this.options.classes.concat(this.options.backClasses) }); + this.more.setOptions({ classes: this.options.classes.concat(this.options.moreClasses) }); + } else if (key === 'content') { + this.setContent(this.options.content); + } else if (key === 'moreClasses') { + this.more.setOptions({ classes: this.options.classes.concat(this.options.moreClasses) }); + } else if (key === 'moreContent') { + this.more.setContent(this.options.content); + } + }.bind(this)); +} +NavigationBar.prototype = Object.create(View.prototype); +NavigationBar.prototype.constructor = NavigationBar; +NavigationBar.DEFAULT_OPTIONS = { + size: [ + undefined, + 50 + ], + backClasses: ['back'], + backContent: '◀', + classes: ['navigation'], + content: '', + moreClasses: ['more'], + moreContent: '✚' +}; +NavigationBar.prototype.setContent = function setContent(content) { + return this.title.setContent(content); +}; +module.exports = NavigationBar; +},{"../core/Scene":12,"../core/Surface":14,"../core/Transform":15,"../core/View":16}],114:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +var CanvasSurface = _dereq_('../surfaces/CanvasSurface'); +var Transform = _dereq_('../core/Transform'); +var EventHandler = _dereq_('../core/EventHandler'); +var Utilities = _dereq_('../math/Utilities'); +var OptionsManager = _dereq_('../core/OptionsManager'); +var MouseSync = _dereq_('../inputs/MouseSync'); +var TouchSync = _dereq_('../inputs/TouchSync'); +var GenericSync = _dereq_('../inputs/GenericSync'); +GenericSync.register({ + mouse: MouseSync, + touch: TouchSync +}); +function Slider(options) { + this.options = Object.create(Slider.DEFAULT_OPTIONS); + this.optionsManager = new OptionsManager(this.options); + if (options) + this.setOptions(options); + this.indicator = new CanvasSurface({ + size: this.options.indicatorSize, + classes: ['slider-back'] + }); + this.label = new Surface({ + size: this.options.labelSize, + content: this.options.label, + properties: { pointerEvents: 'none' }, + classes: ['slider-label'] + }); + this.eventOutput = new EventHandler(); + this.eventInput = new EventHandler(); + EventHandler.setInputHandler(this, this.eventInput); + EventHandler.setOutputHandler(this, this.eventOutput); + var scale = (this.options.range[1] - this.options.range[0]) / this.options.indicatorSize[0]; + this.sync = new GenericSync([ + 'mouse', + 'touch' + ], { + scale: scale, + direction: GenericSync.DIRECTION_X + }); + this.indicator.pipe(this.sync); + this.sync.pipe(this); + this.eventInput.on('update', function (data) { + this.set(data.position); + }.bind(this)); + this._drawPos = 0; + _updateLabel.call(this); +} +Slider.DEFAULT_OPTIONS = { + size: [ + 200, + 60 + ], + indicatorSize: [ + 200, + 30 + ], + labelSize: [ + 200, + 30 + ], + range: [ + 0, + 1 + ], + precision: 2, + value: 0, + label: '', + fillColor: 'rgba(170, 170, 170, 1)' +}; +function _updateLabel() { + this.label.setContent(this.options.label + '' + this.get().toFixed(this.options.precision) + ''); +} +Slider.prototype.setOptions = function setOptions(options) { + return this.optionsManager.setOptions(options); +}; +Slider.prototype.get = function get() { + return this.options.value; +}; +Slider.prototype.set = function set(value) { + if (value === this.options.value) + return; + this.options.value = Utilities.clamp(value, this.options.range); + _updateLabel.call(this); + this.eventOutput.emit('change', { value: value }); +}; +Slider.prototype.getSize = function getSize() { + return this.options.size; +}; +Slider.prototype.render = function render() { + var range = this.options.range; + var fillSize = Math.floor((this.get() - range[0]) / (range[1] - range[0]) * this.options.indicatorSize[0]); + if (fillSize < this._drawPos) { + this.indicator.getContext('2d').clearRect(fillSize, 0, this._drawPos - fillSize + 1, this.options.indicatorSize[1]); + } else if (fillSize > this._drawPos) { + var ctx = this.indicator.getContext('2d'); + ctx.fillStyle = this.options.fillColor; + ctx.fillRect(this._drawPos - 1, 0, fillSize - this._drawPos + 1, this.options.indicatorSize[1]); + } + this._drawPos = fillSize; + return { + size: this.options.size, + target: [ + { + origin: [ + 0, + 0 + ], + target: this.indicator.render() + }, + { + transform: Transform.translate(0, 0, 1), + origin: [ + 0, + 0 + ], + target: this.label.render() + } + ] + }; +}; +module.exports = Slider; +},{"../core/EventHandler":7,"../core/OptionsManager":10,"../core/Surface":14,"../core/Transform":15,"../inputs/GenericSync":27,"../inputs/MouseSync":28,"../inputs/TouchSync":33,"../math/Utilities":40,"../surfaces/CanvasSurface":74}],115:[function(_dereq_,module,exports){ +var Utility = _dereq_('../utilities/Utility'); +var View = _dereq_('../core/View'); +var GridLayout = _dereq_('../views/GridLayout'); +var ToggleButton = _dereq_('./ToggleButton'); +function TabBar(options) { + View.apply(this, arguments); + this.layout = new GridLayout(); + this.buttons = []; + this._buttonIds = {}; + this._buttonCallbacks = {}; + this.layout.sequenceFrom(this.buttons); + this._add(this.layout); + this._optionsManager.on('change', _updateOptions.bind(this)); +} +TabBar.prototype = Object.create(View.prototype); +TabBar.prototype.constructor = TabBar; +TabBar.DEFAULT_OPTIONS = { + sections: [], + widget: ToggleButton, + size: [ + undefined, + 50 + ], + direction: Utility.Direction.X, + buttons: { toggleMode: ToggleButton.ON } +}; +function _updateOptions(data) { + var id = data.id; + var value = data.value; + if (id === 'direction') { + this.layout.setOptions({ dimensions: _resolveGridDimensions.call(this.buttons.length, this.options.direction) }); + } else if (id === 'buttons') { + for (var i in this.buttons) { + this.buttons[i].setOptions(value); + } + } else if (id === 'sections') { + for (var sectionId in this.options.sections) { + this.defineSection(sectionId, this.options.sections[sectionId]); + } + } +} +function _resolveGridDimensions(count, direction) { + if (direction === Utility.Direction.X) + return [ + count, + 1 + ]; + else + return [ + 1, + count + ]; +} +TabBar.prototype.defineSection = function defineSection(id, content) { + var button; + var i = this._buttonIds[id]; + if (i === undefined) { + i = this.buttons.length; + this._buttonIds[id] = i; + var widget = this.options.widget; + button = new widget(); + this.buttons[i] = button; + this.layout.setOptions({ dimensions: _resolveGridDimensions(this.buttons.length, this.options.direction) }); + } else { + button = this.buttons[i]; + button.unbind('select', this._buttonCallbacks[id]); + } + if (this.options.buttons) + button.setOptions(this.options.buttons); + button.setOptions(content); + this._buttonCallbacks[id] = this.select.bind(this, id); + button.on('select', this._buttonCallbacks[id]); +}; +TabBar.prototype.select = function select(id) { + var btn = this._buttonIds[id]; + if (this.buttons[btn] && this.buttons[btn].isSelected()) { + this._eventOutput.emit('select', { id: id }); + } else if (this.buttons[btn]) { + this.buttons[btn].select(); + } + for (var i = 0; i < this.buttons.length; i++) { + if (i !== btn) + this.buttons[i].deselect(); + } +}; +module.exports = TabBar; +},{"../core/View":16,"../utilities/Utility":95,"../views/GridLayout":103,"./ToggleButton":116}],116:[function(_dereq_,module,exports){ +var Surface = _dereq_('../core/Surface'); +var EventHandler = _dereq_('../core/EventHandler'); +var RenderController = _dereq_('../views/RenderController'); +function ToggleButton(options) { + this.options = { + content: [ + '', + '' + ], + offClasses: ['off'], + onClasses: ['on'], + size: undefined, + outTransition: { + curve: 'easeInOut', + duration: 300 + }, + inTransition: { + curve: 'easeInOut', + duration: 300 + }, + toggleMode: ToggleButton.TOGGLE, + crossfade: true + }; + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); + this.offSurface = new Surface(); + this.offSurface.on('click', function () { + if (this.options.toggleMode !== ToggleButton.OFF) + this.select(); + }.bind(this)); + this.offSurface.pipe(this._eventOutput); + this.onSurface = new Surface(); + this.onSurface.on('click', function () { + if (this.options.toggleMode !== ToggleButton.ON) + this.deselect(); + }.bind(this)); + this.onSurface.pipe(this._eventOutput); + this.arbiter = new RenderController({ overlap: this.options.crossfade }); + this.deselect(); + if (options) + this.setOptions(options); +} +ToggleButton.OFF = 0; +ToggleButton.ON = 1; +ToggleButton.TOGGLE = 2; +ToggleButton.prototype.select = function select(suppressEvent) { + this.selected = true; + this.arbiter.show(this.onSurface, this.options.inTransition); + if (!suppressEvent) { + this._eventOutput.emit('select'); + } +}; +ToggleButton.prototype.deselect = function deselect(suppressEvent) { + this.selected = false; + this.arbiter.show(this.offSurface, this.options.outTransition); + if (!suppressEvent) { + this._eventOutput.emit('deselect'); + } +}; +ToggleButton.prototype.isSelected = function isSelected() { + return this.selected; +}; +ToggleButton.prototype.setOptions = function setOptions(options) { + if (options.content !== undefined) { + if (!(options.content instanceof Array)) + options.content = [ + options.content, + options.content + ]; + this.options.content = options.content; + this.offSurface.setContent(this.options.content[0]); + this.onSurface.setContent(this.options.content[1]); + } + if (options.offClasses) { + this.options.offClasses = options.offClasses; + this.offSurface.setClasses(this.options.offClasses); + } + if (options.onClasses) { + this.options.onClasses = options.onClasses; + this.onSurface.setClasses(this.options.onClasses); + } + if (options.size !== undefined) { + this.options.size = options.size; + this.onSurface.setSize(this.options.size); + this.offSurface.setSize(this.options.size); + } + if (options.toggleMode !== undefined) + this.options.toggleMode = options.toggleMode; + if (options.outTransition !== undefined) + this.options.outTransition = options.outTransition; + if (options.inTransition !== undefined) + this.options.inTransition = options.inTransition; + if (options.crossfade !== undefined) { + this.options.crossfade = options.crossfade; + this.arbiter.setOptions({ overlap: this.options.crossfade }); + } +}; +ToggleButton.prototype.getSize = function getSize() { + return this.options.size; +}; +ToggleButton.prototype.render = function render() { + return this.arbiter.render(); +}; +module.exports = ToggleButton; +},{"../core/EventHandler":7,"../core/Surface":14,"../views/RenderController":106}],117:[function(_dereq_,module,exports){ +module.exports = { + NavigationBar: _dereq_('./NavigationBar'), + Slider: _dereq_('./Slider'), + TabBar: _dereq_('./TabBar'), + ToggleButton: _dereq_('./ToggleButton') +}; + +},{"./NavigationBar":113,"./Slider":114,"./TabBar":115,"./ToggleButton":116}]},{},[23])(23) +}); \ No newline at end of file diff --git a/dist/famous-global.min.js b/dist/famous-global.min.js new file mode 100644 index 00000000..6d8b7af3 --- /dev/null +++ b/dist/famous-global.min.js @@ -0,0 +1,14 @@ +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +!function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{var i;"undefined"!=typeof window?i=window:"undefined"!=typeof global?i=global:"undefined"!=typeof self&&(i=self),i.famous=t()}}(function(){return function t(i,e,n){function o(r,a){if(!e[r]){if(!i[r]){var h="function"==typeof require&&require;if(!a&&h)return h(r,!0);if(s)return s(r,!0);var c=new Error("Cannot find module '"+r+"'");throw c.code="MODULE_NOT_FOUND",c}var p=e[r]={exports:{}};i[r][0].call(p.exports,function(t){var e=i[r][1][t];return o(e?e:t)},p,p.exports,t,i,e,n)}return e[r].exports}for(var s="function"==typeof require&&require,r=0;r0?i=e.pop():(i=document.createElement(t),this.container.appendChild(i)),this.nodeCount++,i},e.prototype.deallocate=function(t){var i=t.nodeName.toLowerCase(),e=this.detachedNodes[i];e.push(t),this.nodeCount--},e.prototype.getNodeCount=function(){return this.nodeCount},i.exports=e},{}],3:[function(t,i){function e(t){this._matrix=null,this._opacity=1,this._origin=null,this._size=null,this._eventOutput=new c,this._eventOutput.bindThis(this),this.eventForwarder=function(t){this._eventOutput.emit(t.type,t)}.bind(this),this.id=h.register(this),this._element=null,this._sizeDirty=!1,this._originDirty=!1,this._transformDirty=!1,this._invisible=!1,t&&this.attach(t)}function n(t){for(var i in this._eventOutput.listeners)t.addEventListener(i,this.eventForwarder)}function o(t){for(var i in this._eventOutput.listeners)t.removeEventListener(i,this.eventForwarder)}function s(t){t[12]=Math.round(t[12]*l)/l,t[13]=Math.round(t[13]*l)/l;for(var i="matrix3d(",e=0;15>e;e++)i+=t[e]<1e-6&&t[e]>-1e-6?"0,":t[e]+",";return i+=t[15]+")"}function r(t){return 100*t[0]+"% "+100*t[1]+"%"}function a(t,i){return t&&i?t[0]!==i[0]||t[1]!==i[1]:t!==i}var h=t("./Entity"),c=t("./EventHandler"),p=t("./Transform"),u=!("transform"in document.documentElement.style),l=window.devicePixelRatio||1;e.prototype.on=function(t,i){this._element&&this._element.addEventListener(t,this.eventForwarder),this._eventOutput.on(t,i)},e.prototype.removeListener=function(t,i){this._eventOutput.removeListener(t,i)},e.prototype.emit=function(t,i){i&&!i.origin&&(i.origin=this);var e=this._eventOutput.emit(t,i);return e&&i&&i.stopPropagation&&i.stopPropagation(),e},e.prototype.pipe=function(t){return this._eventOutput.pipe(t)},e.prototype.unpipe=function(t){return this._eventOutput.unpipe(t)},e.prototype.render=function(){return this.id};var f;f=u?function(t,i){t.style.webkitTransform=s(i)}:function(t,i){t.style.transform=s(i)};var d=u?function(t,i){t.style.webkitTransformOrigin=r(i)}:function(t,i){t.style.transformOrigin=r(i)},_=u?function(t){t.style.webkitTransform="scale3d(0.0001,0.0001,0.0001)",t.style.opacity=0}:function(t){t.style.transform="scale3d(0.0001,0.0001,0.0001)",t.style.opacity=0};e.prototype.commit=function(t){var i=this._element;if(i){var e=t.transform,n=t.opacity,o=t.origin;if(t.size,!e&&this._matrix)return this._matrix=null,this._opacity=0,_(i),void 0;if(a(this._origin,o)&&(this._originDirty=!0),p.notEquals(this._matrix,e)&&(this._transformDirty=!0),this._invisible&&(this._invisible=!1,this._element.style.display=""),this._opacity!==n&&(this._opacity=n,i.style.opacity=n>=1?"0.999999":n),this._transformDirty||this._originDirty||this._sizeDirty){this._sizeDirty&&(this._sizeDirty=!1),this._originDirty&&(o?(this._origin||(this._origin=[0,0]),this._origin[0]=o[0],this._origin[1]=o[1]):this._origin=null,d(i,this._origin),this._originDirty=!1),e||(e=p.identity),this._matrix=e;var s=this._size?p.thenMove(e,[-this._size[0]*o[0],-this._size[1]*o[1],0]):e;f(i,s),this._transformDirty=!1}}},e.prototype.cleanup=function(){this._element&&(this._invisible=!0,this._element.style.display="none")},e.prototype.attach=function(t){this._element=t,n.call(this,t)},e.prototype.detach=function(){var t=this._element;return t&&(o.call(this,t),this._invisible&&(this._invisible=!1,this._element.style.display="")),this._element=null,t},i.exports=e},{"./Entity":5,"./EventHandler":7,"./Transform":15}],4:[function(t,i){function e(){w.runLoop?(f.step(),window.requestAnimationFrame(e)):O=!1}function n(){for(var t=0;tt-m)){var i=0;h=t-m,m=t,T.emit("prerender");for(var e=_.length;e--;)_.shift()(g);for(;v.length&&Date.now()-t=0&&d.splice(i,1)},f.nextTick=function(t){_.push(t)},f.defer=function(t){v.push(t)},b.on("change",function(t){"fpsCap"===t.id?f.setFPSCap(t.value):"runLoop"===t.id&&!O&&t.value&&(O=!0,window.requestAnimationFrame(e))}),i.exports=f},{"./Context":1,"./EventHandler":7,"./OptionsManager":10}],5:[function(t,i){function e(t){return r[t]}function n(t,i){r[t]=i}function o(t){var i=r.length;return n(i,t),i}function s(t){n(t,null)}var r=[];i.exports={register:o,unregister:s,get:e,set:n}},{}],6:[function(t,i){function e(){this.listeners={},this._owner=this}e.prototype.emit=function(t,i){var e=this.listeners[t];if(e)for(var n=0;ne&&this.listeners[t].push(i),this},e.prototype.addListener=e.prototype.on,e.prototype.removeListener=function(t,i){var e=this.listeners[t];if(void 0!==e){var n=e.indexOf(i);n>=0&&e.splice(n,1)}return this},e.prototype.bindThis=function(t){this._owner=t},i.exports=e},{}],7:[function(t,i){function e(){n.apply(this,arguments),this.downstream=[],this.downstreamFn=[],this.upstream=[],this.upstreamListeners={}}var n=t("./EventEmitter");e.prototype=Object.create(n.prototype),e.prototype.constructor=e,e.setInputHandler=function(t,i){t.trigger=i.trigger.bind(i),i.subscribe&&i.unsubscribe&&(t.subscribe=i.subscribe.bind(i),t.unsubscribe=i.unsubscribe.bind(i))},e.setOutputHandler=function(t,i){i instanceof e&&i.bindThis(t),t.pipe=i.pipe.bind(i),t.unpipe=i.unpipe.bind(i),t.on=i.on.bind(i),t.addListener=t.on,t.removeListener=i.removeListener.bind(i)},e.prototype.emit=function(t,i){n.prototype.emit.apply(this,arguments);var e=0;for(e=0;ee&&i.push(t),t instanceof Function?t("pipe",null):t.trigger&&t.trigger("pipe",null),t},e.prototype.unpipe=function(t){if(t.unsubscribe instanceof Function)return t.unsubscribe(this);var i=t instanceof Function?this.downstreamFn:this.downstream,e=i.indexOf(t);return e>=0?(i.splice(e,1),t instanceof Function?t("unpipe",null):t.trigger&&t.trigger("unpipe",null),t):!1},e.prototype.on=function(t){if(n.prototype.on.apply(this,arguments),!(t in this.upstreamListeners)){var i=this.trigger.bind(this,t);this.upstreamListeners[t]=i;for(var e=0;ei){this.upstream.push(t);for(var e in this.upstreamListeners)t.on(e,this.upstreamListeners[e])}return this},e.prototype.unsubscribe=function(t){var i=this.upstream.indexOf(t);if(i>=0){this.upstream.splice(i,1);for(var e in this.upstreamListeners)t.removeListener(e,this.upstreamListeners[e])}return this},i.exports=e},{"./EventEmitter":6}],8:[function(t,i){function e(t){s.call(this,t),this._shouldRecalculateSize=!1,this._container=document.createDocumentFragment(),this.context=new n(this._container),this.setContent(this._container),this._groupSize=[void 0,void 0]}var n=t("./Context"),o=t("./Transform"),s=t("./Surface");e.SIZE_ZERO=[0,0],e.prototype=Object.create(s.prototype),e.prototype.elementType="div",e.prototype.elementClass="famous-group",e.prototype.add=function(){return this.context.add.apply(this.context,arguments)},e.prototype.render=function(){return s.prototype.render.call(this)},e.prototype.deploy=function(t){this.context.migrate(t)},e.prototype.recall=function(){this._container=document.createDocumentFragment(),this.context.migrate(this._container)},e.prototype.commit=function(t){var i=t.transform,n=t.origin,r=t.opacity,a=t.size,h=s.prototype.commit.call(this,{allocator:t.allocator,transform:o.thenMove(i,[-n[0]*a[0],-n[1]*a[1],0]),opacity:r,origin:n,size:e.SIZE_ZERO});return(a[0]!==this._groupSize[0]||a[1]!==this._groupSize[1])&&(this._groupSize[0]=a[0],this._groupSize[1]=a[1],this.context.setSize(a)),this.context.update({transform:o.translate(-n[0]*a[0],-n[1]*a[1],0),origin:n,size:a}),h},i.exports=e},{"./Context":1,"./Surface":14,"./Transform":15}],9:[function(t,i){function e(t){this._transformGetter=null,this._opacityGetter=null,this._originGetter=null,this._alignGetter=null,this._sizeGetter=null,this._proportionGetter=null,this._legacyStates={},this._output={transform:o.identity,opacity:1,origin:null,align:null,size:null,proportions:null,target:null},t&&(t.transform&&this.transformFrom(t.transform),void 0!==t.opacity&&this.opacityFrom(t.opacity),t.origin&&this.originFrom(t.origin),t.align&&this.alignFrom(t.align),t.size&&this.sizeFrom(t.size),t.proportions&&this.proportionsFrom(t.proportions))}function n(){this._transformGetter&&(this._output.transform=this._transformGetter()),this._opacityGetter&&(this._output.opacity=this._opacityGetter()),this._originGetter&&(this._output.origin=this._originGetter()),this._alignGetter&&(this._output.align=this._alignGetter()),this._sizeGetter&&(this._output.size=this._sizeGetter()),this._proportionGetter&&(this._output.proportions=this._proportionGetter())}var o=t("./Transform"),s=t("../transitions/Transitionable"),r=t("../transitions/TransitionableTransform");e.prototype.transformFrom=function(t){return t instanceof Function?this._transformGetter=t:t instanceof Object&&t.get?this._transformGetter=t.get.bind(t):(this._transformGetter=null,this._output.transform=t),this},e.prototype.opacityFrom=function(t){return t instanceof Function?this._opacityGetter=t:t instanceof Object&&t.get?this._opacityGetter=t.get.bind(t):(this._opacityGetter=null,this._output.opacity=t),this},e.prototype.originFrom=function(t){return t instanceof Function?this._originGetter=t:t instanceof Object&&t.get?this._originGetter=t.get.bind(t):(this._originGetter=null,this._output.origin=t),this},e.prototype.alignFrom=function(t){return t instanceof Function?this._alignGetter=t:t instanceof Object&&t.get?this._alignGetter=t.get.bind(t):(this._alignGetter=null,this._output.align=t),this},e.prototype.sizeFrom=function(t){return t instanceof Function?this._sizeGetter=t:t instanceof Object&&t.get?this._sizeGetter=t.get.bind(t):(this._sizeGetter=null,this._output.size=t),this},e.prototype.proportionsFrom=function(t){return t instanceof Function?this._proportionGetter=t:t instanceof Object&&t.get?this._proportionGetter=t.get.bind(t):(this._proportionGetter=null,this._output.proportions=t),this},e.prototype.setTransform=function(t,i,e){return i||this._legacyStates.transform?(this._legacyStates.transform||(this._legacyStates.transform=new r(this._output.transform)),this._transformGetter||this.transformFrom(this._legacyStates.transform),this._legacyStates.transform.set(t,i,e),this):this.transformFrom(t)},e.prototype.setOpacity=function(t,i,e){return i||this._legacyStates.opacity?(this._legacyStates.opacity||(this._legacyStates.opacity=new s(this._output.opacity)),this._opacityGetter||this.opacityFrom(this._legacyStates.opacity),this._legacyStates.opacity.set(t,i,e)):this.opacityFrom(t)},e.prototype.setOrigin=function(t,i,e){return i||this._legacyStates.origin?(this._legacyStates.origin||(this._legacyStates.origin=new s(this._output.origin||[0,0])),this._originGetter||this.originFrom(this._legacyStates.origin),this._legacyStates.origin.set(t,i,e),this):this.originFrom(t)},e.prototype.setAlign=function(t,i,e){return i||this._legacyStates.align?(this._legacyStates.align||(this._legacyStates.align=new s(this._output.align||[0,0])),this._alignGetter||this.alignFrom(this._legacyStates.align),this._legacyStates.align.set(t,i,e),this):this.alignFrom(t)},e.prototype.setSize=function(t,i,e){return t&&(i||this._legacyStates.size)?(this._legacyStates.size||(this._legacyStates.size=new s(this._output.size||[0,0])),this._sizeGetter||this.sizeFrom(this._legacyStates.size),this._legacyStates.size.set(t,i,e),this):this.sizeFrom(t)},e.prototype.setProportions=function(t,i,e){return t&&(i||this._legacyStates.proportions)?(this._legacyStates.proportions||(this._legacyStates.proportions=new s(this._output.proportions||[0,0])),this._proportionGetter||this.proportionsFrom(this._legacyStates.proportions),this._legacyStates.proportions.set(t,i,e),this):this.proportionsFrom(t)},e.prototype.halt=function(){this._legacyStates.transform&&this._legacyStates.transform.halt(),this._legacyStates.opacity&&this._legacyStates.opacity.halt(),this._legacyStates.origin&&this._legacyStates.origin.halt(),this._legacyStates.align&&this._legacyStates.align.halt(),this._legacyStates.size&&this._legacyStates.size.halt(),this._legacyStates.proportions&&this._legacyStates.proportions.halt(),this._transformGetter=null,this._opacityGetter=null,this._originGetter=null,this._alignGetter=null,this._sizeGetter=null,this._proportionGetter=null},e.prototype.getTransform=function(){return this._transformGetter()},e.prototype.getFinalTransform=function(){return this._legacyStates.transform?this._legacyStates.transform.getFinal():this._output.transform},e.prototype.getOpacity=function(){return this._opacityGetter()},e.prototype.getOrigin=function(){return this._originGetter()},e.prototype.getAlign=function(){return this._alignGetter()},e.prototype.getSize=function(){return this._sizeGetter?this._sizeGetter():this._output.size},e.prototype.getProportions=function(){return this._proportionGetter?this._proportionGetter():this._output.proportions},e.prototype.modify=function(t){return n.call(this),this._output.target=t,this._output},i.exports=e},{"../transitions/Transitionable":88,"../transitions/TransitionableTransform":89,"./Transform":15}],10:[function(t,i){function e(t){this._value=t,this.eventOutput=null}function n(){this.eventOutput=new o,this.eventOutput.bindThis(this),o.setOutputHandler(this,this.eventOutput)}var o=t("./EventHandler");e.patch=function(t){for(var i=new e(t),n=1;n=0&&(this._dirtyClasses.push(this.classList.splice(i,1)[0]),this._classesDirty=!0),this},e.prototype.toggleClass=function(t){var i=this.classList.indexOf(t);return i>=0?this.removeClass(t):this.addClass(t),this},e.prototype.setClasses=function(t){var i=0,e=[];for(i=0;it?-1:1}var s={};s.precision=1e-6,s.identity=[1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1],s.multiply4x4=function(t,i){return[t[0]*i[0]+t[4]*i[1]+t[8]*i[2]+t[12]*i[3],t[1]*i[0]+t[5]*i[1]+t[9]*i[2]+t[13]*i[3],t[2]*i[0]+t[6]*i[1]+t[10]*i[2]+t[14]*i[3],t[3]*i[0]+t[7]*i[1]+t[11]*i[2]+t[15]*i[3],t[0]*i[4]+t[4]*i[5]+t[8]*i[6]+t[12]*i[7],t[1]*i[4]+t[5]*i[5]+t[9]*i[6]+t[13]*i[7],t[2]*i[4]+t[6]*i[5]+t[10]*i[6]+t[14]*i[7],t[3]*i[4]+t[7]*i[5]+t[11]*i[6]+t[15]*i[7],t[0]*i[8]+t[4]*i[9]+t[8]*i[10]+t[12]*i[11],t[1]*i[8]+t[5]*i[9]+t[9]*i[10]+t[13]*i[11],t[2]*i[8]+t[6]*i[9]+t[10]*i[10]+t[14]*i[11],t[3]*i[8]+t[7]*i[9]+t[11]*i[10]+t[15]*i[11],t[0]*i[12]+t[4]*i[13]+t[8]*i[14]+t[12]*i[15],t[1]*i[12]+t[5]*i[13]+t[9]*i[14]+t[13]*i[15],t[2]*i[12]+t[6]*i[13]+t[10]*i[14]+t[14]*i[15],t[3]*i[12]+t[7]*i[13]+t[11]*i[14]+t[15]*i[15]]},s.multiply=function(t,i){return[t[0]*i[0]+t[4]*i[1]+t[8]*i[2],t[1]*i[0]+t[5]*i[1]+t[9]*i[2],t[2]*i[0]+t[6]*i[1]+t[10]*i[2],0,t[0]*i[4]+t[4]*i[5]+t[8]*i[6],t[1]*i[4]+t[5]*i[5]+t[9]*i[6],t[2]*i[4]+t[6]*i[5]+t[10]*i[6],0,t[0]*i[8]+t[4]*i[9]+t[8]*i[10],t[1]*i[8]+t[5]*i[9]+t[9]*i[10],t[2]*i[8]+t[6]*i[9]+t[10]*i[10],0,t[0]*i[12]+t[4]*i[13]+t[8]*i[14]+t[12],t[1]*i[12]+t[5]*i[13]+t[9]*i[14]+t[13],t[2]*i[12]+t[6]*i[13]+t[10]*i[14]+t[14],1] +},s.thenMove=function(t,i){return i[2]||(i[2]=0),[t[0],t[1],t[2],0,t[4],t[5],t[6],0,t[8],t[9],t[10],0,t[12]+i[0],t[13]+i[1],t[14]+i[2],1]},s.moveThen=function(t,i){t[2]||(t[2]=0);var e=t[0]*i[0]+t[1]*i[4]+t[2]*i[8],n=t[0]*i[1]+t[1]*i[5]+t[2]*i[9],o=t[0]*i[2]+t[1]*i[6]+t[2]*i[10];return s.thenMove(i,[e,n,o])},s.translate=function(t,i,e){return void 0===e&&(e=0),[1,0,0,0,0,1,0,0,0,0,1,0,t,i,e,1]},s.thenScale=function(t,i){return[i[0]*t[0],i[1]*t[1],i[2]*t[2],0,i[0]*t[4],i[1]*t[5],i[2]*t[6],0,i[0]*t[8],i[1]*t[9],i[2]*t[10],0,i[0]*t[12],i[1]*t[13],i[2]*t[14],1]},s.scale=function(t,i,e){return void 0===e&&(e=1),void 0===i&&(i=t),[t,0,0,0,0,i,0,0,0,0,e,0,0,0,0,1]},s.rotateX=function(t){var i=Math.cos(t),e=Math.sin(t);return[1,0,0,0,0,i,e,0,0,-e,i,0,0,0,0,1]},s.rotateY=function(t){var i=Math.cos(t),e=Math.sin(t);return[i,0,-e,0,0,1,0,0,e,0,i,0,0,0,0,1]},s.rotateZ=function(t){var i=Math.cos(t),e=Math.sin(t);return[i,e,0,0,-e,i,0,0,0,0,1,0,0,0,0,1]},s.rotate=function(t,i,e){var n=Math.cos(t),o=Math.sin(t),s=Math.cos(i),r=Math.sin(i),a=Math.cos(e),h=Math.sin(e),c=[s*a,n*h+o*r*a,o*h-n*r*a,0,-s*h,n*a-o*r*h,o*a+n*r*h,0,r,-o*s,n*s,0,0,0,0,1];return c},s.rotateAxis=function(t,i){var e=Math.sin(i),n=Math.cos(i),o=1-n,s=t[0]*t[0]*o,r=t[0]*t[1]*o,a=t[0]*t[2]*o,h=t[1]*t[1]*o,c=t[1]*t[2]*o,p=t[2]*t[2]*o,u=t[0]*e,l=t[1]*e,f=t[2]*e,d=[s+n,r+f,a-l,0,r-f,h+n,c+u,0,a+l,c-u,p+n,0,0,0,0,1];return d},s.aboutOrigin=function(t,i){var e=t[0]-(t[0]*i[0]+t[1]*i[4]+t[2]*i[8]),n=t[1]-(t[0]*i[1]+t[1]*i[5]+t[2]*i[9]),o=t[2]-(t[0]*i[2]+t[1]*i[6]+t[2]*i[10]);return s.thenMove(i,[e,n,o])},s.skew=function(t,i,e){return[1,Math.tan(i),0,0,Math.tan(e),1,0,0,0,Math.tan(t),1,0,0,0,0,1]},s.skewX=function(t){return[1,0,0,0,Math.tan(t),1,0,0,0,0,1,0,0,0,0,1]},s.skewY=function(t){return[1,Math.tan(t),0,0,0,1,0,0,0,0,1,0,0,0,0,1]},s.perspective=function(t){return[1,0,0,0,0,1,0,0,0,0,1,-1/t,0,0,0,1]},s.getTranslate=function(t){return[t[12],t[13],t[14]]},s.inverse=function(t){var i=t[5]*t[10]-t[6]*t[9],e=t[4]*t[10]-t[6]*t[8],n=t[4]*t[9]-t[5]*t[8],o=t[1]*t[10]-t[2]*t[9],s=t[0]*t[10]-t[2]*t[8],r=t[0]*t[9]-t[1]*t[8],a=t[1]*t[6]-t[2]*t[5],h=t[0]*t[6]-t[2]*t[4],c=t[0]*t[5]-t[1]*t[4],p=t[0]*i-t[1]*e+t[2]*n,u=1/p,l=[u*i,-u*o,u*a,0,-u*e,u*s,-u*h,0,u*n,-u*r,u*c,0,0,0,0,1];return l[12]=-t[12]*l[0]-t[13]*l[4]-t[14]*l[8],l[13]=-t[12]*l[1]-t[13]*l[5]-t[14]*l[9],l[14]=-t[12]*l[2]-t[13]*l[6]-t[14]*l[10],l},s.transpose=function(t){return[t[0],t[4],t[8],t[12],t[1],t[5],t[9],t[13],t[2],t[6],t[10],t[14],t[3],t[7],t[11],t[15]]},s.interpret=function(t){var i=[t[0],t[1],t[2]],r=o(i[0]),a=n(i),h=[i[0]+r*a,i[1],i[2]],c=2/e(h);if(c>=1/0)return{translate:s.getTranslate(t),rotate:[0,0,0],scale:[0,0,0],skew:[0,0,0]};var p=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1];p[0]=1-c*h[0]*h[0],p[5]=1-c*h[1]*h[1],p[10]=1-c*h[2]*h[2],p[1]=-c*h[0]*h[1],p[2]=-c*h[0]*h[2],p[6]=-c*h[1]*h[2],p[4]=p[1],p[8]=p[2],p[9]=p[6];var u=s.multiply(p,t),l=[u[5],u[6]],f=o(l[0]),d=n(l),_=[l[0]+f*d,l[1]],g=2/e(_),y=[1,0,0,0,0,0,0,0,0,0,0,0,0,0,0,1];y[5]=1-g*_[0]*_[0],y[10]=1-g*_[1]*_[1],y[6]=-g*_[0]*_[1],y[9]=y[6];var v=s.multiply(y,p),m=s.multiply(v,t),O=s.scale(m[0]<0?-1:1,m[5]<0?-1:1,m[10]<0?-1:1);m=s.multiply(m,O),v=s.multiply(O,v);var S={};return S.translate=s.getTranslate(t),S.rotate=[Math.atan2(-v[6],v[10]),Math.asin(v[2]),Math.atan2(-v[1],v[0])],S.rotate[0]||(S.rotate[0]=0,S.rotate[2]=Math.atan2(v[4],v[5])),S.scale=[m[0],m[5],m[10]],S.skew=[Math.atan2(m[9],S.scale[2]),Math.atan2(m[8],S.scale[2]),Math.atan2(m[4],S.scale[0])],Math.abs(S.rotate[0])+Math.abs(S.rotate[2])>1.5*Math.PI&&(S.rotate[1]=Math.PI-S.rotate[1],S.rotate[1]>Math.PI&&(S.rotate[1]-=2*Math.PI),S.rotate[1]<-Math.PI&&(S.rotate[1]+=2*Math.PI),S.rotate[0]<0?S.rotate[0]+=Math.PI:S.rotate[0]-=Math.PI,S.rotate[2]<0?S.rotate[2]+=Math.PI:S.rotate[2]-=Math.PI),S},s.average=function(t,i,e){e=void 0===e?.5:e;for(var n=s.interpret(t),o=s.interpret(i),r={translate:[0,0,0],rotate:[0,0,0],scale:[0,0,0],skew:[0,0,0]},a=0;3>a;a++)r.translate[a]=(1-e)*n.translate[a]+e*o.translate[a],r.rotate[a]=(1-e)*n.rotate[a]+e*o.rotate[a],r.scale[a]=(1-e)*n.scale[a]+e*o.scale[a],r.skew[a]=(1-e)*n.skew[a]+e*o.skew[a];return s.build(r)},s.build=function(t){var i=s.scale(t.scale[0],t.scale[1],t.scale[2]),e=s.skew(t.skew[0],t.skew[1],t.skew[2]),n=s.rotate(t.rotate[0],t.rotate[1],t.rotate[2]);return s.thenMove(s.multiply(s.multiply(n,e),i),t.translate)},s.equals=function(t,i){return!s.notEquals(t,i)},s.notEquals=function(t,i){return t===i?!1:!(t&&i)||t[12]!==i[12]||t[13]!==i[13]||t[14]!==i[14]||t[0]!==i[0]||t[1]!==i[1]||t[2]!==i[2]||t[4]!==i[4]||t[5]!==i[5]||t[6]!==i[6]||t[8]!==i[8]||t[9]!==i[9]||t[10]!==i[10]},s.normalizeRotation=function(t){var i=t.slice(0);for((i[0]===.5*Math.PI||i[0]===.5*-Math.PI)&&(i[0]=-i[0],i[1]=Math.PI-i[1],i[2]-=Math.PI),i[0]>.5*Math.PI&&(i[0]=i[0]-Math.PI,i[1]=Math.PI-i[1],i[2]-=Math.PI),i[0]<.5*-Math.PI&&(i[0]=i[0]+Math.PI,i[1]=-Math.PI-i[1],i[2]-=Math.PI);i[1]<-Math.PI;)i[1]+=2*Math.PI;for(;i[1]>=Math.PI;)i[1]-=2*Math.PI;for(;i[2]<-Math.PI;)i[2]+=2*Math.PI;for(;i[2]>=Math.PI;)i[2]-=2*Math.PI;return i},s.inFront=[1,0,0,0,0,1,0,0,0,0,1,0,0,0,.001,1],s.behind=[1,0,0,0,0,1,0,0,0,0,1,0,0,0,-.001,1],i.exports=s},{}],16:[function(t,i){function e(t){this._node=new s,this._eventInput=new n,this._eventOutput=new n,n.setInputHandler(this,this._eventInput),n.setOutputHandler(this,this._eventOutput),this.options=r.clone(this.constructor.DEFAULT_OPTIONS||e.DEFAULT_OPTIONS),this._optionsManager=new o(this.options),t&&this.setOptions(t)}var n=t("./EventHandler"),o=t("./OptionsManager"),s=t("./RenderNode"),r=t("../utilities/Utility");e.DEFAULT_OPTIONS={},e.prototype.getOptions=function(t){return this._optionsManager.getOptions(t)},e.prototype.setOptions=function(t){this._optionsManager.patch(t)},e.prototype.add=function(){return this._node.add.apply(this._node,arguments)},e.prototype._add=e.prototype.add,e.prototype.render=function(){return this._node.render()},e.prototype.getSize=function(){return this._node&&this._node.getSize?this._node.getSize.apply(this._node,arguments)||this.options.size:this.options.size},i.exports=e},{"../utilities/Utility":95,"./EventHandler":7,"./OptionsManager":10,"./RenderNode":11}],17:[function(t,i){function e(t){t||(t=[]),t instanceof Array&&(t={array:t}),this._=null,this.index=t.index||0,t.array?this._=new this.constructor.Backing(t.array):t._&&(this._=t._),this.index===this._.firstIndex&&(this._.firstNode=this),this.index===this._.firstIndex+this._.array.length-1&&(this._.lastNode=this),void 0!==t.loop&&(this._.loop=t.loop),void 0!==t.trackSize&&(this._.trackSize=t.trackSize),this._previousNode=null,this._nextNode=null}e.Backing=function(t){this.array=t,this.firstIndex=0,this.loop=!1,this.firstNode=null,this.lastNode=null,this.cumulativeSizes=[[0,0]],this.sizeDirty=!0,this.trackSize=!1},e.Backing.prototype.getValue=function(t){var i=t-this.firstIndex;return 0>i||i>=this.array.length?null:this.array[i]},e.Backing.prototype.setValue=function(t,i){this.array[t-this.firstIndex]=i},e.Backing.prototype.getSize=function(t){return this.cumulativeSizes[t]},e.Backing.prototype.calculateSize=function(t){t=t||this.array.length;for(var i=[0,0],e=0;t>e;e++){var n=this.array[e].getSize();if(!n)return void 0;void 0!==i[0]&&(void 0===n[0]?i[0]=void 0:i[0]+=n[0]),void 0!==i[1]&&(void 0===n[1]?i[1]=void 0:i[1]+=n[1]),this.cumulativeSizes[e+1]=i.slice()}return this.sizeDirty=!1,i},e.Backing.prototype.reindex=function(t,i,e){if(this.array[0]){for(var n=0,o=this.firstIndex,s=e-i,r=this.firstNode;t-1>o;)r=r.getNext(),o++;var a=r;for(n=0;i>n;n++)r=r.getNext(),r&&(r._previousNode=a);var h=r?r.getNext():null;for(a._nextNode=null,r=a,n=0;e>n;n++)r=r.getNext();if(o+=e,r!==h&&(r._nextNode=h,h&&(h._previousNode=r)),h)for(r=h,o++;r&&os-h){var c=new window.CustomEvent("click",{bubbles:!0,detail:a});n[s]=i,i.target.dispatchEvent(c)}delete e[a.identifier]}}),window.addEventListener("click",function(t){var e=o();for(var s in n){var r=n[s];i>e-s?t instanceof window.MouseEvent&&t.target===r.target&&t.stopPropagation():delete n[s]}},!0)}}()},{}],27:[function(t,i){function e(t,i){this._eventInput=new o,this._eventOutput=new o,o.setInputHandler(this,this._eventInput),o.setOutputHandler(this,this._eventOutput),this._syncs={},t&&this.addSync(t),i&&this.setOptions(i)}function n(t,i){s[t]&&(this._syncs[t]=new s[t](i),this.pipeSync(t))}var o=t("../core/EventHandler");e.DIRECTION_X=0,e.DIRECTION_Y=1,e.DIRECTION_Z=2;var s={};e.register=function(t){for(var i in t)if(s[i]){if(s[i]!==t[i])throw new Error("Conflicting sync classes for key: "+i)}else s[i]=t[i]},e.prototype.setOptions=function(t){for(var i in this._syncs)this._syncs[i].setOptions(t)},e.prototype.pipeSync=function(t){var i=this._syncs[t];this._eventInput.pipe(i),i.pipe(this._eventOutput)},e.prototype.unpipeSync=function(t){var i=this._syncs[t];this._eventInput.unpipe(i),i.unpipe(this._eventOutput)},e.prototype.addSync=function(t){if(t instanceof Array)for(var i=0;ithis.options.clickThreshold&&t.stopPropagation()}.bind(this),!0),this._payload={delta:null,position:null,velocity:null,clientX:0,clientY:0,offsetX:0,offsetY:0},this._positionHistory=[],this._position=null,this._prevCoord=void 0,this._prevTime=void 0,this._down=!1,this._moved=!1,this._displacement=[0,0],this._documentActive=!1}function n(t){var i,e;this.options.preventDefault&&t.preventDefault();var n=t.clientX,o=t.clientY;this._prevCoord=[n,o],this._prevTime=Date.now(),this._down=!0,this._move=!1,void 0!==this.options.direction?(this._position=0,i=0,e=0):(this._position=[0,0],i=[0,0],e=[0,0]),this.options.clickThreshold&&(this._displacement=[0,0]);var s=this._payload;s.delta=i,s.position=this._position,s.velocity=e,s.clientX=n,s.clientY=o,s.offsetX=t.offsetX,s.offsetY=t.offsetY,this._positionHistory.push({position:s.position.slice?s.position.slice(0):s.position,time:this._prevTime}),this._eventOutput.emit("start",s),this._documentActive=!1}function o(t){if(this._prevCoord){var i=this._prevCoord;this._prevTime;var n=t.clientX,o=t.clientY,s=Date.now(),r=n-i[0],a=o-i[1];this.options.rails&&(Math.abs(r)>Math.abs(a)?a=0:r=0);var h,p,u=Math.max(s-this._positionHistory[0].time,c),l=this.options.scale;this.options.direction===e.DIRECTION_X?(p=l*r,this._position+=p,h=l*(this._position-this._positionHistory[0].position)/u):this.options.direction===e.DIRECTION_Y?(p=l*a,this._position+=p,h=l*(this._position-this._positionHistory[0].position)/u):(p=[l*r,l*a],h=[l*(this._position[0]-this._positionHistory[0].position[0])/u,l*(this._position[1]-this._positionHistory[0].position[1])/u],this._position[0]+=p[0],this._position[1]+=p[1]),this.options.clickThreshold!==!1&&(this._displacement[0]+=r,this._displacement[1]+=a);var f=this._payload;f.delta=p,f.position=this._position,f.velocity=h,f.clientX=n,f.clientY=o,f.offsetX=t.offsetX,f.offsetY=t.offsetY,this._positionHistory.length===this.options.velocitySampleLength&&this._positionHistory.shift(),this._positionHistory.push({position:f.position.slice?f.position.slice(0):f.position,time:s}),this._eventOutput.emit("update",f),this._prevCoord=[n,o],this._prevTime=s,this._move=!0}}function s(){this._down&&(this._eventOutput.emit("end",this._payload),this._prevCoord=void 0,this._prevTime=void 0,this._down=!1,this._move=!1,this._positionHistory=[])}function r(t){if(this._down&&this._move&&!this._documentActive){var i=o.bind(this),e=function(t){s.call(this,t),document.removeEventListener("mousemove",i),document.removeEventListener("mouseup",e)}.bind(this,t);document.addEventListener("mousemove",i),document.addEventListener("mouseup",e),this._documentActive=!0}}var a=t("../core/EventHandler"),h=t("../core/OptionsManager");e.DEFAULT_OPTIONS={clickThreshold:void 0,direction:void 0,rails:!1,scale:1,propogate:!0,velocitySampleLength:10,preventDefault:!0},e.DIRECTION_X=0,e.DIRECTION_Y=1;var c=8;e.prototype.getOptions=function(){return this.options},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},i.exports=e},{"../core/EventHandler":7,"../core/OptionsManager":10}],29:[function(t,i){function e(t){n.call(this),this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new o(this.options),t&&this.setOptions(t),this._displacement=0,this._previousDistance=0}var n=t("./TwoFingerSync"),o=t("../core/OptionsManager");e.prototype=Object.create(n.prototype),e.prototype.constructor=e,e.DEFAULT_OPTIONS={scale:1},e.prototype._startUpdate=function(t){this._previousDistance=n.calculateDistance(this.posA,this.posB),this._displacement=0,this._eventOutput.emit("start",{count:t.touches.length,touches:[this.touchAId,this.touchBId],distance:this._dist,center:n.calculateCenter(this.posA,this.posB)})},e.prototype._moveUpdate=function(t){var i=n.calculateDistance(this.posA,this.posB),e=n.calculateCenter(this.posA,this.posB),o=this.options.scale,s=o*(i-this._previousDistance),r=s/t;this._previousDistance=i,this._displacement+=s,this._eventOutput.emit("update",{delta:s,velocity:r,distance:i,displacement:this._displacement,center:e,touches:[this.touchAId,this.touchBId]})},e.prototype.getOptions=function(){return this.options},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},i.exports=e},{"../core/OptionsManager":10,"./TwoFingerSync":35}],30:[function(t,i){function e(t){n.call(this),this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new o(this.options),t&&this.setOptions(t),this._angle=0,this._previousAngle=0}var n=t("./TwoFingerSync"),o=t("../core/OptionsManager");e.prototype=Object.create(n.prototype),e.prototype.constructor=e,e.DEFAULT_OPTIONS={scale:1},e.prototype._startUpdate=function(t){this._angle=0,this._previousAngle=n.calculateAngle(this.posA,this.posB);var i=n.calculateCenter(this.posA,this.posB);this._eventOutput.emit("start",{count:t.touches.length,angle:this._angle,center:i,touches:[this.touchAId,this.touchBId]})},e.prototype._moveUpdate=function(t){var i=this.options.scale,e=n.calculateAngle(this.posA,this.posB),o=n.calculateCenter(this.posA,this.posB),s=i*(e-this._previousAngle),r=s/t;this._angle+=s,this._eventOutput.emit("update",{delta:s,velocity:r,angle:this._angle,center:o,touches:[this.touchAId,this.touchBId]}),this._previousAngle=e},e.prototype.getOptions=function(){return this.options},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},i.exports=e},{"../core/OptionsManager":10,"./TwoFingerSync":35}],31:[function(t,i){function e(t){o.call(this),this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new s(this.options),t&&this.setOptions(t),this._scaleFactor=1,this._startDist=0,this._eventInput.on("pipe",n.bind(this))}function n(){this.touchAId=void 0,this.touchBId=void 0}var o=t("./TwoFingerSync"),s=t("../core/OptionsManager");e.prototype=Object.create(o.prototype),e.prototype.constructor=e,e.DEFAULT_OPTIONS={scale:1},e.prototype._startUpdate=function(t){this._scaleFactor=1,this._startDist=o.calculateDistance(this.posA,this.posB),this._eventOutput.emit("start",{count:t.touches.length,touches:[this.touchAId,this.touchBId],distance:this._startDist,center:o.calculateCenter(this.posA,this.posB)})},e.prototype._moveUpdate=function(t){var i=this.options.scale,e=o.calculateDistance(this.posA,this.posB),n=o.calculateCenter(this.posA,this.posB),s=(e-this._startDist)/this._startDist,r=Math.max(1+i*s,0),a=(r-this._scaleFactor)/t;this._eventOutput.emit("update",{delta:s,scale:r,velocity:a,distance:e,center:n,touches:[this.touchAId,this.touchBId]}),this._scaleFactor=r},e.prototype.getOptions=function(){return this.options},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},i.exports=e},{"../core/OptionsManager":10,"./TwoFingerSync":35}],32:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new a(this.options),t&&this.setOptions(t),this._payload={delta:null,position:null,velocity:null,slip:!0},this._eventInput=new s,this._eventOutput=new s,s.setInputHandler(this,this._eventInput),s.setOutputHandler(this,this._eventOutput),this._position=void 0===this.options.direction?[0,0]:0,this._prevTime=void 0,this._prevVel=void 0,this._eventInput.on("mousewheel",o.bind(this)),this._eventInput.on("wheel",o.bind(this)),this._inProgress=!1,this._loopBound=!1}function n(){if(this._inProgress&&c()-this._prevTime>this.options.stallTime){this._inProgress=!1;var t=Math.abs(this._prevVel)>=this.options.minimumEndSpeed?this._prevVel:0,i=this._payload;i.position=this._position,i.velocity=t,i.slip=!0,this._eventOutput.emit("end",i)}}function o(t){this.options.preventDefault&&t.preventDefault(),this._inProgress||(this._inProgress=!0,this._position=void 0===this.options.direction?[0,0]:0,g=this._payload,g.slip=!0,g.position=this._position,g.clientX=t.clientX,g.clientY=t.clientY,g.offsetX=t.offsetX,g.offsetY=t.offsetY,this._eventOutput.emit("start",g),this._loopBound||(r.on("prerender",n.bind(this)),this._loopBound=!0));var i=c(),o=this._prevTime||i,s=void 0!==t.wheelDeltaX?t.wheelDeltaX:-t.deltaX,a=void 0!==t.wheelDeltaY?t.wheelDeltaY:-t.deltaY;1===t.deltaMode&&(s*=this.options.lineHeight,a*=this.options.lineHeight),this.options.rails&&(Math.abs(s)>Math.abs(a)?a=0:s=0);var p,u,l=Math.max(i-o,h),f=s/l,d=a/l,_=this.options.scale;this.options.direction===e.DIRECTION_X?(u=_*s,p=_*f,this._position+=u):this.options.direction===e.DIRECTION_Y?(u=_*a,p=_*d,this._position+=u):(u=[_*s,_*a],p=[_*f,_*d],this._position[0]+=u[0],this._position[1]+=u[1]);var g=this._payload;g.delta=u,g.velocity=p,g.position=this._position,g.slip=!0,this._eventOutput.emit("update",g),this._prevTime=i,this._prevVel=p}var s=t("../core/EventHandler"),r=t("../core/Engine"),a=t("../core/OptionsManager");e.DEFAULT_OPTIONS={direction:void 0,minimumEndSpeed:1/0,rails:!1,scale:1,stallTime:50,lineHeight:40,preventDefault:!0},e.DIRECTION_X=0,e.DIRECTION_Y=1;var h=8,c=Date.now;e.prototype.getOptions=function(){return this.options},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},i.exports=e},{"../core/Engine":4,"../core/EventHandler":7,"../core/OptionsManager":10}],33:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new h(this.options),t&&this.setOptions(t),this._eventOutput=new a,this._touchTracker=new r({touchLimit:this.options.touchLimit}),a.setOutputHandler(this,this._eventOutput),a.setInputHandler(this,this._touchTracker),this._touchTracker.on("trackstart",n.bind(this)),this._touchTracker.on("trackmove",o.bind(this)),this._touchTracker.on("trackend",s.bind(this)),this._payload={delta:null,position:null,velocity:null,clientX:void 0,clientY:void 0,count:0,touch:void 0},this._position=null}function n(t){var i,e;void 0!==this.options.direction?(this._position=0,i=0,e=0):(this._position=[0,0],i=[0,0],e=[0,0]);var n=this._payload;n.delta=e,n.position=this._position,n.velocity=i,n.clientX=t.x,n.clientY=t.y,n.count=t.count,n.touch=t.identifier,this._eventOutput.emit("start",n)}function o(t){var i=t.history,n=i[i.length-1],o=i[i.length-2],s=i[i.length-this.options.velocitySampleLength]?i[i.length-this.options.velocitySampleLength]:i[i.length-2],r=s.timestamp,a=n.timestamp,h=n.x-o.x,p=n.y-o.y,u=n.x-s.x,l=n.y-s.y;this.options.rails&&(Math.abs(h)>Math.abs(p)?p=0:h=0,Math.abs(u)>Math.abs(l)?l=0:u=0);var f,d,_=Math.max(a-r,c),g=u/_,y=l/_,v=this.options.scale;this.options.direction===e.DIRECTION_X?(d=v*h,f=v*g,this._position+=d):this.options.direction===e.DIRECTION_Y?(d=v*p,f=v*y,this._position+=d):(d=[v*h,v*p],f=[v*g,v*y],this._position[0]+=d[0],this._position[1]+=d[1]);var m=this._payload;m.delta=d,m.velocity=f,m.position=this._position,m.clientX=t.x,m.clientY=t.y,m.count=t.count,m.touch=t.identifier,this._eventOutput.emit("update",m)}function s(t){this._payload.count=t.count,this._eventOutput.emit("end",this._payload)}var r=t("./TouchTracker"),a=t("../core/EventHandler"),h=t("../core/OptionsManager");e.DEFAULT_OPTIONS={direction:void 0,rails:!1,touchLimit:1,velocitySampleLength:10,scale:1},e.DIRECTION_X=0,e.DIRECTION_Y=1;var c=8;e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},e.prototype.getOptions=function(){return this.options},i.exports=e},{"../core/EventHandler":7,"../core/OptionsManager":10,"./TouchTracker":34}],34:[function(t,i){function e(t,i,e){return{x:t.clientX,y:t.clientY,identifier:t.identifier,origin:i.origin,timestamp:c(),count:i.touches.length,history:e}}function n(t){if(!(t.touches.length>this.touchLimit)){this.isTouched=!0;for(var i=0;ithis.touchLimit))for(var i=0;in;n++){e[n]=[];for(var s=0;3>s;s++){for(var r=0,a=0;3>a;a++)r+=i[n][a]*t[a][s];e[n][s]=r}}return o.set(e)},e.prototype.transpose=function(){for(var t=[[],[],[]],i=this.get(),e=0;3>e;e++)for(var n=0;3>n;n++)t[e][n]=i[n][e];return o.set(t)},e.prototype.clone=function(){for(var t=this.get(),i=[],n=0;3>n;n++)i[n]=t[n].slice();return new e(i)},i.exports=e},{"./Vector":41}],38:[function(t,i){function e(t,i,e,n){return 1===arguments.length?this.set(t):(this.w=void 0!==t?t:1,this.x=void 0!==i?i:0,this.y=void 0!==e?e:0,this.z=void 0!==n?n:0),this}var n=t("./Matrix"),o=new e(1,0,0,0);e.prototype.add=function(t){return o.setWXYZ(this.w+t.w,this.x+t.x,this.y+t.y,this.z+t.z)},e.prototype.sub=function(t){return o.setWXYZ(this.w-t.w,this.x-t.x,this.y-t.y,this.z-t.z)},e.prototype.scalarDivide=function(t){return this.scalarMultiply(1/t)},e.prototype.scalarMultiply=function(t){return o.setWXYZ(this.w*t,this.x*t,this.y*t,this.z*t)},e.prototype.multiply=function(t){var i=this.x,e=this.y,n=this.z,s=this.w,r=t.x,a=t.y,h=t.z,c=t.w||0;return o.setWXYZ(s*c-i*r-e*a-n*h,i*c+r*s+a*n-e*h,e*c+a*s+i*h-r*n,n*c+h*s+r*e-i*a)};var s=new e(1,0,0,0);e.prototype.rotateVector=function(t){return s.set(this.conj()),o.set(this.multiply(t).multiply(s))},e.prototype.inverse=function(){return o.set(this.conj().scalarDivide(this.normSquared()))},e.prototype.negate=function(){return this.scalarMultiply(-1)},e.prototype.conj=function(){return o.setWXYZ(this.w,-this.x,-this.y,-this.z)},e.prototype.normalize=function(t){return t=void 0===t?1:t,this.scalarDivide(t*this.norm())},e.prototype.makeFromAngleAndAxis=function(t,i){var e=i.normalize(),n=.5*t,o=-Math.sin(n);return this.x=o*e.x,this.y=o*e.y,this.z=o*e.z,this.w=Math.cos(n),this},e.prototype.setWXYZ=function(t,i,e,n){return o.clear(),this.w=t,this.x=i,this.y=e,this.z=n,this},e.prototype.set=function(t){return t instanceof Array?(this.w=0,this.x=t[0],this.y=t[1],this.z=t[2]):(this.w=t.w,this.x=t.x,this.y=t.y,this.z=t.z),this!==o&&o.clear(),this},e.prototype.put=function(t){t.set(o)},e.prototype.clone=function(){return new e(this)},e.prototype.clear=function(){return this.w=1,this.x=0,this.y=0,this.z=0,this},e.prototype.isEqual=function(t){return t.w===this.w&&t.x===this.x&&t.y===this.y&&t.z===this.z},e.prototype.dot=function(t){return this.w*t.w+this.x*t.x+this.y*t.y+this.z*t.z},e.prototype.normSquared=function(){return this.dot(this)},e.prototype.norm=function(){return Math.sqrt(this.normSquared())},e.prototype.isZero=function(){return!(this.x||this.y||this.z)},e.prototype.getTransform=function(){var t=this.normalize(1),i=t.x,e=t.y,n=t.z,o=t.w;return[1-2*e*e-2*n*n,2*i*e-2*n*o,2*i*n+2*e*o,0,2*i*e+2*n*o,1-2*i*i-2*n*n,2*e*n-2*i*o,0,2*i*n-2*e*o,2*e*n+2*i*o,1-2*i*i-2*e*e,0,0,0,0,1]};var r=new n;e.prototype.getMatrix=function(){var t=this.normalize(1),i=t.x,e=t.y,n=t.z,o=t.w;return r.set([[1-2*e*e-2*n*n,2*i*e+2*n*o,2*i*n-2*e*o],[2*i*e-2*n*o,1-2*i*i-2*n*n,2*e*n+2*i*o],[2*i*n+2*e*o,2*e*n-2*i*o,1-2*i*i-2*e*e]])};var a=1e-5;e.prototype.slerp=function(t,i){var e,n,s,r,h;return n=this.dot(t),1-n>a?(e=Math.acos(n),s=Math.sin(e),r=Math.sin((1-i)*e)/s,h=Math.sin(i*e)/s):(r=1-i,h=i),o.set(this.scalarMultiply(r/h).add(t).scalarMultiply(h))},i.exports=e},{"./Matrix":37}],39:[function(t,i){function e(t,i){return t+s()*(i-t)}function n(t,i){return t+(s()*(i-t+1)>>0)}function o(t,i,e,n){if(i=void 0!==i?i:0,e=void 0!==e?e:1,void 0!==n){for(var o=[],s=0;n>s;s++)o.push(t(i,e));return o}return t(i,e)}var s=Math.random,r={};r.integer=function(t,i,e){return o(n,t,i,e)},r.range=function(t,i,n){return o(e,t,i,n)},r.sign=function(t){return r.bool(t)?1:-1},r.bool=function(t){return t=void 0!==t?t:.5,s()1e-7?s.call(a,this.mult(t/i)):n.call(a,t,0,0)},e.prototype.clone=function(){return new e(this)},e.prototype.isZero=function(){return!(this.x||this.y||this.z)},e.prototype.set=function(t){return t instanceof Array?o.call(this,t):"number"==typeof t?r.call(this,t):s.call(this,t)},e.prototype.setXYZ=function(){return n.apply(this,arguments)},e.prototype.set1D=function(t){return r.call(this,t)},e.prototype.put=function(t){this===a?s.call(t,a):s.call(t,this)},e.prototype.clear=function(){return n.call(this,0,0,0)},e.prototype.cap=function h(h){if(1/0===h)return s.call(a,this);var t=this.norm();return t>h?s.call(a,this.mult(h/t)):s.call(a,this)},e.prototype.project=function(t){return t.mult(this.dot(t))},e.prototype.reflectAcross=function(t){return t.normalize().put(t),s(a,this.sub(this.project(t).mult(2)))},e.prototype.get=function(){return[this.x,this.y,this.z]},e.prototype.get1D=function(){return this.x},i.exports=e},{}],42:[function(t,i){i.exports={Matrix:t("./Matrix"),Quaternion:t("./Quaternion"),Random:t("./Random"),Utilities:t("./Utilities"),Vector:t("./Vector")}},{"./Matrix":37,"./Quaternion":38,"./Random":39,"./Utilities":40,"./Vector":41}],43:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),t&&this.setOptions(t),this._positionState=new c([0,0]),this._differential=[0,0],this._active=!0,this.sync=new l(["mouse","touch"],{scale:this.options.scale}),this.eventOutput=new p,p.setInputHandler(this,this.sync),p.setOutputHandler(this,this.eventOutput),a.call(this)}function n(t){var i=this.options,e=i.projection,n=i.snapX,o=i.snapY,s=e&_.x?t[0]:0,r=e&_.y?t[1]:0;return n>0&&(s-=s%n),o>0&&(r-=r%o),[s,r]}function o(){this._active&&(this._positionState.isActive()&&this._positionState.halt(),this.eventOutput.emit("start",{position:this.getPosition()}))}function s(t){if(this._active){var i=this.options;this._differential=t.position;var e=n.call(this,this._differential);this._differential[0]-=e[0],this._differential[1]-=e[1];var o=this.getPosition();if(o[0]+=e[0],o[1]+=e[1],i.xRange){var s=[i.xRange[0]+.5*i.snapX,i.xRange[1]-.5*i.snapX];o[0]=g(o[0],s)}if(i.yRange){var r=[i.yRange[0]+.5*i.snapY,i.yRange[1]-.5*i.snapY];o[1]=g(o[1],r)}this.eventOutput.emit("update",{position:o})}}function r(){this._active&&this.eventOutput.emit("end",{position:this.getPosition()})}function a(){this.sync.on("start",o.bind(this)),this.sync.on("update",s.bind(this)),this.sync.on("end",r.bind(this))}var h=t("../core/Transform"),c=t("../transitions/Transitionable"),p=t("../core/EventHandler"),u=t("../math/Utilities"),l=t("../inputs/GenericSync"),f=t("../inputs/MouseSync"),d=t("../inputs/TouchSync");l.register({mouse:f,touch:d});var _={x:1,y:2};e.DIRECTION_X=_.x,e.DIRECTION_Y=_.y;var g=u.clamp;e.DEFAULT_OPTIONS={projection:_.x|_.y,scale:1,xRange:null,yRange:null,snapX:0,snapY:0,transition:{duration:0}},e.prototype.setOptions=function(t){var i=this.options;if(void 0!==t.projection){var e=t.projection;this.options.projection=0,["x","y"].forEach(function(t){-1!==e.indexOf(t)&&(i.projection|=_[t])})}void 0!==t.scale&&(i.scale=t.scale,this.sync.setOptions({scale:t.scale})),void 0!==t.xRange&&(i.xRange=t.xRange),void 0!==t.yRange&&(i.yRange=t.yRange),void 0!==t.snapX&&(i.snapX=t.snapX),void 0!==t.snapY&&(i.snapY=t.snapY)},e.prototype.getPosition=function(){return this._positionState.get()},e.prototype.setRelativePosition=function(t,i,e){var n=this.getPosition(),o=[n[0]+t[0],n[1]+t[1]];this.setPosition(o,i,e)},e.prototype.setPosition=function(t,i,e){this._positionState.isActive()&&this._positionState.halt(),this._positionState.set(t,i,e)},e.prototype.activate=function(){this._active=!0},e.prototype.deactivate=function(){this._active=!1},e.prototype.toggle=function(){this._active=!this._active},e.prototype.modify=function(t){var i=this.getPosition();return{transform:h.translate(i[0],i[1]),target:t}},i.exports=e},{"../core/EventHandler":7,"../core/Transform":15,"../inputs/GenericSync":27,"../inputs/MouseSync":28,"../inputs/TouchSync":33,"../math/Utilities":40,"../transitions/Transitionable":88}],44:[function(t,i){function e(t,i){this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new o(this.options),t&&this.setOptions(t),i||(i=0),this.transitionHelper=new n(i)}var n=t("../transitions/Transitionable"),o=t("../core/OptionsManager");e.DEFAULT_OPTIONS={cull:!1,transition:!0,pulseInTransition:!0,pulseOutTransition:!0},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},e.prototype.show=function(t,i){t=t||this.options.transition,this.set(1,t,i)},e.prototype.hide=function(t,i){t=t||this.options.transition,this.set(0,t,i)},e.prototype.set=function(t,i,e){this.halt(),this.transitionHelper.set(t,i,e)},e.prototype.halt=function(){this.transitionHelper.halt()},e.prototype.isVisible=function(){return this.transitionHelper.get()>0},e.prototype.modify=function(t){var i=this.transitionHelper.get();return this.options.cull&&!i?void 0:{opacity:i,target:t}},i.exports=e},{"../core/OptionsManager":10,"../transitions/Transitionable":88}],45:[function(t,i){function e(){this._chain=[],arguments.length&&this.addModifier.apply(this,arguments)}e.prototype.addModifier=function(){Array.prototype.push.apply(this._chain,arguments)},e.prototype.removeModifier=function(t){var i=this._chain.indexOf(t);0>i||this._chain.splice(i,1)},e.prototype.modify=function(t){for(var i=this._chain,e=t,n=0;n-1;t--)r.call(this,t)}function h(t,i){var e=this._agentData[this._constraints[t]];return e.agent.applyConstraint(e.targets,e.source,i)}function c(t){for(var i=0;i-1;e--)h.call(this,e,t);i++}}function p(t,i){t.integrateVelocity(i),this.options.velocityCap&&t.velocity.cap(this.options.velocityCap).put(t.velocity)}function u(t,i){t.integrateAngularMomentum(i),t.updateAngularVelocity(),this.options.angularVelocityCap&&t.angularVelocity.cap(this.options.angularVelocityCap).put(t.angularVelocity)}function l(t,i){t.integrateOrientation(i)}function f(t,i){t.integratePosition(i),t.emit(T.update,t)}function d(t){a.call(this,t),this.forEach(p,t),this.forEachBody(u,t),c.call(this,t),this.forEachBody(l,t),this.forEach(f,t)}function _(){var t=0,i=0;return this.forEach(function(e){i=e.getEnergy(),t+=i}),t}function g(){var t=0;for(var i in this._agentData)t+=this.getAgentEnergy(i);return t}var y=t("../core/EventHandler"),v=17,m=1e3/120,O=17,S=Date.now,T={start:"start",update:"update",end:"end"};e.DEFAULT_OPTIONS={constraintSteps:1,sleepTolerance:1e-7,velocityCap:void 0,angularVelocityCap:void 0},e.prototype.setOptions=function(t){for(var i in t)this.options[i]&&(this.options[i]=t[i])},e.prototype.addBody=function(t){return t._engine=this,t.isBody?(this._bodies.push(t),this._hasBodies=!0):this._particles.push(t),t.on("start",this.wake.bind(this)),t},e.prototype.removeBody=function(t){var i=t.isBody?this._bodies:this._particles,e=i.indexOf(t);if(e>-1){for(var n in this._agentData)this._agentData.hasOwnProperty(n)&&this.detachFrom(this._agentData[n].id,t);i.splice(e,1)}0===this.getBodies().length&&(this._hasBodies=!1)},e.prototype.attach=function(t,i,e){if(this.wake(),t instanceof Array){for(var n=[],s=0;s-1&&n.splice(o,1)}},e.prototype.detachAll=function(){this._agentData={},this._forces=[],this._constraints=[],this._currAgentId=0},e.prototype.getAgent=function(t){return s.call(this,t).agent},e.prototype.getParticles=function(){return this._particles},e.prototype.getBodies=function(){return this._bodies},e.prototype.getParticlesAndBodies=function(){return this.getParticles().concat(this.getBodies())},e.prototype.forEachParticle=function(t,i){for(var e=this.getParticles(),n=0,o=e.length;o>n;n++)t.call(this,e[n],i)},e.prototype.forEachBody=function(t,i){if(this._hasBodies)for(var e=this.getBodies(),n=0,o=e.length;o>n;n++)t.call(this,e[n],i)},e.prototype.forEach=function(t,i){this.forEachParticle(t,i),this.forEachBody(t,i)},e.prototype.getAgentEnergy=function(t){var i=s.call(this,t);return i.agent.getEnergy(i.targets,i.source)},e.prototype.getEnergy=function(){return _.call(this)+g.call(this)},e.prototype.step=function(){if(!this.isSleeping()){var t=S(),i=t-this._prevTime;this._prevTime=t,m>i||(i>O&&(i=O),d.call(this,v),this.emit(T.update,this),this.getEnergy()w){if(u.set(l.normalize()),this._eventOutput){var z={target:y,source:i,overlap:w,normal:u};this._eventOutput.emit("preCollision",z),this._eventOutput.emit("collision",z)}var I=c>=w?((1+p)*u.dot(f)+h/e*(w-c))/(E+e/b):(1+p)*u.dot(f)/(E+e/b);u.mult(e*I).put(d),d.mult(-1).put(_),i.applyImpulse(d),y.applyImpulse(_),this._eventOutput&&this._eventOutput.emit("postCollision",z)}}}},i.exports=e},{"../../math/Vector":41,"./Constraint":55}],55:[function(t,i){function e(){this.options=this.options||{},this._eventOutput=new n,n.setOutputHandler(this,this._eventOutput)}var n=t("../../core/EventHandler");e.prototype.setOptions=function(t){this._eventOutput.emit("change",t)},e.prototype.applyConstraint=function(){},e.prototype.getEnergy=function(){return 0},i.exports=e},{"../../core/EventHandler":7}],56:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),t&&this.setOptions(t),this.J=new o,this.impulse=new o,n.call(this)}var n=t("./Constraint"),o=t("../../math/Vector");e.prototype=Object.create(n.prototype),e.prototype.constructor=e;var s=1e-7,r=Math.PI;e.DEFAULT_OPTIONS={equation:function(){return 0},plane:function(t,i,e){return e},period:0,dampingRatio:0},e.prototype.setOptions=function(t){for(var i in t)this.options[i]=t[i]},e.prototype.applyConstraint=function(t,i,e){for(var n=this.options,o=this.impulse,a=this.J,h=n.equation,c=n.plane,p=n.dampingRatio,u=n.period,l=0;li?-((1+c)*a.dot(s)+u/n*(i-l))/(r*n+f):-((1+c)*a.dot(s))/(r*n+f);p.set(a.mult(n*_)),t.applyImpulse(p),t.setPosition(o.add(a.mult(-i)))}this._eventOutput&&this._eventOutput.emit("postCollision",d)}function r(t,i){var n=this.options.onContact,o=t.position,s=this.options.normal;n===e.ON_CONTACT.REFLECT&&t.setPosition(o.add(s.mult(-i)))}var a=t("./Constraint"),h=t("../../math/Vector");e.prototype=Object.create(a.prototype),e.prototype.constructor=e,e.ON_CONTACT={REFLECT:0,SILENT:1},e.DEFAULT_OPTIONS={restitution:.5,drift:.5,slop:0,normal:[1,0,0],distance:0,onContact:e.ON_CONTACT.REFLECT},e.prototype.setOptions=function(t){void 0!==t.normal&&(t.normal instanceof h&&(this.options.normal=t.normal.clone()),t.normal instanceof Array&&(this.options.normal=new h(t.normal))),void 0!==t.restitution&&(this.options.restitution=t.restitution),void 0!==t.drift&&(this.options.drift=t.drift),void 0!==t.slop&&(this.options.slop=t.slop),void 0!==t.distance&&(this.options.distance=t.distance),void 0!==t.onContact&&(this.options.onContact=t.onContact)},e.prototype.applyConstraint=function(t,i,e){for(var a=this.options.normal,h=0;h=f&&(0>d?s.call(this,c,f,e):r.call(this,c,f,e))}},i.exports=e},{"../../math/Vector":41,"./Constraint":55}],61:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),t&&this.setOptions(t),o.call(this,t.sides||this.options.sides),r.call(this)}function n(t,i,n){var o,s=e.SIDES;switch(parseInt(t)){case s.LEFT:o=i[0]*n[0];break;case s.TOP:o=i[1]*n[1];break;case s.FRONT:o=i[2]*n[2];break;case s.RIGHT:o=i[0]*(1-n[0]);break;case s.BOTTOM:o=i[1]*(1-n[1]);break;case s.BACK:o=i[2]*(1-n[2])}return o}function o(t){this.components={};for(var i=this.components,e=0;e_&&_>c&&(n.set(o.normalize(s*f*u(_,h)).cap(a)),l.applyForce(n)))},i.exports=e},{"../../math/Vector":41,"./Force":64}],66:[function(t,i){function e(t){n.call(this,t)}var n=t("./Drag");e.prototype=Object.create(n.prototype),e.prototype.constructor=e,e.DEFAULT_OPTIONS=n.DEFAULT_OPTIONS,e.FORCE_FUNCTIONS=n.FORCE_FUNCTIONS,e.FORCE_FUNCTIONS={LINEAR:function(t){return t},QUADRATIC:function(t){return t.mult(t.norm())}},e.prototype.applyForce=function(t){var i,e,n=this.options.strength,o=this.options.forceFunction,s=this.force;for(i=0;it?-.5*n*Math.pow(2,10*(t-=1))*Math.sin((t-i)*2*Math.PI/e):.5*n*Math.pow(2,-10*(t-=1))*Math.sin((t-i)*2*Math.PI/e)+1)},inBack:function(t,i){return void 0===i&&(i=1.70158),t*t*((i+1)*t-i)},outBack:function(t,i){return void 0===i&&(i=1.70158),--t*t*((i+1)*t+i)+1},inOutBack:function(t,i){return void 0===i&&(i=1.70158),(t/=.5)<1?.5*t*t*(((i*=1.525)+1)*t-i):.5*((t-=2)*t*(((i*=1.525)+1)*t+i)+2)},inBounce:function(t){return 1-e.outBounce(1-t)},outBounce:function(t){return 1/2.75>t?7.5625*t*t:2/2.75>t?7.5625*(t-=1.5/2.75)*t+.75:2.5/2.75>t?7.5625*(t-=2.25/2.75)*t+.9375:7.5625*(t-=2.625/2.75)*t+.984375},inOutBounce:function(t){return.5>t?.5*e.inBounce(2*t):.5*e.outBounce(2*t-1)+.5}};i.exports=e},{}],85:[function(t,i){function e(t){this.method=t,this._instances=[],this.state=[]}var n=t("../utilities/Utility");e.SUPPORTS_MULTIPLE=!0,e.prototype.get=function(){for(var t=0;t=t?2*t*t:-2*t*t+4*t-1},easeOutBounce:function(t){return t*(3-2*t)},spring:function(t){return(1-t)*Math.sin(6*Math.PI*t)+t}},e.SUPPORTS_MULTIPLE=!0,e.DEFAULT_OPTIONS={curve:e.Curves.linear,duration:500,speed:0};var h={};e.registerCurve=function(t,i){return h[t]?!1:(h[t]=i,!0)},e.unregisterCurve=function(t){return h[t]?(delete h[t],!0):!1},e.getCurve=function(t){var i=h[t];if(void 0!==i)return i;throw new Error("curve not registered")},e.getCurves=function(){return h},e.prototype.setOptions=function(t){void 0!==t.curve&&(this.options.curve=t.curve),void 0!==t.duration&&(this.options.duration=t.duration),void 0!==t.speed&&(this.options.speed=t.speed)},e.prototype.set=function(t,i,e){if(!i)return this.reset(t),e&&e(),void 0;if(this._startValue=o(this.get()),i=s(i,this.options),i.speed){var n=this._startValue;if(n instanceof Object){var r=0;for(var a in n)r+=(t[a]-n[a])*(t[a]-n[a]);i.duration=Math.sqrt(r)/i.speed}else i.duration=Math.abs(t-n)/i.speed}this._startTime=Date.now(),this._endValue=o(t),this._startVelocity=o(i.velocity),this._duration=i.duration,this._curve=i.curve,this._active=!0,this._callback=e},e.prototype.reset=function(t,i){if(this._callback){var e=this._callback;this._callback=void 0,e()}this.state=o(t),this.velocity=o(i),this._startTime=0,this._duration=0,this._updateTime=0,this._startValue=this.state,this._startVelocity=this.velocity,this._endValue=this.state,this._active=!1},e.prototype.getVelocity=function(){return this.velocity},e.prototype.get=function(t){return this.update(t),this.state},e.prototype.update=function(t){if(this._active){if(t||(t=Date.now()),!(this._updateTime>=t)){this._updateTime=t;var i=t-this._startTime;if(i>=this._duration)this.state=this._endValue,this.velocity=r(this.state,this._startValue,this._curve,this._duration,1),this._active=!1;else if(0>i)this.state=this._startValue,this.velocity=this._startVelocity;else{var e=i/this._duration;this.state=a(this._startValue,this._endValue,this._curve(e)),this.velocity=r(this.state,this._startValue,this._curve,this._duration,e)}}}else if(this._callback){var n=this._callback;this._callback=void 0,n()}},e.prototype.isActive=function(){return this._active},e.prototype.halt=function(){this.reset(this.get())},e.registerCurve("linear",e.Curves.linear),e.registerCurve("easeIn",e.Curves.easeIn),e.registerCurve("easeOut",e.Curves.easeOut),e.registerCurve("easeInOut",e.Curves.easeInOut),e.registerCurve("easeOutBounce",e.Curves.easeOutBounce),e.registerCurve("spring",e.Curves.spring),e.customCurve=function(t,i){return t=t||0,i=i||0,function(e){return t*e+(-2*t-i+3)*e*e+(t+i-2)*e*e*e}},i.exports=e},{}],91:[function(t,i){function e(t){t=t||0,this.endState=new m(t),this.initState=new m,this.spring=new y({anchor:this.endState}),this.wall=new v,this._restTolerance=1e-10,this._dimensions=1,this._absRestTolerance=this._restTolerance,this._callback=void 0,this.PE=new _,this.particle=new g,this.PE.addBody(this.particle),this.PE.attach([this.wall,this.spring],this.particle)}function n(){return this.particle.getEnergy()+this.spring.getEnergy([this.particle])}function o(){var t=this.endState.sub(this.initState).normSquared();this._absRestTolerance=0===t?this._restTolerance:this._restTolerance*t}function s(){this.PE.wake()}function r(){this.PE.sleep()}function a(t){this.endState.set(t);var i=this.endState.sub(this.initState).norm();this.wall.setOptions({distance:this.endState.norm(),normal:0===i?this.particle.velocity.normalize(-1):this.endState.sub(this.initState).normalize(-1)}),o.call(this)}function h(t){this.particle.position.set(t)}function c(t){this.particle.velocity.set(t)}function p(){return 0===this._dimensions?this.particle.getPosition1D():this.particle.getPosition()}function u(){return 0===this._dimensions?this.particle.getVelocity1D():this.particle.getVelocity()}function l(t){this._callback=t}function f(){if(this.PE.isSleeping()){if(this._callback){var t=this._callback;this._callback=void 0,t()}}else{var i=n.call(this);i=i&&(t.apply(this,arguments),c.removeListener(p,o))};return e(o)}function o(t,i){var n=u(),o=function(){var e=u();e-n>=i&&(t.apply(this,arguments),n=u())};return e(o)}function s(t,i){if(void 0===i)return void 0;var n=function(){i--,0>=i&&(t.apply(this,arguments),a(n))};return e(n)}function r(t,i){i=i||1;var n=i,o=function(){i--,0>=i&&(t.apply(this,arguments),i=n)};return e(o)}function a(t){c.removeListener(p,t)}function h(t,i){var e,o,s,r,h;return function(){o=this,h=arguments,s=u();var c=function(){var a=u-s;i>a?e=n(c,i-a):(e=null,r=t.apply(o,h))};return a(e),e=n(c,i),r}}var c=t("../core/Engine"),p="prerender",u=window.performance&&window.performance.now?function(){return window.performance.now()}:function(){return Date.now()};i.exports={setTimeout:n,setInterval:o,debounce:h,after:s,every:r,clear:a}},{"../core/Engine":4}],95:[function(t,i){var e={};e.Direction={X:0,Y:1,Z:2},e.after=function(t,i){var e=t;return function(){e--,0===e&&i.apply(this,arguments)}},e.loadURL=function(t,i){var e=new XMLHttpRequest;e.onreadystatechange=function(){4===this.readyState&&i&&i(this.responseText)},e.open("GET",t),e.send()},e.createDocumentFragmentFromHTML=function(t){var i=document.createElement("div");i.innerHTML=t;for(var e=document.createDocumentFragment();i.hasChildNodes();)e.appendChild(i.firstChild);return e},e.clone=function(t){var i;if("object"==typeof t){i=t instanceof Array?[]:{};for(var n in t)if("object"==typeof t[n]&&null!==t[n])if(t[n]instanceof Array){i[n]=new Array(t[n].length);for(var o=0;oe?n=e:i>n&&(n=i),this.setPosition(n)}function a(t){var i=t.velocity,e=this._orientation*this.getPosition(),n=this.options,o=this._orientation*this._cachedLength,s=n.positionThreshold||o/2,r=n.velocityThreshold;if(n.transition instanceof Object&&(n.transition.velocity=t.velocity),0===e)return this._isOpen=!1,void 0;if(e===o)return this._isOpen=!0,void 0;var a=Math.abs(i)>r||!this._isOpen&&e>s||this._isOpen&&s>e;a?this.toggle():this.reset()}var h=t("../core/RenderNode"),c=t("../core/Transform"),p=t("../core/OptionsManager"),u=t("../transitions/Transitionable"),l=t("../core/EventHandler"),f=0,d=1;e.SIDES={LEFT:0,TOP:1,RIGHT:2,BOTTOM:3},e.DEFAULT_OPTIONS={side:e.SIDES.LEFT,drawerLength:0,velocityThreshold:0,positionThreshold:0,transition:!0},e.prototype.setOptions=function(t){this._optionsManager.setOptions(t),void 0!==t.side&&(this._direction=n(t.side),this._orientation=o(t.side))},e.prototype.open=function(t,i){t instanceof Function&&(i=t),void 0===t&&(t=this.options.transition),this._cachedLength=s.call(this,this.drawer),this.setPosition(this._cachedLength,t,i),this._isOpen||(this._isOpen=!0,this._eventOutput.emit("open"))},e.prototype.close=function(t,i){t instanceof Function&&(i=t),void 0===t&&(t=this.options.transition),this.setPosition(0,t,i),this._isOpen&&(this._isOpen=!1,this._eventOutput.emit("close"))},e.prototype.setPosition=function(t,i,e){this._position.isActive()&&this._position.halt(),this._position.set(t,i,e)},e.prototype.getPosition=function(){return this._position.get()},e.prototype.setProgress=function(t,i,e){return this._position.set(t*this._cachedLength,i,e)},e.prototype.getProgress=function(){return this._position.get()/this._cachedLength},e.prototype.toggle=function(t){this._isOpen?this.close(t):this.open(t)},e.prototype.reset=function(t){this._isOpen?this.open(t):this.close(t)},e.prototype.isOpen=function(){return this._isOpen},e.prototype.render=function(){var t=this.getPosition();(!this._isOpen&&0>t&&1===this._orientation||t>0&&-1===this._orientation)&&(t=0,this.setPosition(t));var i=this._direction===f?c.translate(t,0,0):c.translate(0,t,0);return[{transform:c.behind,target:this.drawer.render()},{transform:i,target:this.content.render()}]},i.exports=e},{"../core/EventHandler":7,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15,"../transitions/Transitionable":88}],100:[function(t,i){function e(t){this._currentTarget=null,this._size=[void 0,void 0],this._controller=new h(t),this._controller.inTransformFrom(o.create(n.bind(this,1e-4))),this._controller.outTransformFrom(o.create(n.bind(this,-1e-4))),this._eventInput=new r,r.setInputHandler(this,this._eventInput),this._entityId=s.register(this),t&&this.setOptions(t)}function n(t,i){return a.translate(this._size[0]*(1-i),0,t*(1-i))}var o=t("../transitions/CachedMap"),s=t("../core/Entity"),r=t("../core/EventHandler"),a=t("../core/Transform"),h=t("./RenderController");e.prototype.show=function(t){this._currentTarget&&this._eventInput.unpipe(this._currentTarget),this._currentTarget=t,this._currentTarget&&this._currentTarget.trigger&&this._eventInput.pipe(this._currentTarget),this._controller.show.apply(this._controller,arguments)},e.prototype.setOptions=function(t){this._controller.setOptions(t)},e.prototype.render=function(){return this._entityId},e.prototype.commit=function(t){return this._size[0]=t.size[0],this._size[1]=t.size[1],{transform:t.transform,opacity:t.opacity,origin:t.origin,size:t.size,target:this._controller.render()}},i.exports=e},{"../core/Entity":5,"../core/EventHandler":7,"../core/Transform":15,"../transitions/CachedMap":83,"./RenderController":106}],101:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this.optionsManager=new a(this.options),t&&this.setOptions(t),this.id=s.register(this),this._ratios=new c(this.options.ratios),this._nodes=[],this._size=[0,0],this._cachedDirection=null,this._cachedLengths=[],this._cachedTransforms=null,this._ratiosDirty=!1,this._eventOutput=new h,h.setOutputHandler(this,this._eventOutput)}function n(t,i,n){var o,s,a,h,c=0,p=i,u=0;for(this._cachedLengths=[],this._cachedTransforms=[],h=0;hu;u++){r=0;for(var l=0;i>l;l++)void 0===this._modifiers[p]?o.call(this,p,[h,a],[r,c,0],1):s.call(this,p,[h,a],[r,c,0],1),p++,r+=h+this.options.gutterSize[0];c+=a+this.options.gutterSize[1]}for(this._dimensionsCache=[this.options.dimensions[0],this.options.dimensions[1]],this._contextSizeCache=[t[0],t[1]],this._activeCount=e*i,u=this._activeCount;u=this._activeCount&&this._states[u].opacity.isActive()&&(this._modifiers.splice(u,1),this._states.splice(u,1)),l&&p.push(f.modify({origin:o,target:l.render()})),c=c.getNext(),u++}return s&&(i=a.moveThen([-s[0]*o[0],-s[1]*o[1],0],i)),{transform:i,opacity:e,size:s,target:p}},i.exports=e},{"../core/Entity":5,"../core/EventHandler":7,"../core/Modifier":9,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15,"../core/ViewSequence":17,"../transitions/Transitionable":88,"../transitions/TransitionableTransform":89}],104:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new c(this.options),t&&this.setOptions(t),this._entityId=r.register(this),this.header=new a,this.footer=new a,this.content=new a}function n(t,i){var e=t.getSize();return e?e[this.options.direction]:i}function o(t){return this.options.direction===e.DIRECTION_X?h.translate(t,0,0):h.translate(0,t,0)}function s(t,i){return this.options.direction===e.DIRECTION_X?[t,i[1]]:[i[0],t]}var r=t("../core/Entity"),a=t("../core/RenderNode"),h=t("../core/Transform"),c=t("../core/OptionsManager");e.DIRECTION_X=0,e.DIRECTION_Y=1,e.DEFAULT_OPTIONS={direction:e.DIRECTION_Y,headerSize:void 0,footerSize:void 0,defaultHeaderSize:0,defaultFooterSize:0},e.prototype.render=function(){return this._entityId},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},e.prototype.commit=function(t){var i=t.transform,e=t.origin,r=t.size,a=t.opacity,c=void 0!==this.options.headerSize?this.options.headerSize:n.call(this,this.header,this.options.defaultHeaderSize),p=void 0!==this.options.footerSize?this.options.footerSize:n.call(this,this.footer,this.options.defaultFooterSize),u=r[this.options.direction]-c-p;r&&(i=h.moveThen([-r[0]*e[0],-r[1]*e[1],0],i));var l=[{size:s.call(this,c,r),target:this.header.render()},{transform:o.call(this,c),size:s.call(this,u,r),target:this.content.render()},{transform:o.call(this,c+u),size:s.call(this,p,r),target:this.footer.render()}];return{transform:i,opacity:a,size:r,target:l}},i.exports=e},{"../core/Entity":5,"../core/OptionsManager":10,"../core/RenderNode":11,"../core/Transform":15}],105:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new a(this.options),t&&this.setOptions(t),this._showing=!1,this.nodes=[],this.transforms=[],this.states=[]}var n=t("../core/Transform"),o=t("../core/Modifier"),s=t("../core/RenderNode"),r=t("../utilities/Utility"),a=t("../core/OptionsManager"),h=t("../transitions/Transitionable"),c=t("../transitions/TransitionableTransform");e.DEFAULT_OPTIONS={inTransform:n.scale(.001,.001,.001),inOpacity:0,inOrigin:[.5,.5],inAlign:[.5,.5],outTransform:n.scale(.001,.001,.001),outOpacity:0,outOrigin:[.5,.5],outAlign:[.5,.5],showTransform:n.identity,showOpacity:1,showOrigin:[.5,.5],showAlign:[.5,.5],inTransition:!0,outTransition:!0,overlap:!1},e.prototype.setOptions=function(t){return this._optionsManager.setOptions(t)},e.prototype.show=function(t,i,e){if(!t)return this.hide(e);if(i instanceof Function&&(e=i,i=void 0),this._showing){if(!this.options.overlap)return this.hide(this.show.bind(this,t,i,e));this.hide()}this._showing=!0;var n={transform:new c(this.options.inTransform),origin:new h(this.options.inOrigin),align:new h(this.options.inAlign),opacity:new h(this.options.inOpacity)},a=new o({transform:n.transform,opacity:n.opacity,origin:n.origin,align:n.align}),p=new s;p.add(a).add(t),this.nodes.push(p),this.states.push(n),this.transforms.push(a);var u=e?r.after(3,e):void 0;i||(i=this.options.inTransition),n.transform.set(this.options.showTransform,i,u),n.opacity.set(this.options.showOpacity,i,u),n.origin.set(this.options.showOrigin,i,u),n.align.set(this.options.showAlign,i,u) +},e.prototype.hide=function(t,i){if(this._showing){this._showing=!1,t instanceof Function&&(i=t,t=void 0);var e=this.nodes[this.nodes.length-1],n=this.transforms[this.transforms.length-1],o=this.states[this.states.length-1],s=r.after(3,function(){this.nodes.splice(this.nodes.indexOf(e),1),this.states.splice(this.states.indexOf(o),1),this.transforms.splice(this.transforms.indexOf(n),1),i&&i.call(this)}.bind(this));t||(t=this.options.outTransition),o.transform.set(this.options.outTransform,t,s),o.opacity.set(this.options.outOpacity,t,s),o.origin.set(this.options.outOrigin,t,s),o.align.set(this.options.outAlign,t,s)}},e.prototype.render=function(){for(var t=[],i=0;i=0){if(!this.options.overlap)return this._nextRenderable?this._nextRenderable=t:(this._nextRenderable=t,this.hide(function(){this._nextRenderable===t&&this.show(this._nextRenderable,e),this._nextRenderable=null})),void 0;this.hide()}var r=null,h=this._renderables.indexOf(t);if(h>=0){this._showing=h,r=this._states[h],r.halt();var c=this._outgoingRenderables.indexOf(t);c>=0&&this._outgoingRenderables.splice(c,1)}else{r=new a(0);var p=new o({transform:this.inTransformMap?n.bind(this,this.inTransformMap,r):null,opacity:this.inOpacityMap?n.bind(this,this.inOpacityMap,r):null,origin:this.inOriginMap?n.bind(this,this.inOriginMap,r):null,align:this.inAlignMap?n.bind(this,this.inAlignMap,r):null}),u=new s;u.add(p).add(t),this._showing=this._nodes.length,this._nodes.push(u),this._modifiers.push(p),this._states.push(r),this._renderables.push(t)}i||(i=this.options.inTransition),r.set(1,i,e)},e.prototype.hide=function(t,i){if(!(this._showing<0)){var e=this._showing;this._showing=-1,t instanceof Function&&(i=t,t=void 0);var o=this._nodes[e],s=this._modifiers[e],r=this._states[e],a=this._renderables[e];s.transformFrom(this.outTransformMap?n.bind(this,this.outTransformMap,r):null),s.opacityFrom(this.outOpacityMap?n.bind(this,this.outOpacityMap,r):null),s.originFrom(this.outOriginMap?n.bind(this,this.outOriginMap,r):null),s.alignFrom(this.outAlignMap?n.bind(this,this.outAlignMap,r):null),this._outgoingRenderables.indexOf(a)<0&&this._outgoingRenderables.push(a),t||(t=this.options.outTransition),r.halt(),r.set(0,t,function(t,e,n,o){if(this._outgoingRenderables.indexOf(o)>=0){var s=this._nodes.indexOf(t);this._nodes.splice(s,1),this._modifiers.splice(s,1),this._states.splice(s,1),this._renderables.splice(s,1),this._outgoingRenderables.splice(this._outgoingRenderables.indexOf(o),1),this._showing>=s&&this._showing--}i&&i.call(this)}.bind(this,o,s,r,a))}},e.prototype.render=function(){var t=this._output;t.length>this._nodes.length&&t.splice(this._nodes.length);for(var i=0;ithis.getCumulativeSize()[this.options.direction]?n.call(this,this.getCumulativeSize()):n.call(this,this._contextSize)}function r(){for(var t=null,i=this._position,e=[],r=-this._positionOffset,a=s.call(this),h=this._node;h&&r-ir){for(;c&&a>p;)c=c.getPrevious(),c&&(p+=n.call(this,c.getSize()));for(c=this._node;c&&a>p;)c=c.getNext(),c&&(p+=n.call(this,c.getSize()))}for(!h&&a-d>r-i?1!==this._onEdge&&(this._onEdge=1,this._eventOutput.emit("onEdge",{position:r-a})):!this._node.getPrevious()&&-d>i?-1!==this._onEdge&&(this._onEdge=-1,this._eventOutput.emit("onEdge",{position:0})):0!==this._onEdge&&(this._onEdge=0,this._eventOutput.emit("offEdge")),h=this._node&&this._node.getPrevious?this._node.getPrevious():null,r=-this._positionOffset,h&&(t=h.getSize?h.getSize():this._contextSize,r-=n.call(this,t));h&&r-i>-(a+this.options.margin);)o.call(this,h,r,e),h=h.getPrevious?h.getPrevious():null,h&&(t=h.getSize?h.getSize():this._contextSize,r-=n.call(this,t));return e}var a=t("../core/Entity"),h=t("../core/Group"),c=t("../core/OptionsManager"),p=t("../core/Transform"),u=t("../utilities/Utility"),l=t("../core/ViewSequence"),f=t("../core/EventHandler");e.DEFAULT_OPTIONS={direction:u.Direction.Y,margin:0,clipSize:void 0,groupScroll:!1};var d=0;e.prototype.getCumulativeSize=function(t){return void 0===t&&(t=this._node._.cumulativeSizes.length-1),this._node._.getSize(t)},e.prototype.setOptions=function(t){t.groupScroll!==this.options.groupScroll&&(t.groupScroll?this.group.pipe(this._eventOutput):this.group.unpipe(this._eventOutput)),this._optionsManager.setOptions(t)},e.prototype.onEdge=function(){return this._onEdge},e.prototype.outputFrom=function(t,i){t||(t=function(t){return this.options.direction===u.Direction.X?p.translate(t,0):p.translate(0,t)}.bind(this),i||(i=t)),this._outputFunction=t,this._masterOutputFunction=i?i:function(i){return p.inverse(t(-i))}},e.prototype.positionFrom=function(t){t instanceof Function?this._positionGetter=t:t&&t.get?this._positionGetter=t.get.bind(t):(this._positionGetter=null,this._position=t),this._positionGetter&&(this._position=this._positionGetter.call(this))},e.prototype.sequenceFrom=function(t){t instanceof Array&&(t=new l({array:t})),this._node=t,this._positionOffset=0},e.prototype.getSize=function(t){return t?this._contextSize:this._size},e.prototype.render=function(){return this._node?(this._positionGetter&&(this._position=this._positionGetter.call(this)),this._entityId):null},e.prototype.commit=function(t){var i=t.transform,e=t.opacity,n=t.origin,o=t.size;this.options.clipSize||o[0]===this._contextSize[0]&&o[1]===this._contextSize[1]||(this._onEdge=0,this._contextSize[0]=o[0],this._contextSize[1]=o[1],this.options.direction===u.Direction.X?(this._size[0]=s.call(this),this._size[1]=void 0):(this._size[0]=void 0,this._size[1]=s.call(this)));var r=this._masterOutputFunction(-this._position);return{transform:p.multiply(i,r),size:o,opacity:e,origin:n,target:this.group.render()}},i.exports=e},{"../core/Entity":5,"../core/EventHandler":7,"../core/Group":8,"../core/OptionsManager":10,"../core/Transform":15,"../core/ViewSequence":17,"../utilities/Utility":95}],109:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this._optionsManager=new O(this.options),this._scroller=new T(this.options),this.sync=new b(["scroll","touch"],{direction:this.options.direction,scale:this.options.syncScale,rails:this.options.rails,preventDefault:void 0!==this.options.preventDefault?this.options.preventDefault:this.options.direction!==w.Direction.Y}),this._physicsEngine=new _,this._particle=new g,this._physicsEngine.addBody(this._particle),this.spring=new v({anchor:[0,0,0],period:this.options.edgePeriod,dampingRatio:this.options.edgeDamp}),this.drag=new y({forceFunction:y.FORCE_FUNCTIONS.QUADRATIC,strength:this.options.drag}),this.friction=new y({forceFunction:y.FORCE_FUNCTIONS.LINEAR,strength:this.options.friction}),this._node=null,this._touchCount=0,this._springState=I.NONE,this._onEdge=M.NONE,this._pageSpringPosition=0,this._edgeSpringPosition=0,this._touchVelocity=0,this._earlyEnd=!1,this._needsPaginationCheck=!1,this._displacement=0,this._totalShift=0,this._cachedIndex=0,this._scroller.positionFrom(this.getPosition.bind(this)),this._eventInput=new m,this._eventOutput=new m,this._eventInput.pipe(this.sync),this.sync.pipe(this._eventInput),m.setInputHandler(this,this._eventInput),m.setOutputHandler(this,this._eventOutput),r.call(this),t&&this.setOptions(t)}function n(t){this._touchCount=t.count,void 0===t.count&&(this._touchCount=1),h.call(this),this.setVelocity(0),this._touchVelocity=0,this._earlyEnd=!1}function o(t){var i=-t.velocity,e=-t.delta;if(this._onEdge!==M.NONE&&t.slip&&(0>i&&this._onEdge===M.TOP||i>0&&this._onEdge===M.BOTTOM?this._earlyEnd||(s.call(this,t),this._earlyEnd=!0):this._earlyEnd&&Math.abs(i)>Math.abs(this.getVelocity())&&n.call(this,t)),!this._earlyEnd){if(this._touchVelocity=i,t.slip){var o=this.options.speedLimit;-o>i?i=-o:i>o&&(i=o),this.setVelocity(i);var r=16*o;e>r?e=r:-r>e&&(e=-r)}this.setPosition(this.getPosition()+e),this._displacement+=e,this._springState===I.NONE&&f.call(this)}}function s(t){if(this._touchCount=t.count||0,!this._touchCount){h.call(this),this._onEdge!==M.NONE&&l.call(this,this._edgeSpringPosition,I.EDGE),a.call(this);var i=-t.velocity,e=this.options.speedLimit;t.slip&&(e*=this.options.edgeGrip),-e>i?i=-e:i>e&&(i=e),this.setVelocity(i),this._touchVelocity=0,this._needsPaginationCheck=!0}}function r(){this._eventInput.bindThis(this),this._eventInput.on("start",n),this._eventInput.on("update",o),this._eventInput.on("end",s),this._eventInput.on("resize",function(){this._node._.calculateSize()}.bind(this)),this._scroller.on("onEdge",function(t){this._edgeSpringPosition=t.position,p.call(this,this._scroller.onEdge()),this._eventOutput.emit("onEdge")}.bind(this)),this._scroller.on("offEdge",function(){this.sync.setOptions({scale:this.options.syncScale}),this._onEdge=this._scroller.onEdge(),this._eventOutput.emit("offEdge")}.bind(this)),this._particle.on("update",function(t){this._springState===I.NONE&&f.call(this),this._displacement=t.position.x-this._totalShift}.bind(this)),this._particle.on("end",function(){(!this.options.paginated||this.options.paginated&&this._springState!==I.NONE)&&this._eventOutput.emit("settle")}.bind(this))}function a(){this._springState?this._physicsEngine.attach([this.spring],this._particle):this._physicsEngine.attach([this.drag,this.friction],this._particle)}function h(){this._springState=I.NONE,this._physicsEngine.detachAll()}function c(t){var i=this.options.direction,e=t.getSize();return e?e[i]:this._scroller.getSize()[i]}function p(t){this.sync.setOptions({scale:this.options.edgeGrip}),this._onEdge=t,this._touchCount||this._springState===I.EDGE||l.call(this,this._edgeSpringPosition,I.EDGE),this._springState&&Math.abs(this.getVelocity())<.001&&(h.call(this),a.call(this))}function u(){if(!this._touchCount&&this._springState!==I.EDGE){var t=this.getVelocity();if(!(Math.abs(t)>=this.options.pageStopSpeed)){var i=this.getPosition(),e=Math.abs(t)>this.options.pageSwitchSpeed,n=c.call(this,this._node),o=i>.5*n,s=t>0,r=0>t;this._needsPaginationCheck=!1,o&&!e||e&&s?this.goToNextPage():e&&r?this.goToPreviousPage():l.call(this,0,I.PAGE)}}}function l(t,i){var e;i===I.EDGE?(this._edgeSpringPosition=t,e={anchor:[this._edgeSpringPosition,0,0],period:this.options.edgePeriod,dampingRatio:this.options.edgeDamp}):i===I.PAGE&&(this._pageSpringPosition=t,e={anchor:[this._pageSpringPosition,0,0],period:this.options.pagePeriod,dampingRatio:this.options.pageDamp}),this.spring.setOptions(e),i&&!this._springState&&(h.call(this),this._springState=i,a.call(this)),this._springState=i}function f(){var t=0,i=this.getPosition();i+=(0>i?-.5:.5)>>0;for(var e=c.call(this,this._node),n=this._node.getNext();t+i>=e&&n;)t-=e,this._scroller.sequenceFrom(n),this._node=n,n=this._node.getNext(),e=c.call(this,this._node);for(var o,s=this._node.getPrevious();0>=t+i&&s;)o=c.call(this,s),this._scroller.sequenceFrom(s),this._node=s,t+=o,s=this._node.getPrevious();t&&d.call(this,t),this._node&&(this._node.index!==this._cachedIndex?this.getPosition()<.5*e&&(this._cachedIndex=this._node.index,this._eventOutput.emit("pageChange",{direction:-1,index:this._cachedIndex})):this.getPosition()>.5*e&&(this._cachedIndex=this._node.index+1,this._eventOutput.emit("pageChange",{direction:1,index:this._cachedIndex})))}function d(t){this._edgeSpringPosition+=t,this._pageSpringPosition+=t,this.setPosition(this.getPosition()+t),this._totalShift+=t,this._springState===I.EDGE?this.spring.setOptions({anchor:[this._edgeSpringPosition,0,0]}):this._springState===I.PAGE&&this.spring.setOptions({anchor:[this._pageSpringPosition,0,0]})}var _=t("../physics/PhysicsEngine"),g=t("../physics/bodies/Particle"),y=t("../physics/forces/Drag"),v=t("../physics/forces/Spring"),m=t("../core/EventHandler"),O=t("../core/OptionsManager"),S=t("../core/ViewSequence"),T=t("../views/Scroller"),w=t("../utilities/Utility"),b=t("../inputs/GenericSync"),E=t("../inputs/ScrollSync"),z=t("../inputs/TouchSync");b.register({scroll:E,touch:z});var I={NONE:0,EDGE:1,PAGE:2},M={TOP:-1,NONE:0,BOTTOM:1};e.DEFAULT_OPTIONS={direction:w.Direction.Y,rails:!0,friction:.005,drag:1e-4,edgeGrip:.2,edgePeriod:300,edgeDamp:1,margin:1e3,paginated:!1,pagePeriod:500,pageDamp:.8,pageStopSpeed:10,pageSwitchSpeed:.5,speedLimit:5,groupScroll:!1,syncScale:1},e.prototype.getCurrentIndex=function(){return this._node.index},e.prototype.goToPreviousPage=function(){if(!this._node||this._onEdge===M.TOP)return null;if(this.getPosition()>1&&this._springState===I.NONE)return l.call(this,0,I.PAGE),this._node;var t=this._node.getPrevious();if(t){var i=c.call(this,t);this._scroller.sequenceFrom(t),this._node=t,d.call(this,i),l.call(this,0,I.PAGE)}return t},e.prototype.goToNextPage=function(){if(!this._node||this._onEdge===M.BOTTOM)return null;var t=this._node.getNext();if(t){var i=c.call(this,this._node);this._scroller.sequenceFrom(t),this._node=t,d.call(this,-i),l.call(this,0,I.PAGE)}return t},e.prototype.goToPage=function(t){var i,e=this.getCurrentIndex();if(e>t)for(i=0;e-t>i;i++)this.goToPreviousPage();if(t>e)for(i=0;t-e>i;i++)this.goToNextPage()},e.prototype.outputFrom=function(){return this._scroller.outputFrom.apply(this._scroller,arguments)},e.prototype.getPosition=function(){return this._particle.getPosition1D()},e.prototype.getAbsolutePosition=function(){return this._scroller.getCumulativeSize(this.getCurrentIndex())[this.options.direction]+this.getPosition()},e.prototype.getOffset=e.prototype.getPosition,e.prototype.setPosition=function(t){this._particle.setPosition1D(t)},e.prototype.setOffset=e.prototype.setPosition,e.prototype.getVelocity=function(){return this._touchCount?this._touchVelocity:this._particle.getVelocity1D()},e.prototype.setVelocity=function(t){this._particle.setVelocity1D(t)},e.prototype.setOptions=function(t){void 0!==t.direction&&("x"===t.direction?t.direction=w.Direction.X:"y"===t.direction&&(t.direction=w.Direction.Y)),t.groupScroll!==this.options.groupScroll&&(t.groupScroll?this.subscribe(this._scroller):this.unsubscribe(this._scroller)),this._optionsManager.setOptions(t),this._scroller.setOptions(t),void 0!==t.drag&&this.drag.setOptions({strength:this.options.drag}),void 0!==t.friction&&this.friction.setOptions({strength:this.options.friction}),(void 0!==t.edgePeriod||void 0!==t.edgeDamp)&&this.spring.setOptions({period:this.options.edgePeriod,dampingRatio:this.options.edgeDamp}),(t.rails||void 0!==t.direction||void 0!==t.syncScale||t.preventDefault)&&this.sync.setOptions({rails:this.options.rails,direction:this.options.direction===w.Direction.X?b.DIRECTION_X:b.DIRECTION_Y,scale:this.options.syncScale,preventDefault:this.options.preventDefault})},e.prototype.sequenceFrom=function(t){return t instanceof Array&&(t=new S({array:t,trackSize:!0})),this._node=t,this._scroller.sequenceFrom(t)},e.prototype.getSize=function(){return this._scroller.getSize.apply(this._scroller,arguments)},e.prototype.render=function(){return this.options.paginated&&this._needsPaginationCheck&&u.call(this),this._scroller.render()},i.exports=e},{"../core/EventHandler":7,"../core/OptionsManager":10,"../core/ViewSequence":17,"../inputs/GenericSync":27,"../inputs/ScrollSync":32,"../inputs/TouchSync":33,"../physics/PhysicsEngine":48,"../physics/bodies/Particle":51,"../physics/forces/Drag":63,"../physics/forces/Spring":68,"../utilities/Utility":95,"../views/Scroller":108}],110:[function(t,i){function e(t){this._items=null,this._size=null,this._outputFunction=e.DEFAULT_OUTPUT_FUNCTION,this.options=a.clone(this.constructor.DEFAULT_OPTIONS||e.DEFAULT_OPTIONS),this.optionsManager=new n(this.options),this.id=o.register(this),this.cachedSize=[void 0,void 0],t&&this.setOptions(t)}var n=t("../core/OptionsManager"),o=t("../core/Entity"),s=t("../core/Transform"),r=t("../core/ViewSequence"),a=t("../utilities/Utility");e.DEFAULT_OPTIONS={direction:a.Direction.Y,itemSpacing:0},e.DEFAULT_OUTPUT_FUNCTION=function(t,i){var e=this.options.direction===a.Direction.X?s.translate(i,0):s.translate(0,i);return{size:this.cachedSize,transform:e,target:t.render()}},e.prototype.getSize=function(){return this._size||this.render(),this._size},e.prototype.sequenceFrom=function(t){return t instanceof Array&&(t=new r(t)),this._items=t,this},e.prototype.setOptions=function(){return this.optionsManager.setOptions.apply(this.optionsManager,arguments),this},e.prototype.setOutputFunction=function(t){return this._outputFunction=t,this},e.prototype.render=function(){return this.id},e.prototype.commit=function(t){var i=0,e=1^this.options.direction,n=this._items,o=null,s=[],r={},a=[],h=0;for(this._size=[0,0],this.cachedSize=t.size;n&&(o=n.get());)o.getSize&&(s=o.getSize()),r=this._outputFunction.call(this,o,i,h++),a.push(r),s&&(s[this.options.direction]&&(i+=s[this.options.direction]),s[e]>this._size[e]&&(this._size[e]=s[e]),0===s[e]&&(this._size[e]=void 0)),n=n.getNext(),this.options.itemSpacing&&n&&(i+=this.options.itemSpacing);return this._size[this.options.direction]=i,{transform:t.transform,origin:t.origin,opacity:t.opacity,size:this.getSize(),target:a}},i.exports=e},{"../core/Entity":5,"../core/OptionsManager":10,"../core/Transform":15,"../core/ViewSequence":17,"../utilities/Utility":95}],111:[function(t,i){function e(){n.apply(this,arguments),this._id=o.register(this),this._parentSize=[]}var n=t("../core/View"),o=t("../core/Entity"),s=t("../core/Transform");e.prototype=Object.create(n.prototype),e.prototype.constructor=e,e.prototype.commit=function(t){var i=t.transform,e=t.opacity,n=t.origin;return this._parentSize&&this._parentSize[0]===t.size[0]&&this._parentSize[1]===t.size[1]||(this._parentSize[0]=t.size[0],this._parentSize[1]=t.size[1],this._eventInput.emit("parentResize",this._parentSize),this.onResize&&this.onResize(this._parentSize)),this._parentSize&&(i=s.moveThen([-this._parentSize[0]*n[0],-this._parentSize[1]*n[1],0],i)),{transform:i,opacity:e,size:this._parentSize,target:this._node.render()}},e.prototype.getParentSize=function(){return this._parentSize},e.prototype.render=function(){return this._id},i.exports=e},{"../core/Entity":5,"../core/Transform":15,"../core/View":16}],112:[function(t,i){i.exports={ContextualView:t("./ContextualView"),Deck:t("./Deck"),DrawerLayout:t("./DrawerLayout"),EdgeSwapper:t("./EdgeSwapper"),FlexibleLayout:t("./FlexibleLayout"),Flipper:t("./Flipper"),GridLayout:t("./GridLayout"),HeaderFooterLayout:t("./HeaderFooterLayout"),Lightbox:t("./Lightbox"),RenderController:t("./RenderController"),ScrollContainer:t("./ScrollContainer"),Scroller:t("./Scroller"),Scrollview:t("./Scrollview"),SequentialLayout:t("./SequentialLayout"),SizeAwareView:t("./SizeAwareView")}},{"./ContextualView":97,"./Deck":98,"./DrawerLayout":99,"./EdgeSwapper":100,"./FlexibleLayout":101,"./Flipper":102,"./GridLayout":103,"./HeaderFooterLayout":104,"./Lightbox":105,"./RenderController":106,"./ScrollContainer":107,"./Scroller":108,"./Scrollview":109,"./SequentialLayout":110,"./SizeAwareView":111}],113:[function(t,i){function e(){r.apply(this,arguments),this.title=new o({classes:this.options.classes,content:this.options.content}),this.back=new o({size:[this.options.size[1],this.options.size[1]],classes:this.options.classes,content:this.options.backContent}),this.back.on("click",function(){this._eventOutput.emit("back",{})}.bind(this)),this.more=new o({size:[this.options.size[1],this.options.size[1]],classes:this.options.classes,content:this.options.moreContent}),this.more.on("click",function(){this._eventOutput.emit("more",{})}.bind(this)),this.layout=new n({id:"master",size:this.options.size,target:[{transform:s.inFront,origin:[0,.5],align:[0,.5],target:this.back},{origin:[.5,.5],align:[.5,.5],target:this.title},{transform:s.inFront,origin:[1,.5],align:[1,.5],target:this.more}]}),this._add(this.layout),this._optionsManager.on("change",function(t){var i=t.id,e=t.value;"size"===i?(this.layout.id.master.setSize(e),this.title.setSize(e),this.back.setSize([e[1],e[1]]),this.more.setSize([e[1],e[1]])):"backClasses"===i?this.back.setOptions({classes:this.options.classes.concat(this.options.backClasses)}):"backContent"===i?this.back.setContent(this.options.backContent):"classes"===i?(this.title.setOptions({classes:this.options.classes}),this.back.setOptions({classes:this.options.classes.concat(this.options.backClasses)}),this.more.setOptions({classes:this.options.classes.concat(this.options.moreClasses)})):"content"===i?this.setContent(this.options.content):"moreClasses"===i?this.more.setOptions({classes:this.options.classes.concat(this.options.moreClasses)}):"moreContent"===i&&this.more.setContent(this.options.content)}.bind(this))}var n=t("../core/Scene"),o=t("../core/Surface"),s=t("../core/Transform"),r=t("../core/View");e.prototype=Object.create(r.prototype),e.prototype.constructor=e,e.DEFAULT_OPTIONS={size:[void 0,50],backClasses:["back"],backContent:"◀",classes:["navigation"],content:"",moreClasses:["more"],moreContent:"✚"},e.prototype.setContent=function(t){return this.title.setContent(t)},i.exports=e},{"../core/Scene":12,"../core/Surface":14,"../core/Transform":15,"../core/View":16}],114:[function(t,i){function e(t){this.options=Object.create(e.DEFAULT_OPTIONS),this.optionsManager=new c(this.options),t&&this.setOptions(t),this.indicator=new s({size:this.options.indicatorSize,classes:["slider-back"]}),this.label=new o({size:this.options.labelSize,content:this.options.label,properties:{pointerEvents:"none"},classes:["slider-label"]}),this.eventOutput=new a,this.eventInput=new a,a.setInputHandler(this,this.eventInput),a.setOutputHandler(this,this.eventOutput);var i=(this.options.range[1]-this.options.range[0])/this.options.indicatorSize[0];this.sync=new l(["mouse","touch"],{scale:i,direction:l.DIRECTION_X}),this.indicator.pipe(this.sync),this.sync.pipe(this),this.eventInput.on("update",function(t){this.set(t.position)}.bind(this)),this._drawPos=0,n.call(this)}function n(){this.label.setContent(this.options.label+''+this.get().toFixed(this.options.precision)+"")}var o=t("../core/Surface"),s=t("../surfaces/CanvasSurface"),r=t("../core/Transform"),a=t("../core/EventHandler"),h=t("../math/Utilities"),c=t("../core/OptionsManager"),p=t("../inputs/MouseSync"),u=t("../inputs/TouchSync"),l=t("../inputs/GenericSync");l.register({mouse:p,touch:u}),e.DEFAULT_OPTIONS={size:[200,60],indicatorSize:[200,30],labelSize:[200,30],range:[0,1],precision:2,value:0,label:"",fillColor:"rgba(170, 170, 170, 1)"},e.prototype.setOptions=function(t){return this.optionsManager.setOptions(t)},e.prototype.get=function(){return this.options.value},e.prototype.set=function(t){t!==this.options.value&&(this.options.value=h.clamp(t,this.options.range),n.call(this),this.eventOutput.emit("change",{value:t}))},e.prototype.getSize=function(){return this.options.size},e.prototype.render=function(){var t=this.options.range,i=Math.floor((this.get()-t[0])/(t[1]-t[0])*this.options.indicatorSize[0]);if(ithis._drawPos){var e=this.indicator.getContext("2d");e.fillStyle=this.options.fillColor,e.fillRect(this._drawPos-1,0,i-this._drawPos+1,this.options.indicatorSize[1])}return this._drawPos=i,{size:this.options.size,target:[{origin:[0,0],target:this.indicator.render()},{transform:r.translate(0,0,1),origin:[0,0],target:this.label.render()}]}},i.exports=e},{"../core/EventHandler":7,"../core/OptionsManager":10,"../core/Surface":14,"../core/Transform":15,"../inputs/GenericSync":27,"../inputs/MouseSync":28,"../inputs/TouchSync":33,"../math/Utilities":40,"../surfaces/CanvasSurface":74}],115:[function(t,i){function e(){r.apply(this,arguments),this.layout=new a,this.buttons=[],this._buttonIds={},this._buttonCallbacks={},this.layout.sequenceFrom(this.buttons),this._add(this.layout),this._optionsManager.on("change",n.bind(this))}function n(t){var i=t.id,e=t.value;if("direction"===i)this.layout.setOptions({dimensions:o.call(this.buttons.length,this.options.direction)});else if("buttons"===i)for(var n in this.buttons)this.buttons[n].setOptions(e);else if("sections"===i)for(var s in this.options.sections)this.defineSection(s,this.options.sections[s])}function o(t,i){return i===s.Direction.X?[t,1]:[1,t]}var s=t("../utilities/Utility"),r=t("../core/View"),a=t("../views/GridLayout"),h=t("./ToggleButton");e.prototype=Object.create(r.prototype),e.prototype.constructor=e,e.DEFAULT_OPTIONS={sections:[],widget:h,size:[void 0,50],direction:s.Direction.X,buttons:{toggleMode:h.ON}},e.prototype.defineSection=function(t,i){var e,n=this._buttonIds[t];if(void 0===n){n=this.buttons.length,this._buttonIds[t]=n;var s=this.options.widget;e=new s,this.buttons[n]=e,this.layout.setOptions({dimensions:o(this.buttons.length,this.options.direction)})}else e=this.buttons[n],e.unbind("select",this._buttonCallbacks[t]);this.options.buttons&&e.setOptions(this.options.buttons),e.setOptions(i),this._buttonCallbacks[t]=this.select.bind(this,t),e.on("select",this._buttonCallbacks[t])},e.prototype.select=function(t){var i=this._buttonIds[t];this.buttons[i]&&this.buttons[i].isSelected()?this._eventOutput.emit("select",{id:t}):this.buttons[i]&&this.buttons[i].select();for(var e=0;e= Infinity) { + return {translate: Transform.getTranslate(M), rotate: [0, 0, 0], scale: [0, 0, 0], skew: [0, 0, 0]}; + } + + //evaluate Q1 = I - 2vv'/v'v + var Q1 = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1]; + + //diagonals + Q1[0] = 1 - mult * v[0] * v[0]; // 0,0 entry + Q1[5] = 1 - mult * v[1] * v[1]; // 1,1 entry + Q1[10] = 1 - mult * v[2] * v[2]; // 2,2 entry + + //upper diagonal + Q1[1] = -mult * v[0] * v[1]; // 0,1 entry + Q1[2] = -mult * v[0] * v[2]; // 0,2 entry + Q1[6] = -mult * v[1] * v[2]; // 1,2 entry + + //lower diagonal + Q1[4] = Q1[1]; // 1,0 entry + Q1[8] = Q1[2]; // 2,0 entry + Q1[9] = Q1[6]; // 2,1 entry + + //reduce first column of M + var MQ1 = Transform.multiply(Q1, M); + + //SECOND ITERATION on (1,1) minor + var x2 = [MQ1[5], MQ1[6]]; + var sgn2 = _sign(x2[0]); // sign of first component of x (for stability) + var x2Norm = _norm(x2); // norm of first column vector + var v2 = [x2[0] + sgn2 * x2Norm, x2[1]]; // v = x + sign(x[0])|x|e1 + var mult2 = 2 / _normSquared(v2); // mult = 2/v'v + + //evaluate Q2 = I - 2vv'/v'v + var Q2 = [1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1]; + + //diagonal + Q2[5] = 1 - mult2 * v2[0] * v2[0]; // 1,1 entry + Q2[10] = 1 - mult2 * v2[1] * v2[1]; // 2,2 entry + + //off diagonals + Q2[6] = -mult2 * v2[0] * v2[1]; // 2,1 entry + Q2[9] = Q2[6]; // 1,2 entry + + //calc QR decomposition. Q = Q1*Q2, R = Q'*M + var Q = Transform.multiply(Q2, Q1); //note: really Q transpose + var R = Transform.multiply(Q, M); + + //remove negative scaling + var remover = Transform.scale(R[0] < 0 ? -1 : 1, R[5] < 0 ? -1 : 1, R[10] < 0 ? -1 : 1); + R = Transform.multiply(R, remover); + Q = Transform.multiply(remover, Q); + + //decompose into rotate/scale/skew matrices + var result = {}; + result.translate = Transform.getTranslate(M); + result.rotate = [Math.atan2(-Q[6], Q[10]), Math.asin(Q[2]), Math.atan2(-Q[1], Q[0])]; + if (!result.rotate[0]) { + result.rotate[0] = 0; + result.rotate[2] = Math.atan2(Q[4], Q[5]); + } + result.scale = [R[0], R[5], R[10]]; + result.skew = [Math.atan2(R[9], result.scale[2]), Math.atan2(R[8], result.scale[2]), Math.atan2(R[4], result.scale[0])]; + + //double rotation workaround + if (Math.abs(result.rotate[0]) + Math.abs(result.rotate[2]) > 1.5 * Math.PI) { + result.rotate[1] = Math.PI - result.rotate[1]; + if (result.rotate[1] > Math.PI) result.rotate[1] -= 2 * Math.PI; + if (result.rotate[1] < -Math.PI) result.rotate[1] += 2 * Math.PI; + if (result.rotate[0] < 0) result.rotate[0] += Math.PI; + else result.rotate[0] -= Math.PI; + if (result.rotate[2] < 0) result.rotate[2] += Math.PI; + else result.rotate[2] -= Math.PI; + } + + return result; + }; + + /** + * Weighted average between two matrices by averaging their + * translation, rotation, scale, skew components. + * f(M1,M2,t) = (1 - t) * M1 + t * M2 + * + * @method average + * @static + * @param {Transform} M1 f(M1,M2,0) = M1 + * @param {Transform} M2 f(M1,M2,1) = M2 + * @param {Number} t + * @return {Transform} + */ + Transform.average = function average(M1, M2, t) { + t = (t === undefined) ? 0.5 : t; + var specM1 = Transform.interpret(M1); + var specM2 = Transform.interpret(M2); + + var specAvg = { + translate: [0, 0, 0], + rotate: [0, 0, 0], + scale: [0, 0, 0], + skew: [0, 0, 0] + }; + + for (var i = 0; i < 3; i++) { + specAvg.translate[i] = (1 - t) * specM1.translate[i] + t * specM2.translate[i]; + specAvg.rotate[i] = (1 - t) * specM1.rotate[i] + t * specM2.rotate[i]; + specAvg.scale[i] = (1 - t) * specM1.scale[i] + t * specM2.scale[i]; + specAvg.skew[i] = (1 - t) * specM1.skew[i] + t * specM2.skew[i]; + } + return Transform.build(specAvg); + }; + + /** + * Compose .translate, .rotate, .scale, .skew components into + * Transform matrix + * + * @method build + * @static + * @param {matrixSpec} spec object with component matrices .translate, + * .rotate, .scale, .skew + * @return {Transform} composed transform + */ + Transform.build = function build(spec) { + var scaleMatrix = Transform.scale(spec.scale[0], spec.scale[1], spec.scale[2]); + var skewMatrix = Transform.skew(spec.skew[0], spec.skew[1], spec.skew[2]); + var rotateMatrix = Transform.rotate(spec.rotate[0], spec.rotate[1], spec.rotate[2]); + return Transform.thenMove(Transform.multiply(Transform.multiply(rotateMatrix, skewMatrix), scaleMatrix), spec.translate); + }; + + /** + * Determine if two Transforms are component-wise equal + * Warning: breaks on perspective Transforms + * + * @method equals + * @static + * @param {Transform} a matrix + * @param {Transform} b matrix + * @return {boolean} + */ + Transform.equals = function equals(a, b) { + return !Transform.notEquals(a, b); + }; + + /** + * Determine if two Transforms are component-wise unequal + * Warning: breaks on perspective Transforms + * + * @method notEquals + * @static + * @param {Transform} a matrix + * @param {Transform} b matrix + * @return {boolean} + */ + Transform.notEquals = function notEquals(a, b) { + if (a === b) return false; + + // shortci + return !(a && b) || + a[12] !== b[12] || a[13] !== b[13] || a[14] !== b[14] || + a[0] !== b[0] || a[1] !== b[1] || a[2] !== b[2] || + a[4] !== b[4] || a[5] !== b[5] || a[6] !== b[6] || + a[8] !== b[8] || a[9] !== b[9] || a[10] !== b[10]; + }; + + /** + * Constrain angle-trio components to range of [-pi, pi). + * + * @method normalizeRotation + * @static + * @param {Array.Number} rotation phi, theta, psi (array of floats + * && array.length == 3) + * @return {Array.Number} new phi, theta, psi triplet + * (array of floats && array.length == 3) + */ + Transform.normalizeRotation = function normalizeRotation(rotation) { + var result = rotation.slice(0); + if (result[0] === Math.PI * 0.5 || result[0] === -Math.PI * 0.5) { + result[0] = -result[0]; + result[1] = Math.PI - result[1]; + result[2] -= Math.PI; + } + if (result[0] > Math.PI * 0.5) { + result[0] = result[0] - Math.PI; + result[1] = Math.PI - result[1]; + result[2] -= Math.PI; + } + if (result[0] < -Math.PI * 0.5) { + result[0] = result[0] + Math.PI; + result[1] = -Math.PI - result[1]; + result[2] -= Math.PI; + } + while (result[1] < -Math.PI) result[1] += 2 * Math.PI; + while (result[1] >= Math.PI) result[1] -= 2 * Math.PI; + while (result[2] < -Math.PI) result[2] += 2 * Math.PI; + while (result[2] >= Math.PI) result[2] -= 2 * Math.PI; + return result; + }; + + /** + * (Property) Array defining a translation forward in z by 1 + * + * @property {array} inFront + * @static + * @final + */ + Transform.inFront = [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 1e-3, 1]; + + /** + * (Property) Array defining a translation backwards in z by 1 + * + * @property {array} behind + * @static + * @final + */ + Transform.behind = [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, -1e-3, 1]; + + module.exports = Transform; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/SpecParser',['require','exports','module','./Transform'],function(require, exports, module) { + var Transform = require('./Transform'); + + /** + * + * This object translates the rendering instructions ("render specs") + * that renderable components generate into document update + * instructions ("update specs"). Private. + * + * @private + * @class SpecParser + * @constructor + */ + function SpecParser() { + this.result = {}; + } + SpecParser._instance = new SpecParser(); + + /** + * Convert a render spec coming from the context's render chain to an + * update spec for the update chain. This is the only major entry point + * for a consumer of this class. + * + * @method parse + * @static + * @private + * + * @param {renderSpec} spec input render spec + * @param {Object} context context to do the parse in + * @return {Object} the resulting update spec (if no callback + * specified, else none) + */ + SpecParser.parse = function parse(spec, context) { + return SpecParser._instance.parse(spec, context); + }; + + /** + * Convert a renderSpec coming from the context's render chain to an update + * spec for the update chain. This is the only major entrypoint for a + * consumer of this class. + * + * @method parse + * + * @private + * @param {renderSpec} spec input render spec + * @param {Context} context + * @return {updateSpec} the resulting update spec + */ + SpecParser.prototype.parse = function parse(spec, context) { + this.reset(); + this._parseSpec(spec, context, Transform.identity); + return this.result; + }; + + /** + * Prepare SpecParser for re-use (or first use) by setting internal state + * to blank. + * + * @private + * @method reset + */ + SpecParser.prototype.reset = function reset() { + this.result = {}; + }; + + // Multiply matrix M by vector v + function _vecInContext(v, m) { + return [ + v[0] * m[0] + v[1] * m[4] + v[2] * m[8], + v[0] * m[1] + v[1] * m[5] + v[2] * m[9], + v[0] * m[2] + v[1] * m[6] + v[2] * m[10] + ]; + } + + var _zeroZero = [0, 0]; + + // From the provided renderSpec tree, recursively compose opacities, + // origins, transforms, and sizes corresponding to each surface id from + // the provided renderSpec tree structure. On completion, those + // properties of 'this' object should be ready to use to build an + // updateSpec. + SpecParser.prototype._parseSpec = function _parseSpec(spec, parentContext, sizeContext) { + var id; + var target; + var transform; + var opacity; + var origin; + var align; + var size; + + if (typeof spec === 'number') { + id = spec; + transform = parentContext.transform; + align = parentContext.align || _zeroZero; + if (parentContext.size && align && (align[0] || align[1])) { + var alignAdjust = [align[0] * parentContext.size[0], align[1] * parentContext.size[1], 0]; + transform = Transform.thenMove(transform, _vecInContext(alignAdjust, sizeContext)); + } + this.result[id] = { + transform: transform, + opacity: parentContext.opacity, + origin: parentContext.origin || _zeroZero, + align: parentContext.align || _zeroZero, + size: parentContext.size + }; + } + else if (!spec) { // placed here so 0 will be cached earlier + return; + } + else if (spec instanceof Array) { + for (var i = 0; i < spec.length; i++) { + this._parseSpec(spec[i], parentContext, sizeContext); + } + } + else { + target = spec.target; + transform = parentContext.transform; + opacity = parentContext.opacity; + origin = parentContext.origin; + align = parentContext.align; + size = parentContext.size; + var nextSizeContext = sizeContext; + + if (spec.opacity !== undefined) opacity = parentContext.opacity * spec.opacity; + if (spec.transform) transform = Transform.multiply(parentContext.transform, spec.transform); + if (spec.origin) { + origin = spec.origin; + nextSizeContext = parentContext.transform; + } + if (spec.align) align = spec.align; + + if (spec.size || spec.proportions) { + var parentSize = size; + size = [size[0], size[1]]; + + if (spec.size) { + if (spec.size[0] !== undefined) size[0] = spec.size[0]; + if (spec.size[1] !== undefined) size[1] = spec.size[1]; + } + + if (spec.proportions) { + if (spec.proportions[0] !== undefined) size[0] = size[0] * spec.proportions[0]; + if (spec.proportions[1] !== undefined) size[1] = size[1] * spec.proportions[1]; + } + + if (parentSize) { + if (align && (align[0] || align[1])) transform = Transform.thenMove(transform, _vecInContext([align[0] * parentSize[0], align[1] * parentSize[1], 0], sizeContext)); + if (origin && (origin[0] || origin[1])) transform = Transform.moveThen([-origin[0] * size[0], -origin[1] * size[1], 0], transform); + } + + nextSizeContext = parentContext.transform; + origin = null; + align = null; + } + + this._parseSpec(target, { + transform: transform, + opacity: opacity, + origin: origin, + align: align, + size: size + }, nextSizeContext); + } + }; + + module.exports = SpecParser; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/RenderNode',['require','exports','module','./Entity','./SpecParser'],function(require, exports, module) { + var Entity = require('./Entity'); + var SpecParser = require('./SpecParser'); + + /** + * A wrapper for inserting a renderable component (like a Modifer or + * Surface) into the render tree. + * + * @class RenderNode + * @constructor + * + * @param {Object} object Target renderable component + */ + function RenderNode(object) { + this._object = null; + this._child = null; + this._hasMultipleChildren = false; + this._isRenderable = false; + this._isModifier = false; + + this._resultCache = {}; + this._prevResults = {}; + + this._childResult = null; + + if (object) this.set(object); + } + + /** + * Append a renderable to the list of this node's children. + * This produces a new RenderNode in the tree. + * Note: Does not double-wrap if child is a RenderNode already. + * + * @method add + * @param {Object} child renderable object + * @return {RenderNode} new render node wrapping child + */ + RenderNode.prototype.add = function add(child) { + var childNode = (child instanceof RenderNode) ? child : new RenderNode(child); + if (this._child instanceof Array) this._child.push(childNode); + else if (this._child) { + this._child = [this._child, childNode]; + this._hasMultipleChildren = true; + this._childResult = []; // to be used later + } + else this._child = childNode; + + return childNode; + }; + + /** + * Return the single wrapped object. Returns null if this node has multiple child nodes. + * + * @method get + * + * @return {Ojbect} contained renderable object + */ + RenderNode.prototype.get = function get() { + return this._object || (this._hasMultipleChildren ? null : (this._child ? this._child.get() : null)); + }; + + /** + * Overwrite the list of children to contain the single provided object + * + * @method set + * @param {Object} child renderable object + * @return {RenderNode} this render node, or child if it is a RenderNode + */ + RenderNode.prototype.set = function set(child) { + this._childResult = null; + this._hasMultipleChildren = false; + this._isRenderable = child.render ? true : false; + this._isModifier = child.modify ? true : false; + this._object = child; + this._child = null; + if (child instanceof RenderNode) return child; + else return this; + }; + + /** + * Get render size of contained object. + * + * @method getSize + * @return {Array.Number} size of this or size of single child. + */ + RenderNode.prototype.getSize = function getSize() { + var result = null; + var target = this.get(); + if (target && target.getSize) result = target.getSize(); + if (!result && this._child && this._child.getSize) result = this._child.getSize(); + return result; + }; + + // apply results of rendering this subtree to the document + function _applyCommit(spec, context, cacheStorage) { + var result = SpecParser.parse(spec, context); + var keys = Object.keys(result); + for (var i = 0; i < keys.length; i++) { + var id = keys[i]; + var childNode = Entity.get(id); + var commitParams = result[id]; + commitParams.allocator = context.allocator; + var commitResult = childNode.commit(commitParams); + if (commitResult) _applyCommit(commitResult, context, cacheStorage); + else cacheStorage[id] = commitParams; + } + } + + /** + * Commit the content change from this node to the document. + * + * @private + * @method commit + * @param {Context} context render context + */ + RenderNode.prototype.commit = function commit(context) { + // free up some divs from the last loop + var prevKeys = Object.keys(this._prevResults); + for (var i = 0; i < prevKeys.length; i++) { + var id = prevKeys[i]; + if (this._resultCache[id] === undefined) { + var object = Entity.get(id); + if (object.cleanup) object.cleanup(context.allocator); + } + } + + this._prevResults = this._resultCache; + this._resultCache = {}; + _applyCommit(this.render(), context, this._resultCache); + }; + + /** + * Generate a render spec from the contents of the wrapped component. + * + * @private + * @method render + * + * @return {Object} render specification for the component subtree + * only under this node. + */ + RenderNode.prototype.render = function render() { + if (this._isRenderable) return this._object.render(); + + var result = null; + if (this._hasMultipleChildren) { + result = this._childResult; + var children = this._child; + for (var i = 0; i < children.length; i++) { + result[i] = children[i].render(); + } + } + else if (this._child) result = this._child.render(); + + return this._isModifier ? this._object.modify(result) : result; + }; + + module.exports = RenderNode; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/EventEmitter',['require','exports','module'],function(require, exports, module) { + /** + * EventEmitter represents a channel for events. + * + * @class EventEmitter + * @constructor + */ + function EventEmitter() { + this.listeners = {}; + this._owner = this; + } + + /** + * Trigger an event, sending to all downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} event event data + * @return {EventHandler} this + */ + EventEmitter.prototype.emit = function emit(type, event) { + var handlers = this.listeners[type]; + if (handlers) { + for (var i = 0; i < handlers.length; i++) { + handlers[i].call(this._owner, event); + } + } + return this; + }; + + /** + * Bind a callback function to an event type handled by this object. + * + * @method "on" + * + * @param {string} type event type key (for example, 'click') + * @param {function(string, Object)} handler callback + * @return {EventHandler} this + */ + EventEmitter.prototype.on = function on(type, handler) { + if (!(type in this.listeners)) this.listeners[type] = []; + var index = this.listeners[type].indexOf(handler); + if (index < 0) this.listeners[type].push(handler); + return this; + }; + + /** + * Alias for "on". + * @method addListener + */ + EventEmitter.prototype.addListener = EventEmitter.prototype.on; + + /** + * Unbind an event by type and handler. + * This undoes the work of "on". + * + * @method removeListener + * + * @param {string} type event type key (for example, 'click') + * @param {function} handler function object to remove + * @return {EventEmitter} this + */ + EventEmitter.prototype.removeListener = function removeListener(type, handler) { + var listener = this.listeners[type]; + if (listener !== undefined) { + var index = listener.indexOf(handler); + if (index >= 0) listener.splice(index, 1); + } + return this; + }; + + /** + * Call event handlers with this set to owner. + * + * @method bindThis + * + * @param {Object} owner object this EventEmitter belongs to + */ + EventEmitter.prototype.bindThis = function bindThis(owner) { + this._owner = owner; + }; + + module.exports = EventEmitter; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/EventHandler',['require','exports','module','./EventEmitter'],function(require, exports, module) { + var EventEmitter = require('./EventEmitter'); + + /** + * EventHandler forwards received events to a set of provided callback functions. + * It allows events to be captured, processed, and optionally piped through to other event handlers. + * + * @class EventHandler + * @extends EventEmitter + * @constructor + */ + function EventHandler() { + EventEmitter.apply(this, arguments); + + this.downstream = []; // downstream event handlers + this.downstreamFn = []; // downstream functions + + this.upstream = []; // upstream event handlers + this.upstreamListeners = {}; // upstream listeners + } + EventHandler.prototype = Object.create(EventEmitter.prototype); + EventHandler.prototype.constructor = EventHandler; + + /** + * Assign an event handler to receive an object's input events. + * + * @method setInputHandler + * @static + * + * @param {Object} object object to mix trigger, subscribe, and unsubscribe functions into + * @param {EventHandler} handler assigned event handler + */ + EventHandler.setInputHandler = function setInputHandler(object, handler) { + object.trigger = handler.trigger.bind(handler); + if (handler.subscribe && handler.unsubscribe) { + object.subscribe = handler.subscribe.bind(handler); + object.unsubscribe = handler.unsubscribe.bind(handler); + } + }; + + /** + * Assign an event handler to receive an object's output events. + * + * @method setOutputHandler + * @static + * + * @param {Object} object object to mix pipe, unpipe, on, addListener, and removeListener functions into + * @param {EventHandler} handler assigned event handler + */ + EventHandler.setOutputHandler = function setOutputHandler(object, handler) { + if (handler instanceof EventHandler) handler.bindThis(object); + object.pipe = handler.pipe.bind(handler); + object.unpipe = handler.unpipe.bind(handler); + object.on = handler.on.bind(handler); + object.addListener = object.on; + object.removeListener = handler.removeListener.bind(handler); + }; + + /** + * Trigger an event, sending to all downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} event event data + * @return {EventHandler} this + */ + EventHandler.prototype.emit = function emit(type, event) { + EventEmitter.prototype.emit.apply(this, arguments); + var i = 0; + for (i = 0; i < this.downstream.length; i++) { + if (this.downstream[i].trigger) this.downstream[i].trigger(type, event); + } + for (i = 0; i < this.downstreamFn.length; i++) { + this.downstreamFn[i](type, event); + } + return this; + }; + + /** + * Alias for emit + * @method addListener + */ + EventHandler.prototype.trigger = EventHandler.prototype.emit; + + /** + * Add event handler object to set of downstream handlers. + * + * @method pipe + * + * @param {EventHandler} target event handler target object + * @return {EventHandler} passed event handler + */ + EventHandler.prototype.pipe = function pipe(target) { + if (target.subscribe instanceof Function) return target.subscribe(this); + + var downstreamCtx = (target instanceof Function) ? this.downstreamFn : this.downstream; + var index = downstreamCtx.indexOf(target); + if (index < 0) downstreamCtx.push(target); + + if (target instanceof Function) target('pipe', null); + else if (target.trigger) target.trigger('pipe', null); + + return target; + }; + + /** + * Remove handler object from set of downstream handlers. + * Undoes work of "pipe". + * + * @method unpipe + * + * @param {EventHandler} target target handler object + * @return {EventHandler} provided target + */ + EventHandler.prototype.unpipe = function unpipe(target) { + if (target.unsubscribe instanceof Function) return target.unsubscribe(this); + + var downstreamCtx = (target instanceof Function) ? this.downstreamFn : this.downstream; + var index = downstreamCtx.indexOf(target); + if (index >= 0) { + downstreamCtx.splice(index, 1); + if (target instanceof Function) target('unpipe', null); + else if (target.trigger) target.trigger('unpipe', null); + return target; + } + else return false; + }; + + /** + * Bind a callback function to an event type handled by this object. + * + * @method "on" + * + * @param {string} type event type key (for example, 'click') + * @param {function(string, Object)} handler callback + * @return {EventHandler} this + */ + EventHandler.prototype.on = function on(type, handler) { + EventEmitter.prototype.on.apply(this, arguments); + if (!(type in this.upstreamListeners)) { + var upstreamListener = this.trigger.bind(this, type); + this.upstreamListeners[type] = upstreamListener; + for (var i = 0; i < this.upstream.length; i++) { + this.upstream[i].on(type, upstreamListener); + } + } + return this; + }; + + /** + * Alias for "on" + * @method addListener + */ + EventHandler.prototype.addListener = EventHandler.prototype.on; + + /** + * Listen for events from an upstream event handler. + * + * @method subscribe + * + * @param {EventEmitter} source source emitter object + * @return {EventHandler} this + */ + EventHandler.prototype.subscribe = function subscribe(source) { + var index = this.upstream.indexOf(source); + if (index < 0) { + this.upstream.push(source); + for (var type in this.upstreamListeners) { + source.on(type, this.upstreamListeners[type]); + } + } + return this; + }; + + /** + * Stop listening to events from an upstream event handler. + * + * @method unsubscribe + * + * @param {EventEmitter} source source emitter object + * @return {EventHandler} this + */ + EventHandler.prototype.unsubscribe = function unsubscribe(source) { + var index = this.upstream.indexOf(source); + if (index >= 0) { + this.upstream.splice(index, 1); + for (var type in this.upstreamListeners) { + source.removeListener(type, this.upstreamListeners[type]); + } + } + return this; + }; + + module.exports = EventHandler; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/ElementAllocator',['require','exports','module'],function(require, exports, module) { + + /** + * Internal helper object to Context that handles the process of + * creating and allocating DOM elements within a managed div. + * Private. + * + * @class ElementAllocator + * @constructor + * @private + * @param {Node} container document element in which Famo.us content will be inserted + */ + function ElementAllocator(container) { + if (!container) container = document.createDocumentFragment(); + this.container = container; + this.detachedNodes = {}; + this.nodeCount = 0; + } + + /** + * Move the document elements from their original container to a new one. + * + * @private + * @method migrate + * + * @param {Node} container document element to which Famo.us content will be migrated + */ + ElementAllocator.prototype.migrate = function migrate(container) { + var oldContainer = this.container; + if (container === oldContainer) return; + + if (oldContainer instanceof DocumentFragment) { + container.appendChild(oldContainer); + } + else { + while (oldContainer.hasChildNodes()) { + container.appendChild(oldContainer.firstChild); + } + } + + this.container = container; + }; + + /** + * Allocate an element of specified type from the pool. + * + * @private + * @method allocate + * + * @param {string} type type of element, e.g. 'div' + * @return {Node} allocated document element + */ + ElementAllocator.prototype.allocate = function allocate(type) { + type = type.toLowerCase(); + if (!(type in this.detachedNodes)) this.detachedNodes[type] = []; + var nodeStore = this.detachedNodes[type]; + var result; + if (nodeStore.length > 0) { + result = nodeStore.pop(); + } + else { + result = document.createElement(type); + this.container.appendChild(result); + } + this.nodeCount++; + return result; + }; + + /** + * De-allocate an element of specified type to the pool. + * + * @private + * @method deallocate + * + * @param {Node} element document element to deallocate + */ + ElementAllocator.prototype.deallocate = function deallocate(element) { + var nodeType = element.nodeName.toLowerCase(); + var nodeStore = this.detachedNodes[nodeType]; + nodeStore.push(element); + this.nodeCount--; + }; + + /** + * Get count of total allocated nodes in the document. + * + * @private + * @method getNodeCount + * + * @return {Number} total node count + */ + ElementAllocator.prototype.getNodeCount = function getNodeCount() { + return this.nodeCount; + }; + + module.exports = ElementAllocator; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/utilities/Utility',['require','exports','module'],function(require, exports, module) { + /** + * This namespace holds standalone functionality. + * Currently includes name mapping for transition curves, + * name mapping for origin pairs, and the after() function. + * + * @class Utility + * @static + */ + var Utility = {}; + + /** + * Table of direction array positions + * + * @property {object} Direction + * @final + */ + Utility.Direction = { + X: 0, + Y: 1, + Z: 2 + }; + + /** + * Return wrapper around callback function. Once the wrapper is called N + * times, invoke the callback function. Arguments and scope preserved. + * + * @method after + * + * @param {number} count number of calls before callback function invoked + * @param {Function} callback wrapped callback function + * + * @return {function} wrapped callback with coundown feature + */ + Utility.after = function after(count, callback) { + var counter = count; + return function() { + counter--; + if (counter === 0) callback.apply(this, arguments); + }; + }; + + /** + * Load a URL and return its contents in a callback + * + * @method loadURL + * + * @param {string} url URL of object + * @param {function} callback callback to dispatch with content + */ + Utility.loadURL = function loadURL(url, callback) { + var xhr = new XMLHttpRequest(); + xhr.onreadystatechange = function onreadystatechange() { + if (this.readyState === 4) { + if (callback) callback(this.responseText); + } + }; + xhr.open('GET', url); + xhr.send(); + }; + + /** + * Create a document fragment from a string of HTML + * + * @method createDocumentFragmentFromHTML + * + * @param {string} html HTML to convert to DocumentFragment + * + * @return {DocumentFragment} DocumentFragment representing input HTML + */ + Utility.createDocumentFragmentFromHTML = function createDocumentFragmentFromHTML(html) { + var element = document.createElement('div'); + element.innerHTML = html; + var result = document.createDocumentFragment(); + while (element.hasChildNodes()) result.appendChild(element.firstChild); + return result; + }; + + /* + * Deep clone an object. + * @param b {Object} Object to clone + * @return a {Object} Cloned object. + */ + Utility.clone = function clone(b) { + var a; + if (typeof b === 'object') { + a = (b instanceof Array) ? [] : {}; + for (var key in b) { + if (typeof b[key] === 'object' && b[key] !== null) { + if (b[key] instanceof Array) { + a[key] = new Array(b[key].length); + for (var i = 0; i < b[key].length; i++) { + a[key][i] = Utility.clone(b[key][i]); + } + } + else { + a[key] = Utility.clone(b[key]); + } + } + else { + a[key] = b[key]; + } + } + } + else { + a = b; + } + return a; + }; + + module.exports = Utility; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +/*eslint-disable new-cap */ +define('famous/transitions/MultipleTransition',['require','exports','module','../utilities/Utility'],function(require, exports, module) { + var Utility = require('../utilities/Utility'); + + /** + * Transition meta-method to support transitioning multiple + * values with scalar-only methods. + * + * + * @class MultipleTransition + * @constructor + * + * @param {Object} method Transionable class to multiplex + */ + function MultipleTransition(method) { + this.method = method; + this._instances = []; + this.state = []; + } + + MultipleTransition.SUPPORTS_MULTIPLE = true; + + /** + * Get the state of each transition. + * + * @method get + * + * @return state {Number|Array} state array + */ + MultipleTransition.prototype.get = function get() { + for (var i = 0; i < this._instances.length; i++) { + this.state[i] = this._instances[i].get(); + } + return this.state; + }; + + /** + * Set the end states with a shared transition, with optional callback. + * + * @method set + * + * @param {Number|Array} endState Final State. Use a multi-element argument for multiple transitions. + * @param {Object} transition Transition definition, shared among all instances + * @param {Function} callback called when all endStates have been reached. + */ + MultipleTransition.prototype.set = function set(endState, transition, callback) { + var _allCallback = Utility.after(endState.length, callback); + for (var i = 0; i < endState.length; i++) { + if (!this._instances[i]) this._instances[i] = new (this.method)(); + this._instances[i].set(endState[i], transition, _allCallback); + } + }; + + /** + * Reset all transitions to start state. + * + * @method reset + * + * @param {Number|Array} startState Start state + */ + MultipleTransition.prototype.reset = function reset(startState) { + for (var i = 0; i < startState.length; i++) { + if (!this._instances[i]) this._instances[i] = new (this.method)(); + this._instances[i].reset(startState[i]); + } + }; + + module.exports = MultipleTransition; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/transitions/TweenTransition',['require','exports','module'],function(require, exports, module) { + + /** + * + * A state maintainer for a smooth transition between + * numerically-specified states. Example numeric states include floats or + * Transfornm objects. + * + * An initial state is set with the constructor or set(startValue). A + * corresponding end state and transition are set with set(endValue, + * transition). Subsequent calls to set(endValue, transition) begin at + * the last state. Calls to get(timestamp) provide the _interpolated state + * along the way. + * + * Note that there is no event loop here - calls to get() are the only way + * to find out state projected to the current (or provided) time and are + * the only way to trigger callbacks. Usually this kind of object would + * be part of the render() path of a visible component. + * + * @class TweenTransition + * @constructor + * + * @param {Object} options TODO + * beginning state + */ + function TweenTransition(options) { + this.options = Object.create(TweenTransition.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + this._startTime = 0; + this._startValue = 0; + this._updateTime = 0; + this._endValue = 0; + this._curve = undefined; + this._duration = 0; + this._active = false; + this._callback = undefined; + this.state = 0; + this.velocity = undefined; + } + + /** + * Transition curves mapping independent variable t from domain [0,1] to a + * range within [0,1]. Includes functions 'linear', 'easeIn', 'easeOut', + * 'easeInOut', 'easeOutBounce', 'spring'. + * + * @property {object} Curve + * @final + */ + TweenTransition.Curves = { + linear: function(t) { + return t; + }, + easeIn: function(t) { + return t*t; + }, + easeOut: function(t) { + return t*(2-t); + }, + easeInOut: function(t) { + if (t <= 0.5) return 2*t*t; + else return -2*t*t + 4*t - 1; + }, + easeOutBounce: function(t) { + return t*(3 - 2*t); + }, + spring: function(t) { + return (1 - t) * Math.sin(6 * Math.PI * t) + t; + } + }; + + TweenTransition.SUPPORTS_MULTIPLE = true; + TweenTransition.DEFAULT_OPTIONS = { + curve: TweenTransition.Curves.linear, + duration: 500, + speed: 0 /* considered only if positive */ + }; + + var registeredCurves = {}; + + /** + * Add "unit" curve to internal dictionary of registered curves. + * + * @method registerCurve + * + * @static + * + * @param {string} curveName dictionary key + * @param {unitCurve} curve function of one numeric variable mapping [0,1] + * to range inside [0,1] + * @return {boolean} false if key is taken, else true + */ + TweenTransition.registerCurve = function registerCurve(curveName, curve) { + if (!registeredCurves[curveName]) { + registeredCurves[curveName] = curve; + return true; + } + else { + return false; + } + }; + + /** + * Remove object with key "curveName" from internal dictionary of registered + * curves. + * + * @method unregisterCurve + * + * @static + * + * @param {string} curveName dictionary key + * @return {boolean} false if key has no dictionary value + */ + TweenTransition.unregisterCurve = function unregisterCurve(curveName) { + if (registeredCurves[curveName]) { + delete registeredCurves[curveName]; + return true; + } + else { + return false; + } + }; + + /** + * Retrieve function with key "curveName" from internal dictionary of + * registered curves. Default curves are defined in the + * TweenTransition.Curves array, where the values represent + * unitCurve functions. + * + * @method getCurve + * + * @static + * + * @param {string} curveName dictionary key + * @return {unitCurve} curve function of one numeric variable mapping [0,1] + * to range inside [0,1] + */ + TweenTransition.getCurve = function getCurve(curveName) { + var curve = registeredCurves[curveName]; + if (curve !== undefined) return curve; + else throw new Error('curve not registered'); + }; + + /** + * Retrieve all available curves. + * + * @method getCurves + * + * @static + * + * @return {object} curve functions of one numeric variable mapping [0,1] + * to range inside [0,1] + */ + TweenTransition.getCurves = function getCurves() { + return registeredCurves; + }; + + // Interpolate: If a linear function f(0) = a, f(1) = b, then return f(t) + function _interpolate(a, b, t) { + return ((1 - t) * a) + (t * b); + } + + function _clone(obj) { + if (obj instanceof Object) { + if (obj instanceof Array) return obj.slice(0); + else return Object.create(obj); + } + else return obj; + } + + // Fill in missing properties in "transition" with those in defaultTransition, and + // convert internal named curve to function object, returning as new + // object. + function _normalize(transition, defaultTransition) { + var result = {curve: defaultTransition.curve}; + if (defaultTransition.duration) result.duration = defaultTransition.duration; + if (defaultTransition.speed) result.speed = defaultTransition.speed; + if (transition instanceof Object) { + if (transition.duration !== undefined) result.duration = transition.duration; + if (transition.curve) result.curve = transition.curve; + if (transition.speed) result.speed = transition.speed; + } + if (typeof result.curve === 'string') result.curve = TweenTransition.getCurve(result.curve); + return result; + } + + /** + * Set internal options, overriding any default options. + * + * @method setOptions + * + * + * @param {Object} options options object + * @param {Object} [options.curve] function mapping [0,1] to [0,1] or identifier + * @param {Number} [options.duration] duration in ms + * @param {Number} [options.speed] speed in pixels per ms + */ + TweenTransition.prototype.setOptions = function setOptions(options) { + if (options.curve !== undefined) this.options.curve = options.curve; + if (options.duration !== undefined) this.options.duration = options.duration; + if (options.speed !== undefined) this.options.speed = options.speed; + }; + + /** + * Add transition to end state to the queue of pending transitions. Special + * Use: calling without a transition resets the object to that state with + * no pending actions + * + * @method set + * + * + * @param {number|FamousMatrix|Array.Number|Object.} endValue + * end state to which we _interpolate + * @param {transition=} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {function()=} callback Zero-argument function to call on observed + * completion (t=1) + */ + TweenTransition.prototype.set = function set(endValue, transition, callback) { + if (!transition) { + this.reset(endValue); + if (callback) callback(); + return; + } + + this._startValue = _clone(this.get()); + transition = _normalize(transition, this.options); + if (transition.speed) { + var startValue = this._startValue; + if (startValue instanceof Object) { + var variance = 0; + for (var i in startValue) variance += (endValue[i] - startValue[i]) * (endValue[i] - startValue[i]); + transition.duration = Math.sqrt(variance) / transition.speed; + } + else { + transition.duration = Math.abs(endValue - startValue) / transition.speed; + } + } + + this._startTime = Date.now(); + this._endValue = _clone(endValue); + this._startVelocity = _clone(transition.velocity); + this._duration = transition.duration; + this._curve = transition.curve; + this._active = true; + this._callback = callback; + }; + + /** + * Cancel all transitions and reset to a stable state + * + * @method reset + * + * @param {number|Array.Number|Object.} startValue + * starting state + * @param {number} startVelocity + * starting velocity + */ + TweenTransition.prototype.reset = function reset(startValue, startVelocity) { + if (this._callback) { + var callback = this._callback; + this._callback = undefined; + callback(); + } + this.state = _clone(startValue); + this.velocity = _clone(startVelocity); + this._startTime = 0; + this._duration = 0; + this._updateTime = 0; + this._startValue = this.state; + this._startVelocity = this.velocity; + this._endValue = this.state; + this._active = false; + }; + + /** + * Get current velocity + * + * @method getVelocity + * + * @returns {Number} velocity + */ + TweenTransition.prototype.getVelocity = function getVelocity() { + return this.velocity; + }; + + /** + * Get interpolated state of current action at provided time. If the last + * action has completed, invoke its callback. + * + * @method get + * + * + * @param {number=} timestamp Evaluate the curve at a normalized version of this + * time. If omitted, use current time. (Unix epoch time) + * @return {number|Object.} beginning state + * _interpolated to this point in time. + */ + TweenTransition.prototype.get = function get(timestamp) { + this.update(timestamp); + return this.state; + }; + + function _calculateVelocity(current, start, curve, duration, t) { + var velocity; + var eps = 1e-7; + var speed = (curve(t) - curve(t - eps)) / eps; + if (current instanceof Array) { + velocity = []; + for (var i = 0; i < current.length; i++){ + if (typeof current[i] === 'number') + velocity[i] = speed * (current[i] - start[i]) / duration; + else + velocity[i] = 0; + } + + } + else velocity = speed * (current - start) / duration; + return velocity; + } + + function _calculateState(start, end, t) { + var state; + if (start instanceof Array) { + state = []; + for (var i = 0; i < start.length; i++) { + if (typeof start[i] === 'number') + state[i] = _interpolate(start[i], end[i], t); + else + state[i] = start[i]; + } + } + else state = _interpolate(start, end, t); + return state; + } + + /** + * Update internal state to the provided timestamp. This may invoke the last + * callback and begin a new action. + * + * @method update + * + * + * @param {number=} timestamp Evaluate the curve at a normalized version of this + * time. If omitted, use current time. (Unix epoch time) + */ + TweenTransition.prototype.update = function update(timestamp) { + if (!this._active) { + if (this._callback) { + var callback = this._callback; + this._callback = undefined; + callback(); + } + return; + } + + if (!timestamp) timestamp = Date.now(); + if (this._updateTime >= timestamp) return; + this._updateTime = timestamp; + + var timeSinceStart = timestamp - this._startTime; + if (timeSinceStart >= this._duration) { + this.state = this._endValue; + this.velocity = _calculateVelocity(this.state, this._startValue, this._curve, this._duration, 1); + this._active = false; + } + else if (timeSinceStart < 0) { + this.state = this._startValue; + this.velocity = this._startVelocity; + } + else { + var t = timeSinceStart / this._duration; + this.state = _calculateState(this._startValue, this._endValue, this._curve(t)); + this.velocity = _calculateVelocity(this.state, this._startValue, this._curve, this._duration, t); + } + }; + + /** + * Is there at least one action pending completion? + * + * @method isActive + * + * + * @return {boolean} + */ + TweenTransition.prototype.isActive = function isActive() { + return this._active; + }; + + /** + * Halt transition at current state and erase all pending actions. + * + * @method halt + * + */ + TweenTransition.prototype.halt = function halt() { + this.reset(this.get()); + }; + + // Register all the default curves + TweenTransition.registerCurve('linear', TweenTransition.Curves.linear); + TweenTransition.registerCurve('easeIn', TweenTransition.Curves.easeIn); + TweenTransition.registerCurve('easeOut', TweenTransition.Curves.easeOut); + TweenTransition.registerCurve('easeInOut', TweenTransition.Curves.easeInOut); + TweenTransition.registerCurve('easeOutBounce', TweenTransition.Curves.easeOutBounce); + TweenTransition.registerCurve('spring', TweenTransition.Curves.spring); + + TweenTransition.customCurve = function customCurve(v1, v2) { + v1 = v1 || 0; v2 = v2 || 0; + return function(t) { + return v1*t + (-2*v1 - v2 + 3)*t*t + (v1 + v2 - 2)*t*t*t; + }; + }; + + module.exports = TweenTransition; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +/*eslint-disable new-cap */ +define('famous/transitions/Transitionable',['require','exports','module','./MultipleTransition','./TweenTransition'],function(require, exports, module) { + var MultipleTransition = require('./MultipleTransition'); + var TweenTransition = require('./TweenTransition'); + + /** + * A state maintainer for a smooth transition between + * numerically-specified states. Example numeric states include floats or + * Transform objects. + * + * An initial state is set with the constructor or set(startState). A + * corresponding end state and transition are set with set(endState, + * transition). Subsequent calls to set(endState, transition) begin at + * the last state. Calls to get(timestamp) provide the interpolated state + * along the way. + * + * Note that there is no event loop here - calls to get() are the only way + * to find state projected to the current (or provided) time and are + * the only way to trigger callbacks. Usually this kind of object would + * be part of the render() path of a visible component. + * + * @class Transitionable + * @constructor + * @param {number|Array.Number|Object.} start + * beginning state + */ + function Transitionable(start) { + this.currentAction = null; + this.actionQueue = []; + this.callbackQueue = []; + + this.state = 0; + this.velocity = undefined; + this._callback = undefined; + this._engineInstance = null; + this._currentMethod = null; + + this.set(start); + } + + var transitionMethods = {}; + + Transitionable.register = function register(methods) { + var success = true; + for (var method in methods) { + if (!Transitionable.registerMethod(method, methods[method])) + success = false; + } + return success; + }; + + Transitionable.registerMethod = function registerMethod(name, engineClass) { + if (!(name in transitionMethods)) { + transitionMethods[name] = engineClass; + return true; + } + else return false; + }; + + Transitionable.unregisterMethod = function unregisterMethod(name) { + if (name in transitionMethods) { + delete transitionMethods[name]; + return true; + } + else return false; + }; + + function _loadNext() { + if (this._callback) { + var callback = this._callback; + this._callback = undefined; + callback(); + } + if (this.actionQueue.length <= 0) { + this.set(this.get()); // no update required + return; + } + this.currentAction = this.actionQueue.shift(); + this._callback = this.callbackQueue.shift(); + + var method = null; + var endValue = this.currentAction[0]; + var transition = this.currentAction[1]; + if (transition instanceof Object && transition.method) { + method = transition.method; + if (typeof method === 'string') method = transitionMethods[method]; + } + else { + method = TweenTransition; + } + + if (this._currentMethod !== method) { + if (!(endValue instanceof Object) || method.SUPPORTS_MULTIPLE === true || endValue.length <= method.SUPPORTS_MULTIPLE) { + this._engineInstance = new method(); + } + else { + this._engineInstance = new MultipleTransition(method); + } + this._currentMethod = method; + } + + this._engineInstance.reset(this.state, this.velocity); + if (this.velocity !== undefined) transition.velocity = this.velocity; + this._engineInstance.set(endValue, transition, _loadNext.bind(this)); + } + + /** + * Add transition to end state to the queue of pending transitions. Special + * Use: calling without a transition resets the object to that state with + * no pending actions + * + * @method set + * + * @param {number|FamousMatrix|Array.Number|Object.} endState + * end state to which we interpolate + * @param {transition=} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {function()=} callback Zero-argument function to call on observed + * completion (t=1) + */ + Transitionable.prototype.set = function set(endState, transition, callback) { + if (!transition) { + this.reset(endState); + if (callback) callback(); + return this; + } + + var action = [endState, transition]; + this.actionQueue.push(action); + this.callbackQueue.push(callback); + if (!this.currentAction) _loadNext.call(this); + return this; + }; + + /** + * Cancel all transitions and reset to a stable state + * + * @method reset + * + * @param {number|Array.Number|Object.} startState + * stable state to set to + */ + Transitionable.prototype.reset = function reset(startState, startVelocity) { + this._currentMethod = null; + this._engineInstance = null; + this._callback = undefined; + this.state = startState; + this.velocity = startVelocity; + this.currentAction = null; + this.actionQueue = []; + this.callbackQueue = []; + }; + + /** + * Add delay action to the pending action queue queue. + * + * @method delay + * + * @param {number} duration delay time (ms) + * @param {function} callback Zero-argument function to call on observed + * completion (t=1) + */ + Transitionable.prototype.delay = function delay(duration, callback) { + var endValue; + if (this.actionQueue.length) endValue = this.actionQueue[this.actionQueue.length - 1][0]; + else if (this.currentAction) endValue = this.currentAction[0]; + else endValue = this.get(); + + return this.set(endValue, { duration: duration, + curve: function() { + return 0; + }}, + callback + ); + }; + + /** + * Get interpolated state of current action at provided time. If the last + * action has completed, invoke its callback. + * + * @method get + * + * @param {number=} timestamp Evaluate the curve at a normalized version of this + * time. If omitted, use current time. (Unix epoch time) + * @return {number|Object.} beginning state + * interpolated to this point in time. + */ + Transitionable.prototype.get = function get(timestamp) { + if (this._engineInstance) { + if (this._engineInstance.getVelocity) + this.velocity = this._engineInstance.getVelocity(); + this.state = this._engineInstance.get(timestamp); + } + return this.state; + }; + + /** + * Is there at least one action pending completion? + * + * @method isActive + * + * @return {boolean} + */ + Transitionable.prototype.isActive = function isActive() { + return !!this.currentAction; + }; + + /** + * Halt transition at current state and erase all pending actions. + * + * @method halt + */ + Transitionable.prototype.halt = function halt() { + return this.set(this.get()); + }; + + module.exports = Transitionable; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/Context',['require','exports','module','./RenderNode','./EventHandler','./ElementAllocator','./Transform','../transitions/Transitionable'],function(require, exports, module) { + var RenderNode = require('./RenderNode'); + var EventHandler = require('./EventHandler'); + var ElementAllocator = require('./ElementAllocator'); + var Transform = require('./Transform'); + var Transitionable = require('../transitions/Transitionable'); + + var _zeroZero = [0, 0]; + var usePrefix = !('perspective' in document.documentElement.style); + + function _getElementSize() { + var element = this.container; + return [element.clientWidth, element.clientHeight]; + } + + var _setPerspective = usePrefix ? function(element, perspective) { + element.style.webkitPerspective = perspective ? perspective.toFixed() + 'px' : ''; + } : function(element, perspective) { + element.style.perspective = perspective ? perspective.toFixed() + 'px' : ''; + }; + + /** + * The top-level container for a Famous-renderable piece of the document. + * It is directly updated by the process-wide Engine object, and manages one + * render tree root, which can contain other renderables. + * + * @class Context + * @constructor + * @private + * @param {Node} container Element in which content will be inserted + */ + function Context(container) { + this.container = container; + this._allocator = new ElementAllocator(container); + + this._node = new RenderNode(); + this._eventOutput = new EventHandler(); + this._size = _getElementSize.call(this); + + this._perspectiveState = new Transitionable(0); + this._perspective = undefined; + + this._nodeContext = { + allocator: this._allocator, + transform: Transform.identity, + opacity: 1, + origin: _zeroZero, + align: _zeroZero, + size: this._size + }; + + this._eventOutput.on('resize', function() { + this.setSize(_getElementSize.call(this)); + }.bind(this)); + + } + + // Note: Unused + Context.prototype.getAllocator = function getAllocator() { + return this._allocator; + }; + + /** + * Add renderables to this Context's render tree. + * + * @method add + * + * @param {Object} obj renderable object + * @return {RenderNode} RenderNode wrapping this object, if not already a RenderNode + */ + Context.prototype.add = function add(obj) { + return this._node.add(obj); + }; + + /** + * Move this Context to another containing document element. + * + * @method migrate + * + * @param {Node} container Element to which content will be migrated + */ + Context.prototype.migrate = function migrate(container) { + if (container === this.container) return; + this.container = container; + this._allocator.migrate(container); + }; + + /** + * Gets viewport size for Context. + * + * @method getSize + * + * @return {Array.Number} viewport size as [width, height] + */ + Context.prototype.getSize = function getSize() { + return this._size; + }; + + /** + * Sets viewport size for Context. + * + * @method setSize + * + * @param {Array.Number} size [width, height]. If unspecified, use size of root document element. + */ + Context.prototype.setSize = function setSize(size) { + if (!size) size = _getElementSize.call(this); + this._size[0] = size[0]; + this._size[1] = size[1]; + }; + + /** + * Commit this Context's content changes to the document. + * + * @private + * @method update + * @param {Object} contextParameters engine commit specification + */ + Context.prototype.update = function update(contextParameters) { + if (contextParameters) { + if (contextParameters.transform) this._nodeContext.transform = contextParameters.transform; + if (contextParameters.opacity) this._nodeContext.opacity = contextParameters.opacity; + if (contextParameters.origin) this._nodeContext.origin = contextParameters.origin; + if (contextParameters.align) this._nodeContext.align = contextParameters.align; + if (contextParameters.size) this._nodeContext.size = contextParameters.size; + } + var perspective = this._perspectiveState.get(); + if (perspective !== this._perspective) { + _setPerspective(this.container, perspective); + this._perspective = perspective; + } + + this._node.commit(this._nodeContext); + }; + + /** + * Get current perspective of this context in pixels. + * + * @method getPerspective + * @return {Number} depth perspective in pixels + */ + Context.prototype.getPerspective = function getPerspective() { + return this._perspectiveState.get(); + }; + + /** + * Set current perspective of this context in pixels. + * + * @method setPerspective + * @param {Number} perspective in pixels + * @param {Object} [transition] Transitionable object for applying the change + * @param {function(Object)} callback function called on completion of transition + */ + Context.prototype.setPerspective = function setPerspective(perspective, transition, callback) { + return this._perspectiveState.set(perspective, transition, callback); + }; + + /** + * Trigger an event, sending to all downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} event event data + * @return {EventHandler} this + */ + Context.prototype.emit = function emit(type, event) { + return this._eventOutput.emit(type, event); + }; + + /** + * Bind a callback function to an event type handled by this object. + * + * @method "on" + * + * @param {string} type event type key (for example, 'click') + * @param {function(string, Object)} handler callback + * @return {EventHandler} this + */ + Context.prototype.on = function on(type, handler) { + return this._eventOutput.on(type, handler); + }; + + /** + * Unbind an event by type and handler. + * This undoes the work of "on". + * + * @method removeListener + * + * @param {string} type event type key (for example, 'click') + * @param {function} handler function object to remove + * @return {EventHandler} internal event handler object (for chaining) + */ + Context.prototype.removeListener = function removeListener(type, handler) { + return this._eventOutput.removeListener(type, handler); + }; + + /** + * Add event handler object to set of downstream handlers. + * + * @method pipe + * + * @param {EventHandler} target event handler target object + * @return {EventHandler} passed event handler + */ + Context.prototype.pipe = function pipe(target) { + return this._eventOutput.pipe(target); + }; + + /** + * Remove handler object from set of downstream handlers. + * Undoes work of "pipe". + * + * @method unpipe + * + * @param {EventHandler} target target handler object + * @return {EventHandler} provided target + */ + Context.prototype.unpipe = function unpipe(target) { + return this._eventOutput.unpipe(target); + }; + + module.exports = Context; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/ElementOutput',['require','exports','module','./Entity','./EventHandler','./Transform'],function(require, exports, module) { + var Entity = require('./Entity'); + var EventHandler = require('./EventHandler'); + var Transform = require('./Transform'); + + var usePrefix = !('transform' in document.documentElement.style); + var devicePixelRatio = window.devicePixelRatio || 1; + + /** + * A base class for viewable content and event + * targets inside a Famo.us application, containing a renderable document + * fragment. Like an HTML div, it can accept internal markup, + * properties, classes, and handle events. + * + * @class ElementOutput + * @constructor + * + * @param {Node} element document parent of this container + */ + function ElementOutput(element) { + this._matrix = null; + this._opacity = 1; + this._origin = null; + this._size = null; + + this._eventOutput = new EventHandler(); + this._eventOutput.bindThis(this); + + /** @ignore */ + this.eventForwarder = function eventForwarder(event) { + this._eventOutput.emit(event.type, event); + }.bind(this); + + this.id = Entity.register(this); + this._element = null; + this._sizeDirty = false; + this._originDirty = false; + this._transformDirty = false; + + this._invisible = false; + if (element) this.attach(element); + } + + /** + * Bind a callback function to an event type handled by this object. + * + * @method "on" + * + * @param {string} type event type key (for example, 'click') + * @param {function(string, Object)} fn handler callback + * @return {EventHandler} this + */ + ElementOutput.prototype.on = function on(type, fn) { + if (this._element) this._element.addEventListener(type, this.eventForwarder); + this._eventOutput.on(type, fn); + }; + + /** + * Unbind an event by type and handler. + * This undoes the work of "on" + * + * @method removeListener + * @param {string} type event type key (for example, 'click') + * @param {function(string, Object)} fn handler + */ + ElementOutput.prototype.removeListener = function removeListener(type, fn) { + this._eventOutput.removeListener(type, fn); + }; + + /** + * Trigger an event, sending to all downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} [event] event data + * @return {EventHandler} this + */ + ElementOutput.prototype.emit = function emit(type, event) { + if (event && !event.origin) event.origin = this; + var handled = this._eventOutput.emit(type, event); + if (handled && event && event.stopPropagation) event.stopPropagation(); + return handled; + }; + + /** + * Add event handler object to set of downstream handlers. + * + * @method pipe + * + * @param {EventHandler} target event handler target object + * @return {EventHandler} passed event handler + */ + ElementOutput.prototype.pipe = function pipe(target) { + return this._eventOutput.pipe(target); + }; + + /** + * Remove handler object from set of downstream handlers. + * Undoes work of "pipe" + * + * @method unpipe + * + * @param {EventHandler} target target handler object + * @return {EventHandler} provided target + */ + ElementOutput.prototype.unpipe = function unpipe(target) { + return this._eventOutput.unpipe(target); + }; + + /** + * Return spec for this surface. Note that for a base surface, this is + * simply an id. + * + * @method render + * @private + * @return {Object} render spec for this surface (spec id) + */ + ElementOutput.prototype.render = function render() { + return this.id; + }; + + // Attach Famous event handling to document events emanating from target + // document element. This occurs just after attachment to the document. + // Calling this enables methods like #on and #pipe. + function _addEventListeners(target) { + for (var i in this._eventOutput.listeners) { + target.addEventListener(i, this.eventForwarder); + } + } + + // Detach Famous event handling from document events emanating from target + // document element. This occurs just before detach from the document. + function _removeEventListeners(target) { + for (var i in this._eventOutput.listeners) { + target.removeEventListener(i, this.eventForwarder); + } + } + + /** + * Return a Matrix's webkit css representation to be used with the + * CSS3 -webkit-transform style. + * Example: -webkit-transform: matrix3d(1,0,0,0,0,1,0,0,0,0,1,0,716,243,0,1) + * + * @method _formatCSSTransform + * @private + * @param {FamousMatrix} m matrix + * @return {string} matrix3d CSS style representation of the transform + */ + function _formatCSSTransform(m) { + m[12] = Math.round(m[12] * devicePixelRatio) / devicePixelRatio; + m[13] = Math.round(m[13] * devicePixelRatio) / devicePixelRatio; + + var result = 'matrix3d('; + for (var i = 0; i < 15; i++) { + result += (m[i] < 0.000001 && m[i] > -0.000001) ? '0,' : m[i] + ','; + } + result += m[15] + ')'; + return result; + } + + /** + * Directly apply given FamousMatrix to the document element as the + * appropriate webkit CSS style. + * + * @method setMatrix + * + * @static + * @private + * @param {Element} element document element + * @param {FamousMatrix} matrix + */ + + var _setMatrix; + if (usePrefix) { + _setMatrix = function(element, matrix) { + element.style.webkitTransform = _formatCSSTransform(matrix); + }; + } + else { + _setMatrix = function(element, matrix) { + element.style.transform = _formatCSSTransform(matrix); + }; + } + + // format origin as CSS percentage string + function _formatCSSOrigin(origin) { + return (100 * origin[0]) + '% ' + (100 * origin[1]) + '%'; + } + + // Directly apply given origin coordinates to the document element as the + // appropriate webkit CSS style. + var _setOrigin = usePrefix ? function(element, origin) { + element.style.webkitTransformOrigin = _formatCSSOrigin(origin); + } : function(element, origin) { + element.style.transformOrigin = _formatCSSOrigin(origin); + }; + + // Shrink given document element until it is effectively invisible. + var _setInvisible = usePrefix ? function(element) { + element.style.webkitTransform = 'scale3d(0.0001,0.0001,0.0001)'; + element.style.opacity = 0; + } : function(element) { + element.style.transform = 'scale3d(0.0001,0.0001,0.0001)'; + element.style.opacity = 0; + }; + + function _xyNotEquals(a, b) { + return (a && b) ? (a[0] !== b[0] || a[1] !== b[1]) : a !== b; + } + + /** + * Apply changes from this component to the corresponding document element. + * This includes changes to classes, styles, size, content, opacity, origin, + * and matrix transforms. + * + * @private + * @method commit + * @param {Context} context commit context + */ + ElementOutput.prototype.commit = function commit(context) { + var target = this._element; + if (!target) return; + + var matrix = context.transform; + var opacity = context.opacity; + var origin = context.origin; + var size = context.size; + + if (!matrix && this._matrix) { + this._matrix = null; + this._opacity = 0; + _setInvisible(target); + return; + } + + if (_xyNotEquals(this._origin, origin)) this._originDirty = true; + if (Transform.notEquals(this._matrix, matrix)) this._transformDirty = true; + + if (this._invisible) { + this._invisible = false; + this._element.style.display = ''; + } + + if (this._opacity !== opacity) { + this._opacity = opacity; + target.style.opacity = (opacity >= 1) ? '0.999999' : opacity; + } + + if (this._transformDirty || this._originDirty || this._sizeDirty) { + if (this._sizeDirty) this._sizeDirty = false; + + if (this._originDirty) { + if (origin) { + if (!this._origin) this._origin = [0, 0]; + this._origin[0] = origin[0]; + this._origin[1] = origin[1]; + } + else this._origin = null; + _setOrigin(target, this._origin); + this._originDirty = false; + } + + if (!matrix) matrix = Transform.identity; + this._matrix = matrix; + var aaMatrix = this._size ? Transform.thenMove(matrix, [-this._size[0]*origin[0], -this._size[1]*origin[1], 0]) : matrix; + _setMatrix(target, aaMatrix); + this._transformDirty = false; + } + }; + + ElementOutput.prototype.cleanup = function cleanup() { + if (this._element) { + this._invisible = true; + this._element.style.display = 'none'; + } + }; + + /** + * Place the document element that this component manages into the document. + * + * @private + * @method attach + * @param {Node} target document parent of this container + */ + ElementOutput.prototype.attach = function attach(target) { + this._element = target; + _addEventListeners.call(this, target); + }; + + /** + * Remove any contained document content associated with this surface + * from the actual document. + * + * @private + * @method detach + */ + ElementOutput.prototype.detach = function detach() { + var target = this._element; + if (target) { + _removeEventListeners.call(this, target); + if (this._invisible) { + this._invisible = false; + this._element.style.display = ''; + } + } + this._element = null; + return target; + }; + + module.exports = ElementOutput; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/OptionsManager',['require','exports','module','./EventHandler'],function(require, exports, module) { + var EventHandler = require('./EventHandler'); + + /** + * A collection of methods for setting options which can be extended + * onto other classes. + * + * + * **** WARNING **** + * You can only pass through objects that will compile into valid JSON. + * + * Valid options: + * Strings, + * Arrays, + * Objects, + * Numbers, + * Nested Objects, + * Nested Arrays. + * + * This excludes: + * Document Fragments, + * Functions + * @class OptionsManager + * @constructor + * @param {Object} value options dictionary + */ + function OptionsManager(value) { + this._value = value; + this.eventOutput = null; + } + + /** + * Create options manager from source dictionary with arguments overriden by patch dictionary. + * + * @static + * @method OptionsManager.patch + * + * @param {Object} source source arguments + * @param {...Object} data argument additions and overwrites + * @return {Object} source object + */ + OptionsManager.patch = function patchObject(source, data) { + var manager = new OptionsManager(source); + for (var i = 1; i < arguments.length; i++) manager.patch(arguments[i]); + return source; + }; + + function _createEventOutput() { + this.eventOutput = new EventHandler(); + this.eventOutput.bindThis(this); + EventHandler.setOutputHandler(this, this.eventOutput); + } + + /** + * Create OptionsManager from source with arguments overriden by patches. + * Triggers 'change' event on this object's event handler if the state of + * the OptionsManager changes as a result. + * + * @method patch + * + * @param {...Object} arguments list of patch objects + * @return {OptionsManager} this + */ + OptionsManager.prototype.patch = function patch() { + var myState = this._value; + for (var i = 0; i < arguments.length; i++) { + var data = arguments[i]; + for (var k in data) { + if ((k in myState) && (data[k] && data[k].constructor === Object) && (myState[k] && myState[k].constructor === Object)) { + if (!myState.hasOwnProperty(k)) myState[k] = Object.create(myState[k]); + this.key(k).patch(data[k]); + if (this.eventOutput) this.eventOutput.emit('change', {id: k, value: this.key(k).value()}); + } + else this.set(k, data[k]); + } + } + return this; + }; + + /** + * Alias for patch + * + * @method setOptions + * + */ + OptionsManager.prototype.setOptions = OptionsManager.prototype.patch; + + /** + * Return OptionsManager based on sub-object retrieved by key + * + * @method key + * + * @param {string} identifier key + * @return {OptionsManager} new options manager with the value + */ + OptionsManager.prototype.key = function key(identifier) { + var result = new OptionsManager(this._value[identifier]); + if (!(result._value instanceof Object) || result._value instanceof Array) result._value = {}; + return result; + }; + + /** + * Look up value by key or get the full options hash + * @method get + * + * @param {string} key key + * @return {Object} associated object or full options hash + */ + OptionsManager.prototype.get = function get(key) { + return key ? this._value[key] : this._value; + }; + + /** + * Alias for get + * @method getOptions + */ + OptionsManager.prototype.getOptions = OptionsManager.prototype.get; + + /** + * Set key to value. Outputs 'change' event if a value is overwritten. + * + * @method set + * + * @param {string} key key string + * @param {Object} value value object + * @return {OptionsManager} new options manager based on the value object + */ + OptionsManager.prototype.set = function set(key, value) { + var originalValue = this.get(key); + this._value[key] = value; + if (this.eventOutput && value !== originalValue) this.eventOutput.emit('change', {id: key, value: value}); + return this; + }; + + /** + * Bind a callback function to an event type handled by this object. + * + * @method "on" + * + * @param {string} type event type key (for example, 'change') + * @param {function(string, Object)} handler callback + * @return {EventHandler} this + */ + OptionsManager.prototype.on = function on() { + _createEventOutput.call(this); + return this.on.apply(this, arguments); + }; + + /** + * Unbind an event by type and handler. + * This undoes the work of "on". + * + * @method removeListener + * + * @param {string} type event type key (for example, 'change') + * @param {function} handler function object to remove + * @return {EventHandler} internal event handler object (for chaining) + */ + OptionsManager.prototype.removeListener = function removeListener() { + _createEventOutput.call(this); + return this.removeListener.apply(this, arguments); + }; + + /** + * Add event handler object to set of downstream handlers. + * + * @method pipe + * + * @param {EventHandler} target event handler target object + * @return {EventHandler} passed event handler + */ + OptionsManager.prototype.pipe = function pipe() { + _createEventOutput.call(this); + return this.pipe.apply(this, arguments); + }; + + /** + * Remove handler object from set of downstream handlers. + * Undoes work of "pipe" + * + * @method unpipe + * + * @param {EventHandler} target target handler object + * @return {EventHandler} provided target + */ + OptionsManager.prototype.unpipe = function unpipe() { + _createEventOutput.call(this); + return this.unpipe.apply(this, arguments); + }; + + module.exports = OptionsManager; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/Engine',['require','exports','module','./Context','./EventHandler','./OptionsManager'],function(require, exports, module) { + + /** + * The singleton object initiated upon process + * startup which manages all active Context instances, runs + * the render dispatch loop, and acts as a listener and dispatcher + * for events. All methods are therefore static. + * + * On static initialization, window.requestAnimationFrame is called with + * the event loop function. + * + * Note: Any window in which Engine runs will prevent default + * scrolling behavior on the 'touchmove' event. + * + * @static + * @class Engine + */ + var Context = require('./Context'); + var EventHandler = require('./EventHandler'); + var OptionsManager = require('./OptionsManager'); + + var Engine = {}; + + var contexts = []; + + var nextTickQueue = []; + + var currentFrame = 0; + var nextTickFrame = 0; + + var deferQueue = []; + + var lastTime = Date.now(); + var frameTime; + var frameTimeLimit; + var loopEnabled = true; + var eventForwarders = {}; + var eventHandler = new EventHandler(); + + var options = { + containerType: 'div', + containerClass: 'famous-container', + fpsCap: undefined, + runLoop: true, + appMode: true + }; + var optionsManager = new OptionsManager(options); + + /** @const */ + var MAX_DEFER_FRAME_TIME = 10; + + /** + * Inside requestAnimationFrame loop, step() is called, which: + * calculates current FPS (throttling loop if it is over limit set in setFPSCap), + * emits dataless 'prerender' event on start of loop, + * calls in order any one-shot functions registered by nextTick on last loop, + * calls Context.update on all Context objects registered, + * and emits dataless 'postrender' event on end of loop. + * + * @static + * @private + * @method step + */ + Engine.step = function step() { + currentFrame++; + nextTickFrame = currentFrame; + + var currentTime = Date.now(); + + // skip frame if we're over our framerate cap + if (frameTimeLimit && currentTime - lastTime < frameTimeLimit) return; + + var i = 0; + + frameTime = currentTime - lastTime; + lastTime = currentTime; + + eventHandler.emit('prerender'); + + // empty the queue + var numFunctions = nextTickQueue.length; + while (numFunctions--) (nextTickQueue.shift())(currentFrame); + + // limit total execution time for deferrable functions + while (deferQueue.length && (Date.now() - currentTime) < MAX_DEFER_FRAME_TIME) { + deferQueue.shift().call(this); + } + + for (i = 0; i < contexts.length; i++) contexts[i].update(); + + eventHandler.emit('postrender'); + }; + + // engage requestAnimationFrame + function loop() { + if (options.runLoop) { + Engine.step(); + window.requestAnimationFrame(loop); + } + else loopEnabled = false; + } + window.requestAnimationFrame(loop); + + // + // Upon main document window resize (unless on an "input" HTML element): + // scroll to the top left corner of the window, + // and for each managed Context: emit the 'resize' event and update its size. + // @param {Object=} event document event + // + function handleResize(event) { + for (var i = 0; i < contexts.length; i++) { + contexts[i].emit('resize'); + } + eventHandler.emit('resize'); + } + window.addEventListener('resize', handleResize, false); + handleResize(); + + /** + * Initialize famous for app mode + * + * @static + * @private + * @method initialize + */ + function initialize() { + // prevent scrolling via browser + window.addEventListener('touchmove', function(event) { + event.preventDefault(); + }, true); + + addRootClasses(); + } + var initialized = false; + + function addRootClasses() { + if (!document.body) { + Engine.nextTick(addRootClasses); + return; + } + + document.body.classList.add('famous-root'); + document.documentElement.classList.add('famous-root'); + } + + /** + * Add event handler object to set of downstream handlers. + * + * @method pipe + * + * @param {EventHandler} target event handler target object + * @return {EventHandler} passed event handler + */ + Engine.pipe = function pipe(target) { + if (target.subscribe instanceof Function) return target.subscribe(Engine); + else return eventHandler.pipe(target); + }; + + /** + * Remove handler object from set of downstream handlers. + * Undoes work of "pipe". + * + * @method unpipe + * + * @param {EventHandler} target target handler object + * @return {EventHandler} provided target + */ + Engine.unpipe = function unpipe(target) { + if (target.unsubscribe instanceof Function) return target.unsubscribe(Engine); + else return eventHandler.unpipe(target); + }; + + /** + * Bind a callback function to an event type handled by this object. + * + * @static + * @method "on" + * + * @param {string} type event type key (for example, 'click') + * @param {function(string, Object)} handler callback + * @return {EventHandler} this + */ + Engine.on = function on(type, handler) { + if (!(type in eventForwarders)) { + eventForwarders[type] = eventHandler.emit.bind(eventHandler, type); + + addEngineListener(type, eventForwarders[type]); + } + return eventHandler.on(type, handler); + }; + + function addEngineListener(type, forwarder) { + if (!document.body) { + Engine.nextTick(addEventListener.bind(this, type, forwarder)); + return; + } + + document.body.addEventListener(type, forwarder); + } + + /** + * Trigger an event, sending to all downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} event event data + * @return {EventHandler} this + */ + Engine.emit = function emit(type, event) { + return eventHandler.emit(type, event); + }; + + /** + * Unbind an event by type and handler. + * This undoes the work of "on". + * + * @static + * @method removeListener + * + * @param {string} type event type key (for example, 'click') + * @param {function} handler function object to remove + * @return {EventHandler} internal event handler object (for chaining) + */ + Engine.removeListener = function removeListener(type, handler) { + return eventHandler.removeListener(type, handler); + }; + + /** + * Return the current calculated frames per second of the Engine. + * + * @static + * @method getFPS + * + * @return {Number} calculated fps + */ + Engine.getFPS = function getFPS() { + return 1000 / frameTime; + }; + + /** + * Set the maximum fps at which the system should run. If internal render + * loop is called at a greater frequency than this FPSCap, Engine will + * throttle render and update until this rate is achieved. + * + * @static + * @method setFPSCap + * + * @param {Number} fps maximum frames per second + */ + Engine.setFPSCap = function setFPSCap(fps) { + frameTimeLimit = Math.floor(1000 / fps); + }; + + /** + * Return engine options. + * + * @static + * @method getOptions + * @param {string} key + * @return {Object} engine options + */ + Engine.getOptions = function getOptions(key) { + return optionsManager.getOptions(key); + }; + + /** + * Set engine options + * + * @static + * @method setOptions + * + * @param {Object} [options] overrides of default options + * @param {Number} [options.fpsCap] maximum fps at which the system should run + * @param {boolean} [options.runLoop=true] whether the run loop should continue + * @param {string} [options.containerType="div"] type of container element. Defaults to 'div'. + * @param {string} [options.containerClass="famous-container"] type of container element. Defaults to 'famous-container'. + */ + Engine.setOptions = function setOptions(options) { + return optionsManager.setOptions.apply(optionsManager, arguments); + }; + + /** + * Creates a new Context for rendering and event handling with + * provided document element as top of each tree. This will be tracked by the + * process-wide Engine. + * + * @static + * @method createContext + * + * @param {Node} el will be top of Famo.us document element tree + * @return {Context} new Context within el + */ + Engine.createContext = function createContext(el) { + if (!initialized && options.appMode) Engine.nextTick(initialize); + + var needMountContainer = false; + if (!el) { + el = document.createElement(options.containerType); + el.classList.add(options.containerClass); + needMountContainer = true; + } + + var context = new Context(el); + Engine.registerContext(context); + + if (needMountContainer) mount(context, el); + + return context; + }; + + function mount(context, el) { + if (!document.body) { + Engine.nextTick(mount.bind(this, context, el)); + return; + } + + document.body.appendChild(el); + context.emit('resize'); + } + + /** + * Registers an existing context to be updated within the run loop. + * + * @static + * @method registerContext + * + * @param {Context} context Context to register + * @return {FamousContext} provided context + */ + Engine.registerContext = function registerContext(context) { + contexts.push(context); + return context; + }; + + /** + * Returns a list of all contexts. + * + * @static + * @method getContexts + * @return {Array} contexts that are updated on each tick + */ + Engine.getContexts = function getContexts() { + return contexts; + }; + + /** + * Removes a context from the run loop. Note: this does not do any + * cleanup. + * + * @static + * @method deregisterContext + * + * @param {Context} context Context to deregister + */ + Engine.deregisterContext = function deregisterContext(context) { + var i = contexts.indexOf(context); + if (i >= 0) contexts.splice(i, 1); + }; + + /** + * Queue a function to be executed on the next tick of the + * Engine. + * + * @static + * @method nextTick + * + * @param {function(Object)} fn function accepting window object + */ + Engine.nextTick = function nextTick(fn) { + nextTickQueue.push(fn); + }; + + /** + * Queue a function to be executed sometime soon, at a time that is + * unlikely to affect frame rate. + * + * @static + * @method defer + * + * @param {Function} fn + */ + Engine.defer = function defer(fn) { + deferQueue.push(fn); + }; + + optionsManager.on('change', function(data) { + if (data.id === 'fpsCap') Engine.setFPSCap(data.value); + else if (data.id === 'runLoop') { + // kick off the loop only if it was stopped + if (!loopEnabled && data.value) { + loopEnabled = true; + window.requestAnimationFrame(loop); + } + } + }); + + module.exports = Engine; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/Surface',['require','exports','module','./ElementOutput'],function(require, exports, module) { + var ElementOutput = require('./ElementOutput'); + + /** + * A base class for viewable content and event + * targets inside a Famo.us application, containing a renderable document + * fragment. Like an HTML div, it can accept internal markup, + * properties, classes, and handle events. + * + * @class Surface + * @constructor + * + * @param {Object} [options] default option overrides + * @param {Array.Number} [options.size] [width, height] in pixels + * @param {Array.string} [options.classes] CSS classes to set on target div + * @param {Array} [options.properties] string dictionary of CSS properties to set on target div + * @param {Array} [options.attributes] string dictionary of HTML attributes to set on target div + * @param {string} [options.content] inner (HTML) content of surface + */ + function Surface(options) { + ElementOutput.call(this); + + this.options = {}; + + this.properties = {}; + this.attributes = {}; + this.content = ''; + this.classList = []; + this.size = null; + + this._classesDirty = true; + this._stylesDirty = true; + this._attributesDirty = true; + this._sizeDirty = true; + this._contentDirty = true; + this._trueSizeCheck = true; + + this._dirtyClasses = []; + + if (options) this.setOptions(options); + + this._currentTarget = null; + } + Surface.prototype = Object.create(ElementOutput.prototype); + Surface.prototype.constructor = Surface; + Surface.prototype.elementType = 'div'; + Surface.prototype.elementClass = 'famous-surface'; + + /** + * Set HTML attributes on this Surface. Note that this will cause + * dirtying and thus re-rendering, even if values do not change. + * + * @method setAttributes + * @param {Object} attributes property dictionary of "key" => "value" + */ + Surface.prototype.setAttributes = function setAttributes(attributes) { + for (var n in attributes) { + if (n === 'style') throw new Error('Cannot set styles via "setAttributes" as it will break Famo.us. Use "setProperties" instead.'); + this.attributes[n] = attributes[n]; + } + this._attributesDirty = true; + }; + + /** + * Get HTML attributes on this Surface. + * + * @method getAttributes + * + * @return {Object} Dictionary of this Surface's attributes. + */ + Surface.prototype.getAttributes = function getAttributes() { + return this.attributes; + }; + + /** + * Set CSS-style properties on this Surface. Note that this will cause + * dirtying and thus re-rendering, even if values do not change. + * + * @method setProperties + * @chainable + * @param {Object} properties property dictionary of "key" => "value" + */ + Surface.prototype.setProperties = function setProperties(properties) { + for (var n in properties) { + this.properties[n] = properties[n]; + } + this._stylesDirty = true; + return this; + }; + + /** + * Get CSS-style properties on this Surface. + * + * @method getProperties + * + * @return {Object} Dictionary of this Surface's properties. + */ + Surface.prototype.getProperties = function getProperties() { + return this.properties; + }; + + /** + * Add CSS-style class to the list of classes on this Surface. Note + * this will map directly to the HTML property of the actual + * corresponding rendered
. + * + * @method addClass + * @chainable + * @param {string} className name of class to add + */ + Surface.prototype.addClass = function addClass(className) { + if (this.classList.indexOf(className) < 0) { + this.classList.push(className); + this._classesDirty = true; + } + return this; + }; + + /** + * Remove CSS-style class from the list of classes on this Surface. + * Note this will map directly to the HTML property of the actual + * corresponding rendered
. + * + * @method removeClass + * @chainable + * @param {string} className name of class to remove + */ + Surface.prototype.removeClass = function removeClass(className) { + var i = this.classList.indexOf(className); + if (i >= 0) { + this._dirtyClasses.push(this.classList.splice(i, 1)[0]); + this._classesDirty = true; + } + return this; + }; + + /** + * Toggle CSS-style class from the list of classes on this Surface. + * Note this will map directly to the HTML property of the actual + * corresponding rendered
. + * + * @method toggleClass + * @param {string} className name of class to toggle + */ + Surface.prototype.toggleClass = function toggleClass(className) { + var i = this.classList.indexOf(className); + if (i >= 0) { + this.removeClass(className); + } else { + this.addClass(className); + } + return this; + }; + + /** + * Reset class list to provided dictionary. + * @method setClasses + * @chainable + * @param {Array.string} classList + */ + Surface.prototype.setClasses = function setClasses(classList) { + var i = 0; + var removal = []; + for (i = 0; i < this.classList.length; i++) { + if (classList.indexOf(this.classList[i]) < 0) removal.push(this.classList[i]); + } + for (i = 0; i < removal.length; i++) this.removeClass(removal[i]); + // duplicates are already checked by addClass() + for (i = 0; i < classList.length; i++) this.addClass(classList[i]); + return this; + }; + + /** + * Get array of CSS-style classes attached to this div. + * + * @method getClasslist + * @return {Array.string} array of class names + */ + Surface.prototype.getClassList = function getClassList() { + return this.classList; + }; + + /** + * Set or overwrite inner (HTML) content of this surface. Note that this + * causes a re-rendering if the content has changed. + * + * @method setContent + * @chainable + * @param {string|Document Fragment} content HTML content + */ + Surface.prototype.setContent = function setContent(content) { + if (this.content !== content) { + this.content = content; + this._contentDirty = true; + } + return this; + }; + + /** + * Return inner (HTML) content of this surface. + * + * @method getContent + * + * @return {string} inner (HTML) content + */ + Surface.prototype.getContent = function getContent() { + return this.content; + }; + + /** + * Set options for this surface + * + * @method setOptions + * @chainable + * @param {Object} [options] overrides for default options. See constructor. + */ + Surface.prototype.setOptions = function setOptions(options) { + if (options.size) this.setSize(options.size); + if (options.classes) this.setClasses(options.classes); + if (options.properties) this.setProperties(options.properties); + if (options.attributes) this.setAttributes(options.attributes); + if (options.content) this.setContent(options.content); + return this; + }; + + // Apply to document all changes from removeClass() since last setup(). + function _cleanupClasses(target) { + for (var i = 0; i < this._dirtyClasses.length; i++) target.classList.remove(this._dirtyClasses[i]); + this._dirtyClasses = []; + } + + // Apply values of all Famous-managed styles to the document element. + // These will be deployed to the document on call to #setup(). + function _applyStyles(target) { + for (var n in this.properties) { + target.style[n] = this.properties[n]; + } + } + + // Clear all Famous-managed styles from the document element. + // These will be deployed to the document on call to #setup(). + function _cleanupStyles(target) { + for (var n in this.properties) { + target.style[n] = ''; + } + } + + // Apply values of all Famous-managed attributes to the document element. + // These will be deployed to the document on call to #setup(). + function _applyAttributes(target) { + for (var n in this.attributes) { + target.setAttribute(n, this.attributes[n]); + } + } + + // Clear all Famous-managed attributes from the document element. + // These will be deployed to the document on call to #setup(). + function _cleanupAttributes(target) { + for (var n in this.attributes) { + target.removeAttribute(n); + } + } + + function _xyNotEquals(a, b) { + return (a && b) ? (a[0] !== b[0] || a[1] !== b[1]) : a !== b; + } + + /** + * One-time setup for an element to be ready for commits to document. + * + * @private + * @method setup + * + * @param {ElementAllocator} allocator document element pool for this context + */ + Surface.prototype.setup = function setup(allocator) { + var target = allocator.allocate(this.elementType); + if (this.elementClass) { + if (this.elementClass instanceof Array) { + for (var i = 0; i < this.elementClass.length; i++) { + target.classList.add(this.elementClass[i]); + } + } + else { + target.classList.add(this.elementClass); + } + } + target.style.display = ''; + this.attach(target); + this._opacity = null; + this._currentTarget = target; + this._stylesDirty = true; + this._classesDirty = true; + this._attributesDirty = true; + this._sizeDirty = true; + this._contentDirty = true; + this._originDirty = true; + this._transformDirty = true; + }; + + /** + * Apply changes from this component to the corresponding document element. + * This includes changes to classes, styles, size, content, opacity, origin, + * and matrix transforms. + * + * @private + * @method commit + * @param {Context} context commit context + */ + Surface.prototype.commit = function commit(context) { + if (!this._currentTarget) this.setup(context.allocator); + var target = this._currentTarget; + var size = context.size; + + if (this._classesDirty) { + _cleanupClasses.call(this, target); + var classList = this.getClassList(); + for (var i = 0; i < classList.length; i++) target.classList.add(classList[i]); + this._classesDirty = false; + this._trueSizeCheck = true; + } + + if (this._stylesDirty) { + _applyStyles.call(this, target); + this._stylesDirty = false; + this._trueSizeCheck = true; + } + + if (this._attributesDirty) { + _applyAttributes.call(this, target); + this._attributesDirty = false; + this._trueSizeCheck = true; + } + + if (this.size) { + var origSize = context.size; + size = [this.size[0], this.size[1]]; + if (size[0] === undefined) size[0] = origSize[0]; + if (size[1] === undefined) size[1] = origSize[1]; + if (size[0] === true || size[1] === true) { + if (size[0] === true){ + if (this._trueSizeCheck || (this._size[0] === 0)) { + var width = target.offsetWidth; + if (this._size && this._size[0] !== width) { + this._size[0] = width; + this._sizeDirty = true; + } + size[0] = width; + } else { + if (this._size) size[0] = this._size[0]; + } + } + if (size[1] === true){ + if (this._trueSizeCheck || (this._size[1] === 0)) { + var height = target.offsetHeight; + if (this._size && this._size[1] !== height) { + this._size[1] = height; + this._sizeDirty = true; + } + size[1] = height; + } else { + if (this._size) size[1] = this._size[1]; + } + } + this._trueSizeCheck = false; + } + } + + if (_xyNotEquals(this._size, size)) { + if (!this._size) this._size = [0, 0]; + this._size[0] = size[0]; + this._size[1] = size[1]; + + this._sizeDirty = true; + } + + if (this._sizeDirty) { + if (this._size) { + target.style.width = (this.size && this.size[0] === true) ? '' : this._size[0] + 'px'; + target.style.height = (this.size && this.size[1] === true) ? '' : this._size[1] + 'px'; + } + + this._eventOutput.emit('resize'); + } + + if (this._contentDirty) { + this.deploy(target); + this._eventOutput.emit('deploy'); + this._contentDirty = false; + this._trueSizeCheck = true; + } + + ElementOutput.prototype.commit.call(this, context); + }; + + /** + * Remove all Famous-relevant attributes from a document element. + * This is called by SurfaceManager's detach(). + * This is in some sense the reverse of .deploy(). + * + * @private + * @method cleanup + * @param {ElementAllocator} allocator + */ + Surface.prototype.cleanup = function cleanup(allocator) { + var i = 0; + var target = this._currentTarget; + this._eventOutput.emit('recall'); + this.recall(target); + target.style.display = 'none'; + target.style.opacity = ''; + target.style.width = ''; + target.style.height = ''; + _cleanupStyles.call(this, target); + _cleanupAttributes.call(this, target); + var classList = this.getClassList(); + _cleanupClasses.call(this, target); + for (i = 0; i < classList.length; i++) target.classList.remove(classList[i]); + if (this.elementClass) { + if (this.elementClass instanceof Array) { + for (i = 0; i < this.elementClass.length; i++) { + target.classList.remove(this.elementClass[i]); + } + } + else { + target.classList.remove(this.elementClass); + } + } + this.detach(target); + this._currentTarget = null; + allocator.deallocate(target); + }; + + /** + * Place the document element that this component manages into the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + Surface.prototype.deploy = function deploy(target) { + var content = this.getContent(); + if (content instanceof Node) { + while (target.hasChildNodes()) target.removeChild(target.firstChild); + target.appendChild(content); + } + else target.innerHTML = content; + }; + + /** + * Remove any contained document content associated with this surface + * from the actual document. + * + * @private + * @method recall + */ + Surface.prototype.recall = function recall(target) { + var df = document.createDocumentFragment(); + while (target.hasChildNodes()) df.appendChild(target.firstChild); + this.setContent(df); + }; + + /** + * Get the x and y dimensions of the surface. + * + * @method getSize + * @return {Array.Number} [x,y] size of surface + */ + Surface.prototype.getSize = function getSize() { + return this._size ? this._size : this.size; + }; + + /** + * Set x and y dimensions of the surface. + * + * @method setSize + * @chainable + * @param {Array.Number} size as [width, height] + */ + Surface.prototype.setSize = function setSize(size) { + this.size = size ? [size[0], size[1]] : null; + this._sizeDirty = true; + return this; + }; + + module.exports = Surface; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/Group',['require','exports','module','./Context','./Transform','./Surface'],function(require, exports, module) { + var Context = require('./Context'); + var Transform = require('./Transform'); + var Surface = require('./Surface'); + + /** + * A Context designed to contain surfaces and set properties + * to be applied to all of them at once. + * This is primarily used for specific performance improvements in the rendering engine. + * Private. + * + * @private + * @class Group + * @extends Surface + * @constructor + * @param {Object} [options] Surface options array (see Surface}) + */ + function Group(options) { + Surface.call(this, options); + this._shouldRecalculateSize = false; + this._container = document.createDocumentFragment(); + this.context = new Context(this._container); + this.setContent(this._container); + this._groupSize = [undefined, undefined]; + } + + /** @const */ + Group.SIZE_ZERO = [0, 0]; + + Group.prototype = Object.create(Surface.prototype); + Group.prototype.elementType = 'div'; + Group.prototype.elementClass = 'famous-group'; + + /** + * Add renderables to this component's render tree. + * + * @method add + * @private + * @param {Object} obj renderable object + * @return {RenderNode} Render wrapping provided object, if not already a RenderNode + */ + Group.prototype.add = function add() { + return this.context.add.apply(this.context, arguments); + }; + + /** + * Generate a render spec from the contents of this component. + * + * @private + * @method render + * @return {Number} Render spec for this component + */ + Group.prototype.render = function render() { + return Surface.prototype.render.call(this); + }; + + /** + * Place the document element this component manages into the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + Group.prototype.deploy = function deploy(target) { + this.context.migrate(target); + }; + + /** + * Remove this component and contained content from the document + * + * @private + * @method recall + * + * @param {Node} target node to which the component was deployed + */ + Group.prototype.recall = function recall(target) { + this._container = document.createDocumentFragment(); + this.context.migrate(this._container); + }; + + /** + * Apply changes from this component to the corresponding document element. + * + * @private + * @method commit + * + * @param {Object} context update spec passed in from above in the render tree. + */ + Group.prototype.commit = function commit(context) { + var transform = context.transform; + var origin = context.origin; + var opacity = context.opacity; + var size = context.size; + var result = Surface.prototype.commit.call(this, { + allocator: context.allocator, + transform: Transform.thenMove(transform, [-origin[0] * size[0], -origin[1] * size[1], 0]), + opacity: opacity, + origin: origin, + size: Group.SIZE_ZERO + }); + if (size[0] !== this._groupSize[0] || size[1] !== this._groupSize[1]) { + this._groupSize[0] = size[0]; + this._groupSize[1] = size[1]; + this.context.setSize(size); + } + this.context.update({ + transform: Transform.translate(-origin[0] * size[0], -origin[1] * size[1], 0), + origin: origin, + size: size + }); + return result; + }; + + module.exports = Group; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/transitions/TransitionableTransform',['require','exports','module','./Transitionable','../core/Transform','../utilities/Utility'],function(require, exports, module) { + var Transitionable = require('./Transitionable'); + var Transform = require('../core/Transform'); + var Utility = require('../utilities/Utility'); + + /** + * A class for transitioning the state of a Transform by transitioning + * its translate, scale, skew and rotate components independently. + * + * @class TransitionableTransform + * @constructor + * + * @param [transform=Transform.identity] {Transform} The initial transform state + */ + function TransitionableTransform(transform) { + this._final = Transform.identity.slice(); + + this._finalTranslate = [0, 0, 0]; + this._finalRotate = [0, 0, 0]; + this._finalSkew = [0, 0, 0]; + this._finalScale = [1, 1, 1]; + + this.translate = new Transitionable(this._finalTranslate); + this.rotate = new Transitionable(this._finalRotate); + this.skew = new Transitionable(this._finalSkew); + this.scale = new Transitionable(this._finalScale); + + if (transform) this.set(transform); + } + + function _build() { + return Transform.build({ + translate: this.translate.get(), + rotate: this.rotate.get(), + skew: this.skew.get(), + scale: this.scale.get() + }); + } + + function _buildFinal() { + return Transform.build({ + translate: this._finalTranslate, + rotate: this._finalRotate, + skew: this._finalSkew, + scale: this._finalScale + }); + } + + /** + * An optimized way of setting only the translation component of a Transform + * + * @method setTranslate + * @chainable + * + * @param translate {Array} New translation state + * @param [transition] {Object} Transition definition + * @param [callback] {Function} Callback + * @return {TransitionableTransform} + */ + TransitionableTransform.prototype.setTranslate = function setTranslate(translate, transition, callback) { + this._finalTranslate = translate; + this._final = _buildFinal.call(this); + this.translate.set(translate, transition, callback); + return this; + }; + + /** + * An optimized way of setting only the scale component of a Transform + * + * @method setScale + * @chainable + * + * @param scale {Array} New scale state + * @param [transition] {Object} Transition definition + * @param [callback] {Function} Callback + * @return {TransitionableTransform} + */ + TransitionableTransform.prototype.setScale = function setScale(scale, transition, callback) { + this._finalScale = scale; + this._final = _buildFinal.call(this); + this.scale.set(scale, transition, callback); + return this; + }; + + /** + * An optimized way of setting only the rotational component of a Transform + * + * @method setRotate + * @chainable + * + * @param eulerAngles {Array} Euler angles for new rotation state + * @param [transition] {Object} Transition definition + * @param [callback] {Function} Callback + * @return {TransitionableTransform} + */ + TransitionableTransform.prototype.setRotate = function setRotate(eulerAngles, transition, callback) { + this._finalRotate = eulerAngles; + this._final = _buildFinal.call(this); + this.rotate.set(eulerAngles, transition, callback); + return this; + }; + + /** + * An optimized way of setting only the skew component of a Transform + * + * @method setSkew + * @chainable + * + * @param skewAngles {Array} New skew state + * @param [transition] {Object} Transition definition + * @param [callback] {Function} Callback + * @return {TransitionableTransform} + */ + TransitionableTransform.prototype.setSkew = function setSkew(skewAngles, transition, callback) { + this._finalSkew = skewAngles; + this._final = _buildFinal.call(this); + this.skew.set(skewAngles, transition, callback); + return this; + }; + + /** + * Setter for a TransitionableTransform with optional parameters to transition + * between Transforms + * + * @method set + * @chainable + * + * @param transform {Array} New transform state + * @param [transition] {Object} Transition definition + * @param [callback] {Function} Callback + * @return {TransitionableTransform} + */ + TransitionableTransform.prototype.set = function set(transform, transition, callback) { + var components = Transform.interpret(transform); + + this._finalTranslate = components.translate; + this._finalRotate = components.rotate; + this._finalSkew = components.skew; + this._finalScale = components.scale; + this._final = transform; + + var _callback = callback ? Utility.after(4, callback) : null; + this.translate.set(components.translate, transition, _callback); + this.rotate.set(components.rotate, transition, _callback); + this.skew.set(components.skew, transition, _callback); + this.scale.set(components.scale, transition, _callback); + return this; + }; + + /** + * Sets the default transition to use for transitioning betwen Transform states + * + * @method setDefaultTransition + * + * @param transition {Object} Transition definition + */ + TransitionableTransform.prototype.setDefaultTransition = function setDefaultTransition(transition) { + this.translate.setDefault(transition); + this.rotate.setDefault(transition); + this.skew.setDefault(transition); + this.scale.setDefault(transition); + }; + + /** + * Getter. Returns the current state of the Transform + * + * @method get + * + * @return {Transform} + */ + TransitionableTransform.prototype.get = function get() { + if (this.isActive()) { + return _build.call(this); + } + else return this._final; + }; + + /** + * Get the destination state of the Transform + * + * @method getFinal + * + * @return Transform {Transform} + */ + TransitionableTransform.prototype.getFinal = function getFinal() { + return this._final; + }; + + /** + * Determine if the TransitionalTransform is currently transitioning + * + * @method isActive + * + * @return {Boolean} + */ + TransitionableTransform.prototype.isActive = function isActive() { + return this.translate.isActive() || this.rotate.isActive() || this.scale.isActive() || this.skew.isActive(); + }; + + /** + * Halts the transition + * + * @method halt + */ + TransitionableTransform.prototype.halt = function halt() { + this.translate.halt(); + this.rotate.halt(); + this.skew.halt(); + this.scale.halt(); + + this._final = this.get(); + this._finalTranslate = this.translate.get(); + this._finalRotate = this.rotate.get(); + this._finalSkew = this.skew.get(); + this._finalScale = this.scale.get(); + + return this; + }; + + module.exports = TransitionableTransform; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/Modifier',['require','exports','module','./Transform','../transitions/Transitionable','../transitions/TransitionableTransform'],function(require, exports, module) { + var Transform = require('./Transform'); + + /* TODO: remove these dependencies when deprecation complete */ + var Transitionable = require('../transitions/Transitionable'); + var TransitionableTransform = require('../transitions/TransitionableTransform'); + + /** + * + * A collection of visual changes to be + * applied to another renderable component. This collection includes a + * transform matrix, an opacity constant, a size, an origin specifier. + * Modifier objects can be added to any RenderNode or object + * capable of displaying renderables. The Modifier's children and descendants + * are transformed by the amounts specified in the Modifier's properties. + * + * @class Modifier + * @constructor + * @param {Object} [options] overrides of default options + * @param {Transform} [options.transform] affine transformation matrix + * @param {Number} [options.opacity] + * @param {Array.Number} [options.origin] origin adjustment + * @param {Array.Number} [options.size] size to apply to descendants + */ + function Modifier(options) { + this._transformGetter = null; + this._opacityGetter = null; + this._originGetter = null; + this._alignGetter = null; + this._sizeGetter = null; + this._proportionGetter = null; + + /* TODO: remove this when deprecation complete */ + this._legacyStates = {}; + + this._output = { + transform: Transform.identity, + opacity: 1, + origin: null, + align: null, + size: null, + proportions: null, + target: null + }; + + if (options) { + if (options.transform) this.transformFrom(options.transform); + if (options.opacity !== undefined) this.opacityFrom(options.opacity); + if (options.origin) this.originFrom(options.origin); + if (options.align) this.alignFrom(options.align); + if (options.size) this.sizeFrom(options.size); + if (options.proportions) this.proportionsFrom(options.proportions); + } + } + + /** + * Function, object, or static transform matrix which provides the transform. + * This is evaluated on every tick of the engine. + * + * @method transformFrom + * + * @param {Object} transform transform provider object + * @return {Modifier} this + */ + Modifier.prototype.transformFrom = function transformFrom(transform) { + if (transform instanceof Function) this._transformGetter = transform; + else if (transform instanceof Object && transform.get) this._transformGetter = transform.get.bind(transform); + else { + this._transformGetter = null; + this._output.transform = transform; + } + return this; + }; + + /** + * Set function, object, or number to provide opacity, in range [0,1]. + * + * @method opacityFrom + * + * @param {Object} opacity provider object + * @return {Modifier} this + */ + Modifier.prototype.opacityFrom = function opacityFrom(opacity) { + if (opacity instanceof Function) this._opacityGetter = opacity; + else if (opacity instanceof Object && opacity.get) this._opacityGetter = opacity.get.bind(opacity); + else { + this._opacityGetter = null; + this._output.opacity = opacity; + } + return this; + }; + + /** + * Set function, object, or numerical array to provide origin, as [x,y], + * where x and y are in the range [0,1]. + * + * @method originFrom + * + * @param {Object} origin provider object + * @return {Modifier} this + */ + Modifier.prototype.originFrom = function originFrom(origin) { + if (origin instanceof Function) this._originGetter = origin; + else if (origin instanceof Object && origin.get) this._originGetter = origin.get.bind(origin); + else { + this._originGetter = null; + this._output.origin = origin; + } + return this; + }; + + /** + * Set function, object, or numerical array to provide align, as [x,y], + * where x and y are in the range [0,1]. + * + * @method alignFrom + * + * @param {Object} align provider object + * @return {Modifier} this + */ + Modifier.prototype.alignFrom = function alignFrom(align) { + if (align instanceof Function) this._alignGetter = align; + else if (align instanceof Object && align.get) this._alignGetter = align.get.bind(align); + else { + this._alignGetter = null; + this._output.align = align; + } + return this; + }; + + /** + * Set function, object, or numerical array to provide size, as [width, height]. + * + * @method sizeFrom + * + * @param {Object} size provider object + * @return {Modifier} this + */ + Modifier.prototype.sizeFrom = function sizeFrom(size) { + if (size instanceof Function) this._sizeGetter = size; + else if (size instanceof Object && size.get) this._sizeGetter = size.get.bind(size); + else { + this._sizeGetter = null; + this._output.size = size; + } + return this; + }; + + /** + * Set function, object, or numerical array to provide proportions, as [percent of width, percent of height]. + * + * @method proportionsFrom + * + * @param {Object} proportions provider object + * @return {Modifier} this + */ + Modifier.prototype.proportionsFrom = function proportionsFrom(proportions) { + if (proportions instanceof Function) this._proportionGetter = proportions; + else if (proportions instanceof Object && proportions.get) this._proportionGetter = proportions.get.bind(proportions); + else { + this._proportionGetter = null; + this._output.proportions = proportions; + } + return this; + }; + + /** + * Deprecated: Prefer transformFrom with static Transform, or use a TransitionableTransform. + * @deprecated + * @method setTransform + * + * @param {Transform} transform Transform to transition to + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {Modifier} this + */ + Modifier.prototype.setTransform = function setTransform(transform, transition, callback) { + if (transition || this._legacyStates.transform) { + if (!this._legacyStates.transform) { + this._legacyStates.transform = new TransitionableTransform(this._output.transform); + } + if (!this._transformGetter) this.transformFrom(this._legacyStates.transform); + + this._legacyStates.transform.set(transform, transition, callback); + return this; + } + else return this.transformFrom(transform); + }; + + /** + * Deprecated: Prefer opacityFrom with static opacity array, or use a Transitionable with that opacity. + * @deprecated + * @method setOpacity + * + * @param {Number} opacity Opacity value to transition to. + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {Modifier} this + */ + Modifier.prototype.setOpacity = function setOpacity(opacity, transition, callback) { + if (transition || this._legacyStates.opacity) { + if (!this._legacyStates.opacity) { + this._legacyStates.opacity = new Transitionable(this._output.opacity); + } + if (!this._opacityGetter) this.opacityFrom(this._legacyStates.opacity); + + return this._legacyStates.opacity.set(opacity, transition, callback); + } + else return this.opacityFrom(opacity); + }; + + /** + * Deprecated: Prefer originFrom with static origin array, or use a Transitionable with that origin. + * @deprecated + * @method setOrigin + * + * @param {Array.Number} origin two element array with values between 0 and 1. + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {Modifier} this + */ + Modifier.prototype.setOrigin = function setOrigin(origin, transition, callback) { + /* TODO: remove this if statement when deprecation complete */ + if (transition || this._legacyStates.origin) { + + if (!this._legacyStates.origin) { + this._legacyStates.origin = new Transitionable(this._output.origin || [0, 0]); + } + if (!this._originGetter) this.originFrom(this._legacyStates.origin); + + this._legacyStates.origin.set(origin, transition, callback); + return this; + } + else return this.originFrom(origin); + }; + + /** + * Deprecated: Prefer alignFrom with static align array, or use a Transitionable with that align. + * @deprecated + * @method setAlign + * + * @param {Array.Number} align two element array with values between 0 and 1. + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {Modifier} this + */ + Modifier.prototype.setAlign = function setAlign(align, transition, callback) { + /* TODO: remove this if statement when deprecation complete */ + if (transition || this._legacyStates.align) { + + if (!this._legacyStates.align) { + this._legacyStates.align = new Transitionable(this._output.align || [0, 0]); + } + if (!this._alignGetter) this.alignFrom(this._legacyStates.align); + + this._legacyStates.align.set(align, transition, callback); + return this; + } + else return this.alignFrom(align); + }; + + /** + * Deprecated: Prefer sizeFrom with static origin array, or use a Transitionable with that size. + * @deprecated + * @method setSize + * @param {Array.Number} size two element array of [width, height] + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {Modifier} this + */ + Modifier.prototype.setSize = function setSize(size, transition, callback) { + if (size && (transition || this._legacyStates.size)) { + if (!this._legacyStates.size) { + this._legacyStates.size = new Transitionable(this._output.size || [0, 0]); + } + if (!this._sizeGetter) this.sizeFrom(this._legacyStates.size); + + this._legacyStates.size.set(size, transition, callback); + return this; + } + else return this.sizeFrom(size); + }; + + /** + * Deprecated: Prefer proportionsFrom with static origin array, or use a Transitionable with those proportions. + * @deprecated + * @method setProportions + * @param {Array.Number} proportions two element array of [percent of width, percent of height] + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {Modifier} this + */ + Modifier.prototype.setProportions = function setProportions(proportions, transition, callback) { + if (proportions && (transition || this._legacyStates.proportions)) { + if (!this._legacyStates.proportions) { + this._legacyStates.proportions = new Transitionable(this._output.proportions || [0, 0]); + } + if (!this._proportionGetter) this.proportionsFrom(this._legacyStates.proportions); + + this._legacyStates.proportions.set(proportions, transition, callback); + return this; + } + else return this.proportionsFrom(proportions); + }; + + /** + * Deprecated: Prefer to stop transform in your provider object. + * @deprecated + * @method halt + */ + Modifier.prototype.halt = function halt() { + if (this._legacyStates.transform) this._legacyStates.transform.halt(); + if (this._legacyStates.opacity) this._legacyStates.opacity.halt(); + if (this._legacyStates.origin) this._legacyStates.origin.halt(); + if (this._legacyStates.align) this._legacyStates.align.halt(); + if (this._legacyStates.size) this._legacyStates.size.halt(); + if (this._legacyStates.proportions) this._legacyStates.proportions.halt(); + this._transformGetter = null; + this._opacityGetter = null; + this._originGetter = null; + this._alignGetter = null; + this._sizeGetter = null; + this._proportionGetter = null; + }; + + /** + * Deprecated: Prefer to use your provided transform or output of your transform provider. + * @deprecated + * @method getTransform + * @return {Object} transform provider object + */ + Modifier.prototype.getTransform = function getTransform() { + return this._transformGetter(); + }; + + /** + * Deprecated: Prefer to determine the end state of your transform from your transform provider + * @deprecated + * @method getFinalTransform + * @return {Transform} transform matrix + */ + Modifier.prototype.getFinalTransform = function getFinalTransform() { + return this._legacyStates.transform ? this._legacyStates.transform.getFinal() : this._output.transform; + }; + + /** + * Deprecated: Prefer to use your provided opacity or output of your opacity provider. + * @deprecated + * @method getOpacity + * @return {Object} opacity provider object + */ + Modifier.prototype.getOpacity = function getOpacity() { + return this._opacityGetter(); + }; + + /** + * Deprecated: Prefer to use your provided origin or output of your origin provider. + * @deprecated + * @method getOrigin + * @return {Object} origin provider object + */ + Modifier.prototype.getOrigin = function getOrigin() { + return this._originGetter(); + }; + + /** + * Deprecated: Prefer to use your provided align or output of your align provider. + * @deprecated + * @method getAlign + * @return {Object} align provider object + */ + Modifier.prototype.getAlign = function getAlign() { + return this._alignGetter(); + }; + + /** + * Deprecated: Prefer to use your provided size or output of your size provider. + * @deprecated + * @method getSize + * @return {Object} size provider object + */ + Modifier.prototype.getSize = function getSize() { + return this._sizeGetter ? this._sizeGetter() : this._output.size; + }; + + /** + * Deprecated: Prefer to use your provided proportions or output of your proportions provider. + * @deprecated + * @method getProportions + * @return {Object} proportions provider object + */ + Modifier.prototype.getProportions = function getProportions() { + return this._proportionGetter ? this._proportionGetter() : this._output.proportions; + }; + + // call providers on tick to receive render spec elements to apply + function _update() { + if (this._transformGetter) this._output.transform = this._transformGetter(); + if (this._opacityGetter) this._output.opacity = this._opacityGetter(); + if (this._originGetter) this._output.origin = this._originGetter(); + if (this._alignGetter) this._output.align = this._alignGetter(); + if (this._sizeGetter) this._output.size = this._sizeGetter(); + if (this._proportionGetter) this._output.proportions = this._proportionGetter(); + } + + /** + * Return render spec for this Modifier, applying to the provided + * target component. This is similar to render() for Surfaces. + * + * @private + * @method modify + * + * @param {Object} target (already rendered) render spec to + * which to apply the transform. + * @return {Object} render spec for this Modifier, including the + * provided target + */ + Modifier.prototype.modify = function modify(target) { + _update.call(this); + this._output.target = target; + return this._output; + }; + + module.exports = Modifier; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/Scene',['require','exports','module','./Transform','./Modifier','./RenderNode'],function(require, exports, module) { + var Transform = require('./Transform'); + var Modifier = require('./Modifier'); + var RenderNode = require('./RenderNode'); + + /** + * Builds and renders a scene graph based on a declarative structure definition. + * See the Scene examples in the examples distribution (http://github.com/Famous/examples.git). + * + * @class Scene + * @constructor + * @param {Object|Array|Spec} definition in the format of a render spec. + */ + function Scene(definition) { + this.id = null; + this._objects = null; + + this.node = new RenderNode(); + this._definition = null; + + if (definition) this.load(definition); + } + + var _MATRIX_GENERATORS = { + 'translate': Transform.translate, + 'rotate': Transform.rotate, + 'rotateX': Transform.rotateX, + 'rotateY': Transform.rotateY, + 'rotateZ': Transform.rotateZ, + 'rotateAxis': Transform.rotateAxis, + 'scale': Transform.scale, + 'skew': Transform.skew, + 'matrix3d': function() { + return arguments; + } + }; + + /** + * Clone this scene + * + * @method create + * @return {Scene} deep copy of this scene + */ + Scene.prototype.create = function create() { + return new Scene(this._definition); + }; + + function _resolveTransformMatrix(matrixDefinition) { + for (var type in _MATRIX_GENERATORS) { + if (type in matrixDefinition) { + var args = matrixDefinition[type]; + if (!(args instanceof Array)) args = [args]; + return _MATRIX_GENERATORS[type].apply(this, args); + } + } + } + + // parse transform into tree of render nodes, doing matrix multiplication + // when available + function _parseTransform(definition) { + var transformDefinition = definition.transform; + var opacity = definition.opacity; + var origin = definition.origin; + var align = definition.align; + var size = definition.size; + var transform = Transform.identity; + if (transformDefinition instanceof Array) { + if (transformDefinition.length === 16 && typeof transformDefinition[0] === 'number') { + transform = transformDefinition; + } + else { + for (var i = 0; i < transformDefinition.length; i++) { + transform = Transform.multiply(transform, _resolveTransformMatrix(transformDefinition[i])); + } + } + } + else if (transformDefinition instanceof Function) { + transform = transformDefinition; + } + else if (transformDefinition instanceof Object) { + transform = _resolveTransformMatrix(transformDefinition); + } + + var result = new Modifier({ + transform: transform, + opacity: opacity, + origin: origin, + align: align, + size: size + }); + return result; + } + + function _parseArray(definition) { + var result = new RenderNode(); + for (var i = 0; i < definition.length; i++) { + var obj = _parse.call(this, definition[i]); + if (obj) result.add(obj); + } + return result; + } + + // parse object directly into tree of RenderNodes + function _parse(definition) { + var result; + var id; + if (definition instanceof Array) { + result = _parseArray.call(this, definition); + } + else { + id = this._objects.length; + if (definition.render && (definition.render instanceof Function)) { + result = definition; + } + else if (definition.target) { + var targetObj = _parse.call(this, definition.target); + var obj = _parseTransform.call(this, definition); + + result = new RenderNode(obj); + result.add(targetObj); + if (definition.id) this.id[definition.id] = obj; + } + else if (definition.id) { + result = new RenderNode(); + this.id[definition.id] = result; + } + } + this._objects[id] = result; + return result; + } + + /** + * Builds and renders a scene graph based on a canonical declarative scene definition. + * See examples/Scene/example.js. + * + * @method load + * @param {Object} definition definition in the format of a render spec. + */ + Scene.prototype.load = function load(definition) { + this._definition = definition; + this.id = {}; + this._objects = []; + this.node.set(_parse.call(this, definition)); + }; + + /** + * Add renderables to this component's render tree + * + * @method add + * + * @param {Object} obj renderable object + * @return {RenderNode} Render wrapping provided object, if not already a RenderNode + */ + Scene.prototype.add = function add() { + return this.node.add.apply(this.node, arguments); + }; + + /** + * Generate a render spec from the contents of this component. + * + * @private + * @method render + * @return {number} Render spec for this component + */ + Scene.prototype.render = function render() { + return this.node.render.apply(this.node, arguments); + }; + + module.exports = Scene; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/View',['require','exports','module','./EventHandler','./OptionsManager','./RenderNode','../utilities/Utility'],function(require, exports, module) { + var EventHandler = require('./EventHandler'); + var OptionsManager = require('./OptionsManager'); + var RenderNode = require('./RenderNode'); + var Utility = require('../utilities/Utility'); + + /** + * Useful for quickly creating elements within applications + * with large event systems. Consists of a RenderNode paired with + * an input EventHandler and an output EventHandler. + * Meant to be extended by the developer. + * + * @class View + * @uses EventHandler + * @uses OptionsManager + * @uses RenderNode + * @constructor + */ + function View(options) { + this._node = new RenderNode(); + + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + + this.options = Utility.clone(this.constructor.DEFAULT_OPTIONS || View.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + + if (options) this.setOptions(options); + } + + View.DEFAULT_OPTIONS = {}; // no defaults + + /** + * Look up options value by key + * @method getOptions + * + * @param {string} key key + * @return {Object} associated object + */ + View.prototype.getOptions = function getOptions(key) { + return this._optionsManager.getOptions(key); + }; + + /* + * Set internal options. + * No defaults options are set in View. + * + * @method setOptions + * @param {Object} options + */ + View.prototype.setOptions = function setOptions(options) { + this._optionsManager.patch(options); + }; + + /** + * Add a child renderable to the view. + * Note: This is meant to be used by an inheriting class + * rather than from outside the prototype chain. + * + * @method add + * @return {RenderNode} + * @protected + */ + View.prototype.add = function add() { + return this._node.add.apply(this._node, arguments); + }; + + /** + * Alias for add + * @method _add + */ + View.prototype._add = View.prototype.add; + + /** + * Generate a render spec from the contents of this component. + * + * @private + * @method render + * @return {number} Render spec for this component + */ + View.prototype.render = function render() { + return this._node.render(); + }; + + /** + * Return size of contained element. + * + * @method getSize + * @return {Array.Number} [width, height] + */ + View.prototype.getSize = function getSize() { + if (this._node && this._node.getSize) { + return this._node.getSize.apply(this._node, arguments) || this.options.size; + } + else return this.options.size; + }; + + module.exports = View; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/core/ViewSequence',['require','exports','module'],function(require, exports, module) { + + /** + * Helper object used to iterate through items sequentially. Used in + * views that deal with layout. A ViewSequence object conceptually points + * to a node in a linked list. + * + * @class ViewSequence + * + * @constructor + * @param {Object|Array} options Options object, or content array. + * @param {Number} [options.index] starting index. + * @param {Number} [options.array] Array of elements to populate the ViewSequence + * @param {Object} [options._] Optional backing store (internal + * @param {Boolean} [options.loop] Whether to wrap when accessing elements just past the end + * (or beginning) of the sequence. + */ + function ViewSequence(options) { + if (!options) options = []; + if (options instanceof Array) options = {array: options}; + + this._ = null; + this.index = options.index || 0; + + if (options.array) this._ = new (this.constructor.Backing)(options.array); + else if (options._) this._ = options._; + + if (this.index === this._.firstIndex) this._.firstNode = this; + if (this.index === this._.firstIndex + this._.array.length - 1) this._.lastNode = this; + + if (options.loop !== undefined) this._.loop = options.loop; + + if (options.trackSize !== undefined) this._.trackSize = options.trackSize; + + this._previousNode = null; + this._nextNode = null; + } + + // constructor for internal storage + ViewSequence.Backing = function Backing(array) { + this.array = array; + this.firstIndex = 0; + this.loop = false; + this.firstNode = null; + this.lastNode = null; + this.cumulativeSizes = [[0, 0]]; + this.sizeDirty = true; + this.trackSize = false; + }; + + // Get value "i" slots away from the first index. + ViewSequence.Backing.prototype.getValue = function getValue(i) { + var _i = i - this.firstIndex; + if (_i < 0 || _i >= this.array.length) return null; + return this.array[_i]; + }; + + // Set value "i" slots away from the first index. + ViewSequence.Backing.prototype.setValue = function setValue(i, value) { + this.array[i - this.firstIndex] = value; + }; + + // Get sequence size from backing up to index + // TODO: remove from viewSequence with proper abstraction + ViewSequence.Backing.prototype.getSize = function getSize(index) { + return this.cumulativeSizes[index]; + }; + + // Calculates cumulative size + // TODO: remove from viewSequence with proper abstraction + ViewSequence.Backing.prototype.calculateSize = function calculateSize(index) { + index = index || this.array.length; + var size = [0, 0]; + for (var i = 0; i < index; i++) { + var nodeSize = this.array[i].getSize(); + if (!nodeSize) return undefined; + if (size[0] !== undefined) { + if (nodeSize[0] === undefined) size[0] = undefined; + else size[0] += nodeSize[0]; + } + if (size[1] !== undefined) { + if (nodeSize[1] === undefined) size[1] = undefined; + else size[1] += nodeSize[1]; + } + this.cumulativeSizes[i + 1] = size.slice(); + } + this.sizeDirty = false; + return size; + }; + + // After splicing into the backing store, restore the indexes of each node correctly. + ViewSequence.Backing.prototype.reindex = function reindex(start, removeCount, insertCount) { + if (!this.array[0]) return; + + var i = 0; + var index = this.firstIndex; + var indexShiftAmount = insertCount - removeCount; + var node = this.firstNode; + + // find node to begin + while (index < start - 1) { + node = node.getNext(); + index++; + } + // skip removed nodes + var spliceStartNode = node; + for (i = 0; i < removeCount; i++) { + node = node.getNext(); + if (node) node._previousNode = spliceStartNode; + } + var spliceResumeNode = node ? node.getNext() : null; + // generate nodes for inserted items + spliceStartNode._nextNode = null; + node = spliceStartNode; + for (i = 0; i < insertCount; i++) node = node.getNext(); + index += insertCount; + // resume the chain + if (node !== spliceResumeNode) { + node._nextNode = spliceResumeNode; + if (spliceResumeNode) spliceResumeNode._previousNode = node; + } + if (spliceResumeNode) { + node = spliceResumeNode; + index++; + while (node && index < this.array.length + this.firstIndex) { + if (node._nextNode) node.index += indexShiftAmount; + else node.index = index; + node = node.getNext(); + index++; + } + } + if (this.trackSize) this.sizeDirty = true; + }; + + /** + * Return ViewSequence node previous to this node in the list, respecting looping if applied. + * + * @method getPrevious + * @return {ViewSequence} previous node. + */ + ViewSequence.prototype.getPrevious = function getPrevious() { + var len = this._.array.length; + if (!len) { + this._previousNode = null; + } + else if (this.index === this._.firstIndex) { + if (this._.loop) { + this._previousNode = this._.lastNode || new (this.constructor)({_: this._, index: this._.firstIndex + len - 1}); + this._previousNode._nextNode = this; + } + else { + this._previousNode = null; + } + } + else if (!this._previousNode) { + this._previousNode = new (this.constructor)({_: this._, index: this.index - 1}); + this._previousNode._nextNode = this; + } + return this._previousNode; + }; + + /** + * Return ViewSequence node next after this node in the list, respecting looping if applied. + * + * @method getNext + * @return {ViewSequence} previous node. + */ + ViewSequence.prototype.getNext = function getNext() { + var len = this._.array.length; + if (!len) { + this._nextNode = null; + } + else if (this.index === this._.firstIndex + len - 1) { + if (this._.loop) { + this._nextNode = this._.firstNode || new (this.constructor)({_: this._, index: this._.firstIndex}); + this._nextNode._previousNode = this; + } + else { + this._nextNode = null; + } + } + else if (!this._nextNode) { + this._nextNode = new (this.constructor)({_: this._, index: this.index + 1}); + this._nextNode._previousNode = this; + } + return this._nextNode; + }; + + /** + * Return index of the provided item in the backing array + * + * @method indexOf + * @return {Number} index or -1 if not found + */ + ViewSequence.prototype.indexOf = function indexOf(item) { + return this._.array.indexOf(item); + }; + + /** + * Return index of this ViewSequence node. + * + * @method getIndex + * @return {Number} index + */ + ViewSequence.prototype.getIndex = function getIndex() { + return this.index; + }; + + /** + * Return printable version of this ViewSequence node. + * + * @method toString + * @return {string} this index as a string + */ + ViewSequence.prototype.toString = function toString() { + return '' + this.index; + }; + + /** + * Add one or more objects to the beginning of the sequence. + * + * @method unshift + * @param {...Object} value arguments array of objects + */ + ViewSequence.prototype.unshift = function unshift(value) { + this._.array.unshift.apply(this._.array, arguments); + this._.firstIndex -= arguments.length; + if (this._.trackSize) this._.sizeDirty = true; + }; + + /** + * Add one or more objects to the end of the sequence. + * + * @method push + * @param {...Object} value arguments array of objects + */ + ViewSequence.prototype.push = function push(value) { + this._.array.push.apply(this._.array, arguments); + if (this._.trackSize) this._.sizeDirty = true; + }; + + /** + * Remove objects from the sequence + * + * @method splice + * @param {Number} index starting index for removal + * @param {Number} howMany how many elements to remove + * @param {...Object} value arguments array of objects + */ + ViewSequence.prototype.splice = function splice(index, howMany) { + var values = Array.prototype.slice.call(arguments, 2); + this._.array.splice.apply(this._.array, [index - this._.firstIndex, howMany].concat(values)); + this._.reindex(index, howMany, values.length); + }; + + /** + * Exchange this element's sequence position with another's. + * + * @method swap + * @param {ViewSequence} other element to swap with. + */ + ViewSequence.prototype.swap = function swap(other) { + var otherValue = other.get(); + var myValue = this.get(); + this._.setValue(this.index, otherValue); + this._.setValue(other.index, myValue); + + var myPrevious = this._previousNode; + var myNext = this._nextNode; + var myIndex = this.index; + var otherPrevious = other._previousNode; + var otherNext = other._nextNode; + var otherIndex = other.index; + + this.index = otherIndex; + this._previousNode = (otherPrevious === this) ? other : otherPrevious; + if (this._previousNode) this._previousNode._nextNode = this; + this._nextNode = (otherNext === this) ? other : otherNext; + if (this._nextNode) this._nextNode._previousNode = this; + + other.index = myIndex; + other._previousNode = (myPrevious === other) ? this : myPrevious; + if (other._previousNode) other._previousNode._nextNode = other; + other._nextNode = (myNext === other) ? this : myNext; + if (other._nextNode) other._nextNode._previousNode = other; + + if (this.index === this._.firstIndex) this._.firstNode = this; + else if (this.index === this._.firstIndex + this._.array.length - 1) this._.lastNode = this; + if (other.index === this._.firstIndex) this._.firstNode = other; + else if (other.index === this._.firstIndex + this._.array.length - 1) this._.lastNode = other; + if (this._.trackSize) this._.sizeDirty = true; + }; + + /** + * Return value of this ViewSequence node. + * + * @method get + * @return {Object} value of thiss + */ + ViewSequence.prototype.get = function get() { + return this._.getValue(this.index); + }; + + /** + * Call getSize() on the contained View. + * + * @method getSize + * @return {Array.Number} [width, height] + */ + ViewSequence.prototype.getSize = function getSize() { + var target = this.get(); + return target ? target.getSize() : null; + }; + + /** + * Generate a render spec from the contents of this component. + * Specifically, this will render the value at the current index. + * @private + * @method render + * @return {number} Render spec for this component + */ + ViewSequence.prototype.render = function render() { + if (this._.trackSize && this._.sizeDirty) this._.calculateSize(); + var target = this.get(); + return target ? target.render.apply(target, arguments) : null; + }; + + module.exports = ViewSequence; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/events/EventArbiter',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + + /** + * A switch which wraps several event destinations and + * redirects received events to at most one of them. + * Setting the 'mode' of the object dictates which one + * of these destinations will receive events. + * + * @class EventArbiter + * @constructor + * + * @param {Number | string} startMode initial setting of switch, + */ + function EventArbiter(startMode) { + this.dispatchers = {}; + this.currMode = undefined; + this.setMode(startMode); + } + + /** + * Set switch to this mode, passing events to the corresponding + * EventHandler. If mode has changed, emits 'change' event, + * emits 'unpipe' event to the old mode's handler, and emits 'pipe' + * event to the new mode's handler. + * + * @method setMode + * + * @param {string | number} mode indicating which event handler to send to. + */ + EventArbiter.prototype.setMode = function setMode(mode) { + if (mode !== this.currMode) { + var startMode = this.currMode; + + if (this.dispatchers[this.currMode]) this.dispatchers[this.currMode].trigger('unpipe'); + this.currMode = mode; + if (this.dispatchers[mode]) this.dispatchers[mode].emit('pipe'); + this.emit('change', {from: startMode, to: mode}); + } + }; + + /** + * Return the existing EventHandler corresponding to this + * mode, creating one if it doesn't exist. + * + * @method forMode + * + * @param {string | number} mode mode to which this eventHandler corresponds + * + * @return {EventHandler} eventHandler corresponding to this mode + */ + EventArbiter.prototype.forMode = function forMode(mode) { + if (!this.dispatchers[mode]) this.dispatchers[mode] = new EventHandler(); + return this.dispatchers[mode]; + }; + + /** + * Trigger an event, sending to currently selected handler, if + * it is listening for provided 'type' key. + * + * @method emit + * + * @param {string} eventType event type key (for example, 'click') + * @param {Object} event event data + * @return {EventHandler} this + */ + EventArbiter.prototype.emit = function emit(eventType, event) { + if (this.currMode === undefined) return false; + if (!event) event = {}; + var dispatcher = this.dispatchers[this.currMode]; + if (dispatcher) return dispatcher.trigger(eventType, event); + }; + + module.exports = EventArbiter; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/events/EventFilter',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + + /** + * EventFilter regulates the broadcasting of events based on + * a specified condition function of standard event type: function(type, data). + * + * @class EventFilter + * @constructor + * + * @param {function} condition function to determine whether or not + * events are emitted. + */ + function EventFilter(condition) { + EventHandler.call(this); + this._condition = condition; + } + EventFilter.prototype = Object.create(EventHandler.prototype); + EventFilter.prototype.constructor = EventFilter; + + /** + * If filter condition is met, trigger an event, sending to all downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} data event data + * @return {EventHandler} this + */ + EventFilter.prototype.emit = function emit(type, data) { + if (this._condition(type, data)) + return EventHandler.prototype.emit.apply(this, arguments); + }; + + /** + * An alias of emit. Trigger determines whether to send + * events based on the return value of it's condition function + * when passed the event type and associated data. + * + * @method trigger + * @param {string} type name of the event + * @param {object} data associated data + */ + EventFilter.prototype.trigger = EventFilter.prototype.emit; + + module.exports = EventFilter; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/events/EventMapper',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + + /** + * EventMapper routes events to various event destinations + * based on custom logic. The function signature is arbitrary. + * + * @class EventMapper + * @constructor + * + * @param {function} mappingFunction function to determine where + * events are routed to. + */ + function EventMapper(mappingFunction) { + EventHandler.call(this); + this._mappingFunction = mappingFunction; + } + EventMapper.prototype = Object.create(EventHandler.prototype); + EventMapper.prototype.constructor = EventMapper; + + EventMapper.prototype.subscribe = null; + EventMapper.prototype.unsubscribe = null; + + /** + * Trigger an event, sending to all mapped downstream handlers + * listening for provided 'type' key. + * + * @method emit + * + * @param {string} type event type key (for example, 'click') + * @param {Object} data event data + * @return {EventHandler} this + */ + EventMapper.prototype.emit = function emit(type, data) { + var target = this._mappingFunction.apply(this, arguments); + if (target && (target.emit instanceof Function)) target.emit(type, data); + }; + + /** + * Alias of emit. + * @method trigger + */ + EventMapper.prototype.trigger = EventMapper.prototype.emit; + + module.exports = EventMapper; +}); + +define('famous/inputs/Accumulator',['require','exports','module','../core/EventHandler','../transitions/Transitionable'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + var Transitionable = require('../transitions/Transitionable'); + + /** + * Accumulates differentials of event sources that emit a `delta` + * attribute taking a Number or Array of Number types. The accumulated + * value is stored in a getter/setter. + * + * @class Accumulator + * @constructor + * @param value {Number|Array|Transitionable} Initializing value + * @param [eventName='update'] {String} Name of update event + */ + function Accumulator(value, eventName) { + if (eventName === undefined) eventName = 'update'; + + this._state = (value && value.get && value.set) + ? value + : new Transitionable(value || 0); + + this._eventInput = new EventHandler(); + EventHandler.setInputHandler(this, this._eventInput); + + this._eventInput.on(eventName, _handleUpdate.bind(this)); + } + + function _handleUpdate(data) { + var delta = data.delta; + var state = this.get(); + + if (delta.constructor === state.constructor){ + var newState = (delta instanceof Array) + ? [state[0] + delta[0], state[1] + delta[1]] + : state + delta; + this.set(newState); + } + } + + /** + * Basic getter + * + * @method get + * @return {Number|Array} current value + */ + Accumulator.prototype.get = function get() { + return this._state.get(); + }; + + /** + * Basic setter + * + * @method set + * @param value {Number|Array} new value + */ + Accumulator.prototype.set = function set(value) { + this._state.set(value); + }; + + module.exports = Accumulator; +}); + +define('famous/inputs/DesktopEmulationMode',['require','exports','module'],function(require, exports, module) { + var hasTouch = 'ontouchstart' in window; + + function kill(type) { + window.addEventListener(type, function(event) { + event.stopPropagation(); + return false; + }, true); + } + + if (hasTouch) { + kill('mousedown'); + kill('mousemove'); + kill('mouseup'); + kill('mouseleave'); + } +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/inputs/FastClick',['require','exports','module'],function(require, exports, module) { + /** + * FastClick is an override shim which maps event pairs of + * 'touchstart' and 'touchend' which differ by less than a certain + * threshold to the 'click' event. + * This is used to speed up clicks on some browsers. + */ + (function() { + if (!window.CustomEvent) return; + var clickThreshold = 300; + var clickWindow = 500; + var potentialClicks = {}; + var recentlyDispatched = {}; + var _now = Date.now; + + window.addEventListener('touchstart', function(event) { + var timestamp = _now(); + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + potentialClicks[touch.identifier] = timestamp; + } + }); + + window.addEventListener('touchmove', function(event) { + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + delete potentialClicks[touch.identifier]; + } + }); + + window.addEventListener('touchend', function(event) { + var currTime = _now(); + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var startTime = potentialClicks[touch.identifier]; + if (startTime && currTime - startTime < clickThreshold) { + var clickEvt = new window.CustomEvent('click', { + 'bubbles': true, + 'detail': touch + }); + recentlyDispatched[currTime] = event; + event.target.dispatchEvent(clickEvt); + } + delete potentialClicks[touch.identifier]; + } + }); + + window.addEventListener('click', function(event) { + var currTime = _now(); + for (var i in recentlyDispatched) { + var previousEvent = recentlyDispatched[i]; + if (currTime - i < clickWindow) { + if (event instanceof window.MouseEvent && event.target === previousEvent.target) event.stopPropagation(); + } + else delete recentlyDispatched[i]; + } + }, true); + })(); +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/GenericSync',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + + var EventHandler = require('../core/EventHandler'); + + /** + * Combines multiple types of sync classes (e.g. mouse, touch, + * scrolling) into one standardized interface for inclusion in widgets. + * + * Sync classes are first registered with a key, and then can be accessed + * globally by key. + * + * Emits 'start', 'update' and 'end' events as a union of the sync class + * providers. + * + * @class GenericSync + * @constructor + * @param syncs {Object|Array} object with fields {sync key : sync options} + * or an array of registered sync keys + * @param [options] {Object|Array} options object to set on all syncs + */ + function GenericSync(syncs, options) { + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + + this._syncs = {}; + if (syncs) this.addSync(syncs); + if (options) this.setOptions(options); + } + + GenericSync.DIRECTION_X = 0; + GenericSync.DIRECTION_Y = 1; + GenericSync.DIRECTION_Z = 2; + + // Global registry of sync classes. Append only. + var registry = {}; + + /** + * Register a global sync class with an identifying key + * + * @static + * @method register + * + * @param syncObject {Object} an object of {sync key : sync options} fields + */ + GenericSync.register = function register(syncObject) { + for (var key in syncObject){ + if (registry[key]){ // skip redundant registration + if (registry[key] !== syncObject[key]) // only if same registered class + throw new Error('Conflicting sync classes for key: ' + key); + } + else registry[key] = syncObject[key]; + } + }; + + /** + * Helper to set options on all sync instances + * + * @method setOptions + * @param options {Object} options object + */ + GenericSync.prototype.setOptions = function(options) { + for (var key in this._syncs){ + this._syncs[key].setOptions(options); + } + }; + + /** + * Pipe events to a sync class + * + * @method pipeSync + * @param key {String} identifier for sync class + */ + GenericSync.prototype.pipeSync = function pipeToSync(key) { + var sync = this._syncs[key]; + this._eventInput.pipe(sync); + sync.pipe(this._eventOutput); + }; + + /** + * Unpipe events from a sync class + * + * @method unpipeSync + * @param key {String} identifier for sync class + */ + GenericSync.prototype.unpipeSync = function unpipeFromSync(key) { + var sync = this._syncs[key]; + this._eventInput.unpipe(sync); + sync.unpipe(this._eventOutput); + }; + + function _addSingleSync(key, options) { + if (!registry[key]) return; + this._syncs[key] = new (registry[key])(options); + this.pipeSync(key); + } + + /** + * Add a sync class to from the registered classes + * + * @method addSync + * @param syncs {Object|Array.String} an array of registered sync keys + * or an object with fields {sync key : sync options} + */ + GenericSync.prototype.addSync = function addSync(syncs) { + if (syncs instanceof Array) + for (var i = 0; i < syncs.length; i++) + _addSingleSync.call(this, syncs[i]); + else if (syncs instanceof Object) + for (var key in syncs) + _addSingleSync.call(this, key, syncs[key]); + }; + + module.exports = GenericSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/MouseSync',['require','exports','module','../core/EventHandler','../core/OptionsManager'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Handles piped in mouse drag events. Outputs an object with the position delta from last frame, position from start, + * current velocity averaged out over the velocitySampleLength (set via options), clientX, clientY, offsetX, and offsetY. + * + * Emits 'start', 'update' and 'end' events. Designed to be used either as a standalone MouseSync, or as part of a + * GenericSync. + * + * @class MouseSync + * @constructor + * + * @example + * var Surface = require('../core/Surface'); + * var MouseSync = require('../inputs/MouseSync'); + * + * var surface = new Surface({ size: [100, 100] }); + * var mouseSync = new MouseSync(); + * surface.pipe(mouseSync); + * + * mouseSync.on('start', function (e) { // react to start }); + * mouseSync.on('update', function (e) { // react to update }); + * mouseSync.on('end', function (e) { // react to end }); + * + * @param [options] {Object} An object of the following configurable options. + * @param [options.clickThreshold] {Number} Absolute distance from click origin that will still trigger a click. + * @param [options.direction] {Number} Read from a particular axis. Valid options are: undefined, 0 or 1. 0 corresponds to x, and 1 to y. Default is undefined, which allows both x and y. + * @param [options.rails] {Boolean} Read from axis with the greatest differential. + * @param [options.velocitySampleLength] {Number} Number of previous frames to check velocity against. + * @param [options.propogate] {Boolean} Add a listener to document on mouseleave. This allows drag events to continue across the entire page. + */ + function MouseSync(options) { + this.options = Object.create(MouseSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + + if (options) this.setOptions(options); + + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + + this._eventInput.on('mousedown', _handleStart.bind(this)); + this._eventInput.on('mousemove', _handleMove.bind(this)); + this._eventInput.on('mouseup', _handleEnd.bind(this)); + + if (this.options.propogate) this._eventInput.on('mouseleave', _handleLeave.bind(this)); + else this._eventInput.on('mouseleave', _handleEnd.bind(this)); + + if (this.options.clickThreshold) { + window.addEventListener('click', function(event) { + if (Math.sqrt(Math.pow(this._displacement[0], 2) + Math.pow(this._displacement[1], 2)) > this.options.clickThreshold) { + event.stopPropagation(); + } + }.bind(this), true); + } + + this._payload = { + delta : null, + position : null, + velocity : null, + clientX : 0, + clientY : 0, + offsetX : 0, + offsetY : 0 + }; + + this._positionHistory = []; + this._position = null; // to be deprecated + this._prevCoord = undefined; + this._prevTime = undefined; + this._down = false; + this._moved = false; + this._displacement = [0,0]; + this._documentActive = false; + } + + MouseSync.DEFAULT_OPTIONS = { + clickThreshold: undefined, + direction: undefined, + rails: false, + scale: 1, + propogate: true, // events piped to document on mouseleave + velocitySampleLength: 10, + preventDefault: true + }; + + MouseSync.DIRECTION_X = 0; + MouseSync.DIRECTION_Y = 1; + + var MINIMUM_TICK_TIME = 8; + + /** + * Triggered by mousedown. + * + * @method _handleStart + * @private + */ + function _handleStart(event) { + var delta; + var velocity; + if (this.options.preventDefault) event.preventDefault(); // prevent drag + + var x = event.clientX; + var y = event.clientY; + + this._prevCoord = [x, y]; + this._prevTime = Date.now(); + this._down = true; + this._move = false; + + if (this.options.direction !== undefined) { + this._position = 0; + delta = 0; + velocity = 0; + } + else { + this._position = [0, 0]; + delta = [0, 0]; + velocity = [0, 0]; + } + + if (this.options.clickThreshold) { + this._displacement = [0,0]; + } + + var payload = this._payload; + payload.delta = delta; + payload.position = this._position; + payload.velocity = velocity; + payload.clientX = x; + payload.clientY = y; + payload.offsetX = event.offsetX; + payload.offsetY = event.offsetY; + + this._positionHistory.push({ + position: payload.position.slice ? payload.position.slice(0) : payload.position, + time: this._prevTime + }); + + this._eventOutput.emit('start', payload); + this._documentActive = false; + } + + /** + * Triggered by mousemove. + * + * @method _handleMove + * @private + */ + function _handleMove(event) { + if (!this._prevCoord) return; + + var prevCoord = this._prevCoord; + var prevTime = this._prevTime; + + var x = event.clientX; + var y = event.clientY; + + var currTime = Date.now(); + + var diffX = x - prevCoord[0]; + var diffY = y - prevCoord[1]; + + if (this.options.rails) { + if (Math.abs(diffX) > Math.abs(diffY)) diffY = 0; + else diffX = 0; + } + + var diffTime = Math.max(currTime - this._positionHistory[0].time, MINIMUM_TICK_TIME); // minimum tick time + + var scale = this.options.scale; + var nextVel; + var nextDelta; + + if (this.options.direction === MouseSync.DIRECTION_X) { + nextDelta = scale * diffX; + this._position += nextDelta; + nextVel = scale * (this._position - this._positionHistory[0].position) / diffTime; + } + else if (this.options.direction === MouseSync.DIRECTION_Y) { + nextDelta = scale * diffY; + this._position += nextDelta; + nextVel = scale * (this._position - this._positionHistory[0].position) / diffTime; + } + else { + nextDelta = [scale * diffX, scale * diffY]; + nextVel = [ + scale * (this._position[0] - this._positionHistory[0].position[0]) / diffTime, + scale * (this._position[1] - this._positionHistory[0].position[1]) / diffTime + ]; + this._position[0] += nextDelta[0]; + this._position[1] += nextDelta[1]; + } + + if (this.options.clickThreshold !== false) { + this._displacement[0] += diffX; + this._displacement[1] += diffY; + } + + var payload = this._payload; + payload.delta = nextDelta; + payload.position = this._position; + payload.velocity = nextVel; + payload.clientX = x; + payload.clientY = y; + payload.offsetX = event.offsetX; + payload.offsetY = event.offsetY; + + if (this._positionHistory.length === this.options.velocitySampleLength) { + this._positionHistory.shift(); + } + + this._positionHistory.push({ + position: payload.position.slice ? payload.position.slice(0) : payload.position, + time: currTime + }); + + this._eventOutput.emit('update', payload); + + this._prevCoord = [x, y]; + this._prevTime = currTime; + this._move = true; + } + + /** + * Triggered by mouseup on the element or document body if propagation is enabled, or + * mouseleave if propagation is off. + * + * @method _handleEnd + * @private + */ + function _handleEnd(event) { + if (!this._down) return; + + this._eventOutput.emit('end', this._payload); + this._prevCoord = undefined; + this._prevTime = undefined; + this._down = false; + this._move = false; + this._positionHistory = []; + } + + /** + * Switches the mousemove listener to the document body, if propagation is enabled. + * @method _handleLeave + * @private + */ + function _handleLeave(event) { + if (!this._down || !this._move) return; + + if (!this._documentActive) { + var boundMove = _handleMove.bind(this); + var boundEnd = function(event) { + _handleEnd.call(this, event); + document.removeEventListener('mousemove', boundMove); + document.removeEventListener('mouseup', boundEnd); + }.bind(this, event); + document.addEventListener('mousemove', boundMove); + document.addEventListener('mouseup', boundEnd); + this._documentActive = true; + } + } + + /** + * Return entire options dictionary, including defaults. + * + * @method getOptions + * @return {Object} configuration options + */ + MouseSync.prototype.getOptions = function getOptions() { + return this.options; + }; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param [options] {Object} default options overrides + * @param [options.direction] {Number} read from a particular axis + * @param [options.rails] {Boolean} read from axis with greatest differential + * @param [options.propogate] {Boolean} add listened to document on mouseleave + */ + MouseSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + module.exports = MouseSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/TwoFingerSync',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + + /** + * Helper to PinchSync, RotateSync, and ScaleSync. Generalized handling of + * two-finger touch events. + * This class is meant to be overridden and not used directly. + * + * @class TwoFingerSync + * @constructor + */ + function TwoFingerSync() { + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + + this.touchAEnabled = false; + this.touchAId = 0; + this.posA = null; + this.timestampA = 0; + this.touchBEnabled = false; + this.touchBId = 0; + this.posB = null; + this.timestampB = 0; + + this._eventInput.on('touchstart', this.handleStart.bind(this)); + this._eventInput.on('touchmove', this.handleMove.bind(this)); + this._eventInput.on('touchend', this.handleEnd.bind(this)); + this._eventInput.on('touchcancel', this.handleEnd.bind(this)); + } + + TwoFingerSync.calculateAngle = function(posA, posB) { + var diffX = posB[0] - posA[0]; + var diffY = posB[1] - posA[1]; + return Math.atan2(diffY, diffX); + }; + + TwoFingerSync.calculateDistance = function(posA, posB) { + var diffX = posB[0] - posA[0]; + var diffY = posB[1] - posA[1]; + return Math.sqrt(diffX * diffX + diffY * diffY); + }; + + TwoFingerSync.calculateCenter = function(posA, posB) { + return [(posA[0] + posB[0]) / 2.0, (posA[1] + posB[1]) / 2.0]; + }; + + var _now = Date.now; + + // private + TwoFingerSync.prototype.handleStart = function handleStart(event) { + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + if (!this.touchAEnabled) { + this.touchAId = touch.identifier; + this.touchAEnabled = true; + this.posA = [touch.pageX, touch.pageY]; + this.timestampA = _now(); + } + else if (!this.touchBEnabled) { + this.touchBId = touch.identifier; + this.touchBEnabled = true; + this.posB = [touch.pageX, touch.pageY]; + this.timestampB = _now(); + this._startUpdate(event); + } + } + }; + + // private + TwoFingerSync.prototype.handleMove = function handleMove(event) { + if (!(this.touchAEnabled && this.touchBEnabled)) return; + var prevTimeA = this.timestampA; + var prevTimeB = this.timestampB; + var diffTime; + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + if (touch.identifier === this.touchAId) { + this.posA = [touch.pageX, touch.pageY]; + this.timestampA = _now(); + diffTime = this.timestampA - prevTimeA; + } + else if (touch.identifier === this.touchBId) { + this.posB = [touch.pageX, touch.pageY]; + this.timestampB = _now(); + diffTime = this.timestampB - prevTimeB; + } + } + if (diffTime) this._moveUpdate(diffTime); + }; + + // private + TwoFingerSync.prototype.handleEnd = function handleEnd(event) { + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + if (touch.identifier === this.touchAId || touch.identifier === this.touchBId) { + if (this.touchAEnabled && this.touchBEnabled) { + this._eventOutput.emit('end', { + touches : [this.touchAId, this.touchBId], + angle : this._angle + }); + } + this.touchAEnabled = false; + this.touchAId = 0; + this.touchBEnabled = false; + this.touchBId = 0; + } + } + }; + + module.exports = TwoFingerSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/PinchSync',['require','exports','module','./TwoFingerSync','../core/OptionsManager'],function(require, exports, module) { + var TwoFingerSync = require('./TwoFingerSync'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Handles piped in two-finger touch events to change position via pinching / expanding. + * Emits 'start', 'update' and 'end' events with + * position, velocity, touch ids, and distance between fingers. + * + * @class PinchSync + * @extends TwoFingerSync + * @constructor + * @param {Object} options default options overrides + * @param {Number} [options.scale] scale velocity by this factor + */ + function PinchSync(options) { + TwoFingerSync.call(this); + + this.options = Object.create(PinchSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) this.setOptions(options); + + this._displacement = 0; + this._previousDistance = 0; + } + + PinchSync.prototype = Object.create(TwoFingerSync.prototype); + PinchSync.prototype.constructor = PinchSync; + + PinchSync.DEFAULT_OPTIONS = { + scale : 1 + }; + + PinchSync.prototype._startUpdate = function _startUpdate(event) { + this._previousDistance = TwoFingerSync.calculateDistance(this.posA, this.posB); + this._displacement = 0; + + this._eventOutput.emit('start', { + count: event.touches.length, + touches: [this.touchAId, this.touchBId], + distance: this._dist, + center: TwoFingerSync.calculateCenter(this.posA, this.posB) + }); + }; + + PinchSync.prototype._moveUpdate = function _moveUpdate(diffTime) { + var currDist = TwoFingerSync.calculateDistance(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + + var scale = this.options.scale; + var delta = scale * (currDist - this._previousDistance); + var velocity = delta / diffTime; + + this._previousDistance = currDist; + this._displacement += delta; + + this._eventOutput.emit('update', { + delta : delta, + velocity: velocity, + distance: currDist, + displacement: this._displacement, + center: center, + touches: [this.touchAId, this.touchBId] + }); + }; + + /** + * Return entire options dictionary, including defaults. + * + * @method getOptions + * @return {Object} configuration options + */ + PinchSync.prototype.getOptions = function getOptions() { + return this.options; + }; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options + * @param {Number} [options.scale] scale velocity by this factor + */ + PinchSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + module.exports = PinchSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/RotateSync',['require','exports','module','./TwoFingerSync','../core/OptionsManager'],function(require, exports, module) { + var TwoFingerSync = require('./TwoFingerSync'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Handles piped in two-finger touch events to increase or decrease scale via pinching / expanding. + * Emits 'start', 'update' and 'end' events an object with position, velocity, touch ids, and angle. + * Useful for determining a rotation factor from initial two-finger touch. + * + * @class RotateSync + * @extends TwoFingerSync + * @constructor + * @param {Object} options default options overrides + * @param {Number} [options.scale] scale velocity by this factor + */ + function RotateSync(options) { + TwoFingerSync.call(this); + + this.options = Object.create(RotateSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) this.setOptions(options); + + this._angle = 0; + this._previousAngle = 0; + } + + RotateSync.prototype = Object.create(TwoFingerSync.prototype); + RotateSync.prototype.constructor = RotateSync; + + RotateSync.DEFAULT_OPTIONS = { + scale : 1 + }; + + RotateSync.prototype._startUpdate = function _startUpdate(event) { + this._angle = 0; + this._previousAngle = TwoFingerSync.calculateAngle(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + this._eventOutput.emit('start', { + count: event.touches.length, + angle: this._angle, + center: center, + touches: [this.touchAId, this.touchBId] + }); + }; + + RotateSync.prototype._moveUpdate = function _moveUpdate(diffTime) { + var scale = this.options.scale; + + var currAngle = TwoFingerSync.calculateAngle(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + + var diffTheta = scale * (currAngle - this._previousAngle); + var velTheta = diffTheta / diffTime; + + this._angle += diffTheta; + + this._eventOutput.emit('update', { + delta : diffTheta, + velocity: velTheta, + angle: this._angle, + center: center, + touches: [this.touchAId, this.touchBId] + }); + + this._previousAngle = currAngle; + }; + + /** + * Return entire options dictionary, including defaults. + * + * @method getOptions + * @return {Object} configuration options + */ + RotateSync.prototype.getOptions = function getOptions() { + return this.options; + }; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options + * @param {Number} [options.scale] scale velocity by this factor + */ + RotateSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + module.exports = RotateSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/ScaleSync',['require','exports','module','./TwoFingerSync','../core/OptionsManager'],function(require, exports, module) { + var TwoFingerSync = require('./TwoFingerSync'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Handles piped in two-finger touch events to increase or decrease scale via pinching / expanding. + * Emits 'start', 'update' and 'end' events an object with position, velocity, touch ids, distance, and scale factor. + * Useful for determining a scaling factor from initial two-finger touch. + * + * @class ScaleSync + * @extends TwoFingerSync + * @constructor + * @param {Object} options default options overrides + * @param {Number} [options.scale] scale velocity by this factor + */ + function ScaleSync(options) { + TwoFingerSync.call(this); + + this.options = Object.create(ScaleSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) this.setOptions(options); + + this._scaleFactor = 1; + this._startDist = 0; + this._eventInput.on('pipe', _reset.bind(this)); + } + + ScaleSync.prototype = Object.create(TwoFingerSync.prototype); + ScaleSync.prototype.constructor = ScaleSync; + + ScaleSync.DEFAULT_OPTIONS = { + scale : 1 + }; + + function _reset() { + this.touchAId = undefined; + this.touchBId = undefined; + } + + // handles initial touch of two fingers + ScaleSync.prototype._startUpdate = function _startUpdate(event) { + this._scaleFactor = 1; + this._startDist = TwoFingerSync.calculateDistance(this.posA, this.posB); + this._eventOutput.emit('start', { + count: event.touches.length, + touches: [this.touchAId, this.touchBId], + distance: this._startDist, + center: TwoFingerSync.calculateCenter(this.posA, this.posB) + }); + }; + + // handles movement of two fingers + ScaleSync.prototype._moveUpdate = function _moveUpdate(diffTime) { + var scale = this.options.scale; + + var currDist = TwoFingerSync.calculateDistance(this.posA, this.posB); + var center = TwoFingerSync.calculateCenter(this.posA, this.posB); + + var delta = (currDist - this._startDist) / this._startDist; + var newScaleFactor = Math.max(1 + scale * delta, 0); + var veloScale = (newScaleFactor - this._scaleFactor) / diffTime; + + this._eventOutput.emit('update', { + delta : delta, + scale: newScaleFactor, + velocity: veloScale, + distance: currDist, + center : center, + touches: [this.touchAId, this.touchBId] + }); + + this._scaleFactor = newScaleFactor; + }; + + /** + * Return entire options dictionary, including defaults. + * + * @method getOptions + * @return {Object} configuration options + */ + ScaleSync.prototype.getOptions = function getOptions() { + return this.options; + }; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options + * @param {Number} [options.scale] scale velocity by this factor + */ + ScaleSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + module.exports = ScaleSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/ScrollSync',['require','exports','module','../core/EventHandler','../core/Engine','../core/OptionsManager'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + var Engine = require('../core/Engine'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Handles piped in mousewheel events. + * Emits 'start', 'update', and 'end' events with payloads including: + * delta: change since last position, + * position: accumulated deltas, + * velocity: speed of change in pixels per ms, + * slip: true (unused). + * + * Can be used as delegate of GenericSync. + * + * @class ScrollSync + * @constructor + * @param {Object} [options] overrides of default options + * @param {Number} [options.direction] Pay attention to x changes (ScrollSync.DIRECTION_X), + * y changes (ScrollSync.DIRECTION_Y) or both (undefined) + * @param {Number} [options.minimumEndSpeed] End speed calculation floors at this number, in pixels per ms + * @param {boolean} [options.rails] whether to snap position calculations to nearest axis + * @param {Number | Array.Number} [options.scale] scale outputs in by scalar or pair of scalars + * @param {Number} [options.stallTime] reset time for velocity calculation in ms + */ + function ScrollSync(options) { + this.options = Object.create(ScrollSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) this.setOptions(options); + + this._payload = { + delta : null, + position : null, + velocity : null, + slip : true + }; + + this._eventInput = new EventHandler(); + this._eventOutput = new EventHandler(); + + EventHandler.setInputHandler(this, this._eventInput); + EventHandler.setOutputHandler(this, this._eventOutput); + + this._position = (this.options.direction === undefined) ? [0,0] : 0; + this._prevTime = undefined; + this._prevVel = undefined; + this._eventInput.on('mousewheel', _handleMove.bind(this)); + this._eventInput.on('wheel', _handleMove.bind(this)); + this._inProgress = false; + this._loopBound = false; + } + + ScrollSync.DEFAULT_OPTIONS = { + direction: undefined, + minimumEndSpeed: Infinity, + rails: false, + scale: 1, + stallTime: 50, + lineHeight: 40, + preventDefault: true + }; + + ScrollSync.DIRECTION_X = 0; + ScrollSync.DIRECTION_Y = 1; + + var MINIMUM_TICK_TIME = 8; + + var _now = Date.now; + + function _newFrame() { + if (this._inProgress && (_now() - this._prevTime) > this.options.stallTime) { + this._inProgress = false; + + var finalVel = (Math.abs(this._prevVel) >= this.options.minimumEndSpeed) + ? this._prevVel + : 0; + + var payload = this._payload; + payload.position = this._position; + payload.velocity = finalVel; + payload.slip = true; + + this._eventOutput.emit('end', payload); + } + } + + function _handleMove(event) { + if (this.options.preventDefault) event.preventDefault(); + + if (!this._inProgress) { + this._inProgress = true; + this._position = (this.options.direction === undefined) ? [0,0] : 0; + payload = this._payload; + payload.slip = true; + payload.position = this._position; + payload.clientX = event.clientX; + payload.clientY = event.clientY; + payload.offsetX = event.offsetX; + payload.offsetY = event.offsetY; + this._eventOutput.emit('start', payload); + if (!this._loopBound) { + Engine.on('prerender', _newFrame.bind(this)); + this._loopBound = true; + } + } + + var currTime = _now(); + var prevTime = this._prevTime || currTime; + + var diffX = (event.wheelDeltaX !== undefined) ? event.wheelDeltaX : -event.deltaX; + var diffY = (event.wheelDeltaY !== undefined) ? event.wheelDeltaY : -event.deltaY; + + if (event.deltaMode === 1) { // units in lines, not pixels + diffX *= this.options.lineHeight; + diffY *= this.options.lineHeight; + } + + if (this.options.rails) { + if (Math.abs(diffX) > Math.abs(diffY)) diffY = 0; + else diffX = 0; + } + + var diffTime = Math.max(currTime - prevTime, MINIMUM_TICK_TIME); // minimum tick time + + var velX = diffX / diffTime; + var velY = diffY / diffTime; + + var scale = this.options.scale; + var nextVel; + var nextDelta; + + if (this.options.direction === ScrollSync.DIRECTION_X) { + nextDelta = scale * diffX; + nextVel = scale * velX; + this._position += nextDelta; + } + else if (this.options.direction === ScrollSync.DIRECTION_Y) { + nextDelta = scale * diffY; + nextVel = scale * velY; + this._position += nextDelta; + } + else { + nextDelta = [scale * diffX, scale * diffY]; + nextVel = [scale * velX, scale * velY]; + this._position[0] += nextDelta[0]; + this._position[1] += nextDelta[1]; + } + + var payload = this._payload; + payload.delta = nextDelta; + payload.velocity = nextVel; + payload.position = this._position; + payload.slip = true; + + this._eventOutput.emit('update', payload); + + this._prevTime = currTime; + this._prevVel = nextVel; + } + + /** + * Return entire options dictionary, including defaults. + * + * @method getOptions + * @return {Object} configuration options + */ + ScrollSync.prototype.getOptions = function getOptions() { + return this.options; + }; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options + * @param {Number} [options.minimimEndSpeed] If final velocity smaller than this, round down to 0. + * @param {Number} [options.stallTime] ms of non-motion before 'end' emitted + * @param {Number} [options.rails] whether to constrain to nearest axis. + * @param {Number} [options.direction] ScrollSync.DIRECTION_X, DIRECTION_Y - + * pay attention to one specific direction. + * @param {Number} [options.scale] constant factor to scale velocity output + */ + ScrollSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + module.exports = ScrollSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/TouchTracker',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + + var _now = Date.now; + + function _timestampTouch(touch, event, history) { + return { + x: touch.clientX, + y: touch.clientY, + identifier : touch.identifier, + origin: event.origin, + timestamp: _now(), + count: event.touches.length, + history: history + }; + } + + function _handleStart(event) { + if (event.touches.length > this.touchLimit) return; + this.isTouched = true; + + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var data = _timestampTouch(touch, event, null); + this.eventOutput.emit('trackstart', data); + if (!this.selective && !this.touchHistory[touch.identifier]) this.track(data); + } + } + + function _handleMove(event) { + if (event.touches.length > this.touchLimit) return; + + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var history = this.touchHistory[touch.identifier]; + if (history) { + var data = _timestampTouch(touch, event, history); + this.touchHistory[touch.identifier].push(data); + this.eventOutput.emit('trackmove', data); + } + } + } + + function _handleEnd(event) { + if (!this.isTouched) return; + + for (var i = 0; i < event.changedTouches.length; i++) { + var touch = event.changedTouches[i]; + var history = this.touchHistory[touch.identifier]; + if (history) { + var data = _timestampTouch(touch, event, history); + this.eventOutput.emit('trackend', data); + delete this.touchHistory[touch.identifier]; + } + } + + this.isTouched = false; + } + + function _handleUnpipe() { + for (var i in this.touchHistory) { + var history = this.touchHistory[i]; + this.eventOutput.emit('trackend', { + touch: history[history.length - 1].touch, + timestamp: Date.now(), + count: 0, + history: history + }); + delete this.touchHistory[i]; + } + } + + /** + * Helper to TouchSync – tracks piped in touch events, organizes touch + * events by ID, and emits track events back to TouchSync. + * Emits 'trackstart', 'trackmove', and 'trackend' events upstream. + * + * @class TouchTracker + * @constructor + * @param {Object} options default options overrides + * @param [options.selective] {Boolean} selective if false, saves state for each touch + * @param [options.touchLimit] {Number} touchLimit upper bound for emitting events based on number of touches + */ + function TouchTracker(options) { + this.selective = options.selective; + this.touchLimit = options.touchLimit || 1; + + this.touchHistory = {}; + + this.eventInput = new EventHandler(); + this.eventOutput = new EventHandler(); + + EventHandler.setInputHandler(this, this.eventInput); + EventHandler.setOutputHandler(this, this.eventOutput); + + this.eventInput.on('touchstart', _handleStart.bind(this)); + this.eventInput.on('touchmove', _handleMove.bind(this)); + this.eventInput.on('touchend', _handleEnd.bind(this)); + this.eventInput.on('touchcancel', _handleEnd.bind(this)); + this.eventInput.on('unpipe', _handleUnpipe.bind(this)); + + this.isTouched = false; + } + + /** + * Record touch data, if selective is false. + * @private + * @method track + * @param {Object} data touch data + */ + TouchTracker.prototype.track = function track(data) { + this.touchHistory[data.identifier] = [data]; + }; + + module.exports = TouchTracker; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/inputs/TouchSync',['require','exports','module','./TouchTracker','../core/EventHandler','../core/OptionsManager'],function(require, exports, module) { + var TouchTracker = require('./TouchTracker'); + var EventHandler = require('../core/EventHandler'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Handles piped in touch events. Emits 'start', 'update', and 'events' + * events with delta, position, velocity, acceleration, clientX, clientY, count, and touch id. + * Useful for dealing with inputs on touch devices. Designed to be used either as standalone, or + * included in a GenericSync. + * + * @class TouchSync + * @constructor + * + * @example + * var Surface = require('../core/Surface'); + * var TouchSync = require('../inputs/TouchSync'); + * + * var surface = new Surface({ size: [100, 100] }); + * var touchSync = new TouchSync(); + * surface.pipe(touchSync); + * + * touchSync.on('start', function (e) { // react to start }); + * touchSync.on('update', function (e) { // react to update }); + * touchSync.on('end', function (e) { // react to end });* + * + * @param [options] {Object} default options overrides + * @param [options.direction] {Number} read from a particular axis + * @param [options.rails] {Boolean} read from axis with greatest differential + * @param [options.velocitySampleLength] {Number} Number of previous frames to check velocity against. + * @param [options.scale] {Number} constant factor to scale velocity output + * @param [options.touchLimit] {Number} touchLimit upper bound for emitting events based on number of touches + */ + function TouchSync(options) { + this.options = Object.create(TouchSync.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + if (options) this.setOptions(options); + + this._eventOutput = new EventHandler(); + this._touchTracker = new TouchTracker({ + touchLimit: this.options.touchLimit + }); + + EventHandler.setOutputHandler(this, this._eventOutput); + EventHandler.setInputHandler(this, this._touchTracker); + + this._touchTracker.on('trackstart', _handleStart.bind(this)); + this._touchTracker.on('trackmove', _handleMove.bind(this)); + this._touchTracker.on('trackend', _handleEnd.bind(this)); + + this._payload = { + delta : null, + position : null, + velocity : null, + clientX : undefined, + clientY : undefined, + count : 0, + touch : undefined + }; + + this._position = null; // to be deprecated + } + + TouchSync.DEFAULT_OPTIONS = { + direction: undefined, + rails: false, + touchLimit: 1, + velocitySampleLength: 10, + scale: 1 + }; + + TouchSync.DIRECTION_X = 0; + TouchSync.DIRECTION_Y = 1; + + var MINIMUM_TICK_TIME = 8; + + /** + * Triggered by trackstart. + * @method _handleStart + * @private + */ + function _handleStart(data) { + var velocity; + var delta; + if (this.options.direction !== undefined){ + this._position = 0; + velocity = 0; + delta = 0; + } + else { + this._position = [0, 0]; + velocity = [0, 0]; + delta = [0, 0]; + } + + var payload = this._payload; + payload.delta = delta; + payload.position = this._position; + payload.velocity = velocity; + payload.clientX = data.x; + payload.clientY = data.y; + payload.count = data.count; + payload.touch = data.identifier; + + this._eventOutput.emit('start', payload); + } + + /** + * Triggered by trackmove. + * @method _handleMove + * @private + */ + function _handleMove(data) { + var history = data.history; + + var currHistory = history[history.length - 1]; + var prevHistory = history[history.length - 2]; + + var distantHistory = history[history.length - this.options.velocitySampleLength] ? + history[history.length - this.options.velocitySampleLength] : + history[history.length - 2]; + + var distantTime = distantHistory.timestamp; + var currTime = currHistory.timestamp; + + var diffX = currHistory.x - prevHistory.x; + var diffY = currHistory.y - prevHistory.y; + + var velDiffX = currHistory.x - distantHistory.x; + var velDiffY = currHistory.y - distantHistory.y; + + if (this.options.rails) { + if (Math.abs(diffX) > Math.abs(diffY)) diffY = 0; + else diffX = 0; + + if (Math.abs(velDiffX) > Math.abs(velDiffY)) velDiffY = 0; + else velDiffX = 0; + } + + var diffTime = Math.max(currTime - distantTime, MINIMUM_TICK_TIME); + + var velX = velDiffX / diffTime; + var velY = velDiffY / diffTime; + + var scale = this.options.scale; + var nextVel; + var nextDelta; + + if (this.options.direction === TouchSync.DIRECTION_X) { + nextDelta = scale * diffX; + nextVel = scale * velX; + this._position += nextDelta; + } + else if (this.options.direction === TouchSync.DIRECTION_Y) { + nextDelta = scale * diffY; + nextVel = scale * velY; + this._position += nextDelta; + } + else { + nextDelta = [scale * diffX, scale * diffY]; + nextVel = [scale * velX, scale * velY]; + this._position[0] += nextDelta[0]; + this._position[1] += nextDelta[1]; + } + + var payload = this._payload; + payload.delta = nextDelta; + payload.velocity = nextVel; + payload.position = this._position; + payload.clientX = data.x; + payload.clientY = data.y; + payload.count = data.count; + payload.touch = data.identifier; + + this._eventOutput.emit('update', payload); + } + + /** + * Triggered by trackend. + * @method _handleEnd + * @private + */ + function _handleEnd(data) { + this._payload.count = data.count; + this._eventOutput.emit('end', this._payload); + } + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param [options] {Object} default options overrides + * @param [options.direction] {Number} read from a particular axis + * @param [options.rails] {Boolean} read from axis with greatest differential + * @param [options.scale] {Number} constant factor to scale velocity output + */ + TouchSync.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + /** + * Return entire options dictionary, including defaults. + * + * @method getOptions + * @return {Object} configuration options + */ + TouchSync.prototype.getOptions = function getOptions() { + return this.options; + }; + + module.exports = TouchSync; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/math/Vector',['require','exports','module'],function(require, exports, module) { + + /** + * Three-element floating point vector. + * + * @class Vector + * @constructor + * + * @param {number} x x element value + * @param {number} y y element value + * @param {number} z z element value + */ + function Vector(x,y,z) { + if (arguments.length === 1 && x !== undefined) this.set(x); + else { + this.x = x || 0; + this.y = y || 0; + this.z = z || 0; + } + return this; + } + + var _register = new Vector(0,0,0); + + /** + * Add this element-wise to another Vector, element-wise. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method add + * @param {Vector} v addend + * @return {Vector} vector sum + */ + Vector.prototype.add = function add(v) { + return _setXYZ.call(_register, + this.x + v.x, + this.y + v.y, + this.z + v.z + ); + }; + + /** + * Subtract another vector from this vector, element-wise. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method sub + * @param {Vector} v subtrahend + * @return {Vector} vector difference + */ + Vector.prototype.sub = function sub(v) { + return _setXYZ.call(_register, + this.x - v.x, + this.y - v.y, + this.z - v.z + ); + }; + + /** + * Scale Vector by floating point r. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method mult + * + * @param {number} r scalar + * @return {Vector} vector result + */ + Vector.prototype.mult = function mult(r) { + return _setXYZ.call(_register, + r * this.x, + r * this.y, + r * this.z + ); + }; + + /** + * Scale Vector by floating point 1/r. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method div + * + * @param {number} r scalar + * @return {Vector} vector result + */ + Vector.prototype.div = function div(r) { + return this.mult(1 / r); + }; + + /** + * Given another vector v, return cross product (v)x(this). + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method cross + * @param {Vector} v Left Hand Vector + * @return {Vector} vector result + */ + Vector.prototype.cross = function cross(v) { + var x = this.x; + var y = this.y; + var z = this.z; + var vx = v.x; + var vy = v.y; + var vz = v.z; + + return _setXYZ.call(_register, + z * vy - y * vz, + x * vz - z * vx, + y * vx - x * vy + ); + }; + + /** + * Component-wise equality test between this and Vector v. + * @method equals + * @param {Vector} v vector to compare + * @return {boolean} + */ + Vector.prototype.equals = function equals(v) { + return (v.x === this.x && v.y === this.y && v.z === this.z); + }; + + /** + * Rotate clockwise around x-axis by theta radians. + * Note: This sets the internal result register, so other references to that vector will change. + * @method rotateX + * @param {number} theta radians + * @return {Vector} rotated vector + */ + Vector.prototype.rotateX = function rotateX(theta) { + var x = this.x; + var y = this.y; + var z = this.z; + + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + + return _setXYZ.call(_register, + x, + y * cosTheta - z * sinTheta, + y * sinTheta + z * cosTheta + ); + }; + + /** + * Rotate clockwise around y-axis by theta radians. + * Note: This sets the internal result register, so other references to that vector will change. + * @method rotateY + * @param {number} theta radians + * @return {Vector} rotated vector + */ + Vector.prototype.rotateY = function rotateY(theta) { + var x = this.x; + var y = this.y; + var z = this.z; + + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + + return _setXYZ.call(_register, + z * sinTheta + x * cosTheta, + y, + z * cosTheta - x * sinTheta + ); + }; + + /** + * Rotate clockwise around z-axis by theta radians. + * Note: This sets the internal result register, so other references to that vector will change. + * @method rotateZ + * @param {number} theta radians + * @return {Vector} rotated vector + */ + Vector.prototype.rotateZ = function rotateZ(theta) { + var x = this.x; + var y = this.y; + var z = this.z; + + var cosTheta = Math.cos(theta); + var sinTheta = Math.sin(theta); + + return _setXYZ.call(_register, + x * cosTheta - y * sinTheta, + x * sinTheta + y * cosTheta, + z + ); + }; + + /** + * Return dot product of this with a second Vector + * @method dot + * @param {Vector} v second vector + * @return {number} dot product + */ + Vector.prototype.dot = function dot(v) { + return this.x * v.x + this.y * v.y + this.z * v.z; + }; + + /** + * Return squared length of this vector + * @method normSquared + * @return {number} squared length + */ + Vector.prototype.normSquared = function normSquared() { + return this.dot(this); + }; + + /** + * Return length of this vector + * @method norm + * @return {number} length + */ + Vector.prototype.norm = function norm() { + return Math.sqrt(this.normSquared()); + }; + + /** + * Scale Vector to specified length. + * If length is less than internal tolerance, set vector to [length, 0, 0]. + * Note: This sets the internal result register, so other references to that vector will change. + * @method normalize + * + * @param {number} length target length, default 1.0 + * @return {Vector} + */ + Vector.prototype.normalize = function normalize(length) { + if (arguments.length === 0) length = 1; + var norm = this.norm(); + + if (norm > 1e-7) return _setFromVector.call(_register, this.mult(length / norm)); + else return _setXYZ.call(_register, length, 0, 0); + }; + + /** + * Make a separate copy of the Vector. + * + * @method clone + * + * @return {Vector} + */ + Vector.prototype.clone = function clone() { + return new Vector(this); + }; + + /** + * True if and only if every value is 0 (or falsy) + * + * @method isZero + * + * @return {boolean} + */ + Vector.prototype.isZero = function isZero() { + return !(this.x || this.y || this.z); + }; + + function _setXYZ(x,y,z) { + this.x = x; + this.y = y; + this.z = z; + return this; + } + + function _setFromArray(v) { + return _setXYZ.call(this,v[0],v[1],v[2] || 0); + } + + function _setFromVector(v) { + return _setXYZ.call(this, v.x, v.y, v.z); + } + + function _setFromNumber(x) { + return _setXYZ.call(this,x,0,0); + } + + /** + * Set this Vector to the values in the provided Array or Vector. + * + * @method set + * @param {object} v array, Vector, or number + * @return {Vector} this + */ + Vector.prototype.set = function set(v) { + if (v instanceof Array) return _setFromArray.call(this, v); + if (typeof v === 'number') return _setFromNumber.call(this, v); + return _setFromVector.call(this, v); + }; + + Vector.prototype.setXYZ = function(x,y,z) { + return _setXYZ.apply(this, arguments); + }; + + Vector.prototype.set1D = function(x) { + return _setFromNumber.call(this, x); + }; + + /** + * Put result of last internal register calculation in specified output vector. + * + * @method put + * @param {Vector} v destination vector + * @return {Vector} destination vector + */ + + Vector.prototype.put = function put(v) { + if (this === _register) _setFromVector.call(v, _register); + else _setFromVector.call(v, this); + }; + + /** + * Set this vector to [0,0,0] + * + * @method clear + */ + Vector.prototype.clear = function clear() { + return _setXYZ.call(this,0,0,0); + }; + + /** + * Scale this Vector down to specified "cap" length. + * If Vector shorter than cap, or cap is Infinity, do nothing. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method cap + * @return {Vector} capped vector + */ + Vector.prototype.cap = function cap(cap) { + if (cap === Infinity) return _setFromVector.call(_register, this); + var norm = this.norm(); + if (norm > cap) return _setFromVector.call(_register, this.mult(cap / norm)); + else return _setFromVector.call(_register, this); + }; + + /** + * Return projection of this Vector onto another. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method project + * @param {Vector} n vector to project upon + * @return {Vector} projected vector + */ + Vector.prototype.project = function project(n) { + return n.mult(this.dot(n)); + }; + + /** + * Reflect this Vector across provided vector. + * Note: This sets the internal result register, so other references to that vector will change. + * + * @method reflectAcross + * @param {Vector} n vector to reflect across + * @return {Vector} reflected vector + */ + Vector.prototype.reflectAcross = function reflectAcross(n) { + n.normalize().put(n); + return _setFromVector(_register, this.sub(this.project(n).mult(2))); + }; + + /** + * Convert Vector to three-element array. + * + * @method get + * @return {array} three-element array + */ + Vector.prototype.get = function get() { + return [this.x, this.y, this.z]; + }; + + Vector.prototype.get1D = function() { + return this.x; + }; + + module.exports = Vector; + +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/math/Matrix',['require','exports','module','./Vector'],function(require, exports, module) { + var Vector = require('./Vector'); + + /** + * A library for using a 3x3 numerical matrix, represented as a two-level array. + * + * @class Matrix + * @constructor + * + * @param {Array.Array} values array of rows + */ + function Matrix(values) { + this.values = values || + [ + [1,0,0], + [0,1,0], + [0,0,1] + ]; + + return this; + } + + var _register = new Matrix(); + var _vectorRegister = new Vector(); + + /** + * Return the values in the matrix as an array of numerical row arrays + * + * @method get + * + * @return {Array.array} matrix values as array of rows. + */ + Matrix.prototype.get = function get() { + return this.values; + }; + + /** + * Set the nested array of rows in the matrix. + * + * @method set + * + * @param {Array.array} values matrix values as array of rows. + */ + Matrix.prototype.set = function set(values) { + this.values = values; + }; + + /** + * Take this matrix as A, input vector V as a column vector, and return matrix product (A)(V). + * Note: This sets the internal vector register. Current handles to the vector register + * will see values changed. + * + * @method vectorMultiply + * + * @param {Vector} v input vector V + * @return {Vector} result of multiplication, as a handle to the internal vector register + */ + Matrix.prototype.vectorMultiply = function vectorMultiply(v) { + var M = this.get(); + var v0 = v.x; + var v1 = v.y; + var v2 = v.z; + + var M0 = M[0]; + var M1 = M[1]; + var M2 = M[2]; + + var M00 = M0[0]; + var M01 = M0[1]; + var M02 = M0[2]; + var M10 = M1[0]; + var M11 = M1[1]; + var M12 = M1[2]; + var M20 = M2[0]; + var M21 = M2[1]; + var M22 = M2[2]; + + return _vectorRegister.setXYZ( + M00*v0 + M01*v1 + M02*v2, + M10*v0 + M11*v1 + M12*v2, + M20*v0 + M21*v1 + M22*v2 + ); + }; + + /** + * Multiply the provided matrix M2 with this matrix. Result is (this) * (M2). + * Note: This sets the internal matrix register. Current handles to the register + * will see values changed. + * + * @method multiply + * + * @param {Matrix} M2 input matrix to multiply on the right + * @return {Matrix} result of multiplication, as a handle to the internal register + */ + Matrix.prototype.multiply = function multiply(M2) { + var M1 = this.get(); + var result = [[]]; + for (var i = 0; i < 3; i++) { + result[i] = []; + for (var j = 0; j < 3; j++) { + var sum = 0; + for (var k = 0; k < 3; k++) { + sum += M1[i][k] * M2[k][j]; + } + result[i][j] = sum; + } + } + return _register.set(result); + }; + + /** + * Creates a Matrix which is the transpose of this matrix. + * Note: This sets the internal matrix register. Current handles to the register + * will see values changed. + * + * @method transpose + * + * @return {Matrix} result of transpose, as a handle to the internal register + */ + Matrix.prototype.transpose = function transpose() { + var result = [[], [], []]; + var M = this.get(); + for (var row = 0; row < 3; row++) { + for (var col = 0; col < 3; col++) { + result[row][col] = M[col][row]; + } + } + return _register.set(result); + }; + + /** + * Clones the matrix + * + * @method clone + * @return {Matrix} New copy of the original matrix + */ + Matrix.prototype.clone = function clone() { + var values = this.get(); + var M = []; + for (var row = 0; row < 3; row++) + M[row] = values[row].slice(); + return new Matrix(M); + }; + + module.exports = Matrix; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/math/Quaternion',['require','exports','module','./Matrix'],function(require, exports, module) { + var Matrix = require('./Matrix'); + + /** + * @class Quaternion + * @constructor + * + * @param {Number} w + * @param {Number} x + * @param {Number} y + * @param {Number} z + */ + function Quaternion(w,x,y,z) { + if (arguments.length === 1) this.set(w); + else { + this.w = (w !== undefined) ? w : 1; //Angle + this.x = (x !== undefined) ? x : 0; //Axis.x + this.y = (y !== undefined) ? y : 0; //Axis.y + this.z = (z !== undefined) ? z : 0; //Axis.z + } + return this; + } + + var register = new Quaternion(1,0,0,0); + + /** + * Doc: TODO + * @method add + * @param {Quaternion} q + * @return {Quaternion} + */ + Quaternion.prototype.add = function add(q) { + return register.setWXYZ( + this.w + q.w, + this.x + q.x, + this.y + q.y, + this.z + q.z + ); + }; + + /* + * Docs: TODO + * + * @method sub + * @param {Quaternion} q + * @return {Quaternion} + */ + Quaternion.prototype.sub = function sub(q) { + return register.setWXYZ( + this.w - q.w, + this.x - q.x, + this.y - q.y, + this.z - q.z + ); + }; + + /** + * Doc: TODO + * + * @method scalarDivide + * @param {Number} s + * @return {Quaternion} + */ + Quaternion.prototype.scalarDivide = function scalarDivide(s) { + return this.scalarMultiply(1/s); + }; + + /* + * Docs: TODO + * + * @method scalarMultiply + * @param {Number} s + * @return {Quaternion} + */ + Quaternion.prototype.scalarMultiply = function scalarMultiply(s) { + return register.setWXYZ( + this.w * s, + this.x * s, + this.y * s, + this.z * s + ); + }; + + /* + * Docs: TODO + * + * @method multiply + * @param {Quaternion} q + * @return {Quaternion} + */ + Quaternion.prototype.multiply = function multiply(q) { + //left-handed coordinate system multiplication + var x1 = this.x; + var y1 = this.y; + var z1 = this.z; + var w1 = this.w; + var x2 = q.x; + var y2 = q.y; + var z2 = q.z; + var w2 = q.w || 0; + + return register.setWXYZ( + w1*w2 - x1*x2 - y1*y2 - z1*z2, + x1*w2 + x2*w1 + y2*z1 - y1*z2, + y1*w2 + y2*w1 + x1*z2 - x2*z1, + z1*w2 + z2*w1 + x2*y1 - x1*y2 + ); + }; + + var conj = new Quaternion(1,0,0,0); + + /* + * Docs: TODO + * + * @method rotateVector + * @param {Vector} v + * @return {Quaternion} + */ + Quaternion.prototype.rotateVector = function rotateVector(v) { + conj.set(this.conj()); + return register.set(this.multiply(v).multiply(conj)); + }; + + /* + * Docs: TODO + * + * @method inverse + * @return {Quaternion} + */ + Quaternion.prototype.inverse = function inverse() { + return register.set(this.conj().scalarDivide(this.normSquared())); + }; + + /* + * Docs: TODO + * + * @method negate + * @return {Quaternion} + */ + Quaternion.prototype.negate = function negate() { + return this.scalarMultiply(-1); + }; + + /* + * Docs: TODO + * + * @method conj + * @return {Quaternion} + */ + Quaternion.prototype.conj = function conj() { + return register.setWXYZ( + this.w, + -this.x, + -this.y, + -this.z + ); + }; + + /* + * Docs: TODO + * + * @method normalize + * @param {Number} length + * @return {Quaternion} + */ + Quaternion.prototype.normalize = function normalize(length) { + length = (length === undefined) ? 1 : length; + return this.scalarDivide(length * this.norm()); + }; + + /* + * Docs: TODO + * + * @method makeFromAngleAndAxis + * @param {Number} angle + * @param {Vector} v + * @return {Quaternion} + */ + Quaternion.prototype.makeFromAngleAndAxis = function makeFromAngleAndAxis(angle, v) { + //left handed quaternion creation: theta -> -theta + var n = v.normalize(); + var ha = angle*0.5; + var s = -Math.sin(ha); + this.x = s*n.x; + this.y = s*n.y; + this.z = s*n.z; + this.w = Math.cos(ha); + return this; + }; + + /* + * Docs: TODO + * + * @method setWXYZ + * @param {Number} w + * @param {Number} x + * @param {Number} y + * @param {Number} z + * @return {Quaternion} + */ + Quaternion.prototype.setWXYZ = function setWXYZ(w,x,y,z) { + register.clear(); + this.w = w; + this.x = x; + this.y = y; + this.z = z; + return this; + }; + + /* + * Docs: TODO + * + * @method set + * @param {Array|Quaternion} v + * @return {Quaternion} + */ + Quaternion.prototype.set = function set(v) { + if (v instanceof Array) { + this.w = 0; + this.x = v[0]; + this.y = v[1]; + this.z = v[2]; + } + else { + this.w = v.w; + this.x = v.x; + this.y = v.y; + this.z = v.z; + } + if (this !== register) register.clear(); + return this; + }; + + /** + * Docs: TODO + * + * @method put + * @param {Quaternion} q + * @return {Quaternion} + */ + Quaternion.prototype.put = function put(q) { + q.set(register); + }; + + /** + * Doc: TODO + * + * @method clone + * @return {Quaternion} + */ + Quaternion.prototype.clone = function clone() { + return new Quaternion(this); + }; + + /** + * Doc: TODO + * + * @method clear + * @return {Quaternion} + */ + Quaternion.prototype.clear = function clear() { + this.w = 1; + this.x = 0; + this.y = 0; + this.z = 0; + return this; + }; + + /** + * Doc: TODO + * + * @method isEqual + * @param {Quaternion} q + * @return {Boolean} + */ + Quaternion.prototype.isEqual = function isEqual(q) { + return q.w === this.w && q.x === this.x && q.y === this.y && q.z === this.z; + }; + + /** + * Doc: TODO + * + * @method dot + * @param {Quaternion} q + * @return {Number} + */ + Quaternion.prototype.dot = function dot(q) { + return this.w * q.w + this.x * q.x + this.y * q.y + this.z * q.z; + }; + + /** + * Doc: TODO + * + * @method normSquared + * @return {Number} + */ + Quaternion.prototype.normSquared = function normSquared() { + return this.dot(this); + }; + + /** + * Doc: TODO + * + * @method norm + * @return {Number} + */ + Quaternion.prototype.norm = function norm() { + return Math.sqrt(this.normSquared()); + }; + + /** + * Doc: TODO + * + * @method isZero + * @return {Boolean} + */ + Quaternion.prototype.isZero = function isZero() { + return !(this.x || this.y || this.z); + }; + + /** + * Doc: TODO + * + * @method getTransform + * @return {Transform} + */ + Quaternion.prototype.getTransform = function getTransform() { + var temp = this.normalize(1); + var x = temp.x; + var y = temp.y; + var z = temp.z; + var w = temp.w; + + //LHC system flattened to column major = RHC flattened to row major + return [ + 1 - 2*y*y - 2*z*z, + 2*x*y - 2*z*w, + 2*x*z + 2*y*w, + 0, + 2*x*y + 2*z*w, + 1 - 2*x*x - 2*z*z, + 2*y*z - 2*x*w, + 0, + 2*x*z - 2*y*w, + 2*y*z + 2*x*w, + 1 - 2*x*x - 2*y*y, + 0, + 0, + 0, + 0, + 1 + ]; + }; + + var matrixRegister = new Matrix(); + + /** + * Doc: TODO + * + * @method getMatrix + * @return {Transform} + */ + Quaternion.prototype.getMatrix = function getMatrix() { + var temp = this.normalize(1); + var x = temp.x; + var y = temp.y; + var z = temp.z; + var w = temp.w; + + //LHC system flattened to row major + return matrixRegister.set([ + [ + 1 - 2*y*y - 2*z*z, + 2*x*y + 2*z*w, + 2*x*z - 2*y*w + ], + [ + 2*x*y - 2*z*w, + 1 - 2*x*x - 2*z*z, + 2*y*z + 2*x*w + ], + [ + 2*x*z + 2*y*w, + 2*y*z - 2*x*w, + 1 - 2*x*x - 2*y*y + ] + ]); + }; + + var epsilon = 1e-5; + + /** + * Doc: TODO + * + * @method slerp + * @param {Quaternion} q + * @param {Number} t + * @return {Transform} + */ + Quaternion.prototype.slerp = function slerp(q, t) { + var omega; + var cosomega; + var sinomega; + var scaleFrom; + var scaleTo; + + cosomega = this.dot(q); + if ((1.0 - cosomega) > epsilon) { + omega = Math.acos(cosomega); + sinomega = Math.sin(omega); + scaleFrom = Math.sin((1.0 - t) * omega) / sinomega; + scaleTo = Math.sin(t * omega) / sinomega; + } + else { + scaleFrom = 1.0 - t; + scaleTo = t; + } + return register.set(this.scalarMultiply(scaleFrom/scaleTo).add(q).scalarMultiply(scaleTo)); + }; + + module.exports = Quaternion; + +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +/*eslint-disable new-cap */ +define('famous/math/Random',['require','exports','module'],function(require, exports, module) { + var RAND = Math.random; + + function _randomFloat(min, max) { + return min + RAND() * (max - min); + } + + function _randomInteger(min, max) { + return min + ((RAND() * (max - min + 1)) >> 0); + } + + function _range(randomFunction, min, max, dim) { + min = (min !== undefined) ? min : 0; + max = (max !== undefined) ? max : 1; + if (dim !== undefined) { + var result = []; + for (var i = 0; i < dim; i++) result.push(randomFunction(min, max)); + return result; + } + else return randomFunction(min, max); + } + + /** + * Very simple uniform random number generator library wrapping Math.random(). + * + * @class Random + * @static + */ + var Random = {}; + + /** + * Get single random integer between min and max (inclusive), or array + * of size dim if specified. + * + * @method integer + * + * @param {Number} min lower bound, default 0 + * @param {Number} max upper bound, default 1 + * @param {Number} dim (optional) dimension of output array, if specified + * @return {number | array} random integer, or optionally, an array of random integers + */ + Random.integer = function integer(min, max, dim) { + return _range(_randomInteger, min, max, dim); + }; + + /** + * Get single random float between min and max (inclusive), or array + * of size dim if specified + * + * @method range + * + * @param {Number} min lower bound, default 0 + * @param {Number} max upper bound, default 1 + * @param {Number} [dim] dimension of output array, if specified + * @return {Number} random float, or optionally an array + */ + Random.range = function range(min, max, dim) { + return _range(_randomFloat, min, max, dim); + }; + + /** + * Return random number among the set {-1 ,1} + * + * @method sign + * + * @param {Number} prob probability of returning 1, default 0.5 + * @return {Number} random sign (-1 or 1) + */ + Random.sign = function sign(prob) { + return Random.bool(prob) ? 1 : -1; + }; + + /** + * Return random boolean value, true or false. + * + * @method bool + * + * @param {Number} prob probability of returning true, default 0.5 + * @return {Boolean} random boolean + */ + Random.bool = function bool(prob) { + prob = (prob !== undefined) ? prob : 0.5; + return RAND() < prob; + }; + + module.exports = Random; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/math/Utilities',['require','exports','module'],function(require, exports, module) { + /** + * A few static methods. + * + * @class Utilities + * @static + */ + var Utilities = {}; + + /** + * Constrain input to range. + * + * @method clamp + * @param {Number} value input + * @param {Array.Number} range [min, max] + * @static + */ + Utilities.clamp = function clamp(value, range) { + return Math.max(Math.min(value, range[1]), range[0]); + }; + + /** + * Euclidean length of numerical array. + * + * @method length + * @param {Array.Number} array array of numbers + * @static + */ + Utilities.length = function length(array) { + var distanceSquared = 0; + for (var i = 0; i < array.length; i++) { + distanceSquared += array[i] * array[i]; + } + return Math.sqrt(distanceSquared); + }; + + module.exports = Utilities; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/modifiers/Draggable',['require','exports','module','../core/Transform','../transitions/Transitionable','../core/EventHandler','../math/Utilities','../inputs/GenericSync','../inputs/MouseSync','../inputs/TouchSync'],function(require, exports, module) { + var Transform = require('../core/Transform'); + var Transitionable = require('../transitions/Transitionable'); + var EventHandler = require('../core/EventHandler'); + var Utilities = require('../math/Utilities'); + + var GenericSync = require('../inputs/GenericSync'); + var MouseSync = require('../inputs/MouseSync'); + var TouchSync = require('../inputs/TouchSync'); + GenericSync.register({'mouse': MouseSync, 'touch': TouchSync}); + + /** + * Makes added render nodes responsive to drag beahvior. + * Emits events 'start', 'update', 'end'. + * @class Draggable + * @constructor + * @param {Object} [options] options configuration object. + * @param {Number} [options.snapX] grid width for snapping during drag + * @param {Number} [options.snapY] grid height for snapping during drag + * @param {Array.Number} [options.xRange] maxmimum [negative, positive] x displacement from start of drag + * @param {Array.Number} [options.yRange] maxmimum [negative, positive] y displacement from start of drag + * @param {Number} [options.scale] one pixel of input motion translates to this many pixels of output drag motion + * @param {Number} [options.projection] User should set to Draggable._direction.x or + * Draggable._direction.y to constrain to one axis. + * + */ + function Draggable(options) { + this.options = Object.create(Draggable.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + this._positionState = new Transitionable([0,0]); + this._differential = [0,0]; + this._active = true; + + this.sync = new GenericSync(['mouse', 'touch'], {scale : this.options.scale}); + this.eventOutput = new EventHandler(); + EventHandler.setInputHandler(this, this.sync); + EventHandler.setOutputHandler(this, this.eventOutput); + + _bindEvents.call(this); + } + + //binary representation of directions for bitwise operations + var _direction = { + x : 0x01, //001 + y : 0x02 //010 + }; + + Draggable.DIRECTION_X = _direction.x; + Draggable.DIRECTION_Y = _direction.y; + + var _clamp = Utilities.clamp; + + Draggable.DEFAULT_OPTIONS = { + projection : _direction.x | _direction.y, + scale : 1, + xRange : null, + yRange : null, + snapX : 0, + snapY : 0, + transition : {duration : 0} + }; + + function _mapDifferential(differential) { + var opts = this.options; + var projection = opts.projection; + var snapX = opts.snapX; + var snapY = opts.snapY; + + //axes + var tx = (projection & _direction.x) ? differential[0] : 0; + var ty = (projection & _direction.y) ? differential[1] : 0; + + //snapping + if (snapX > 0) tx -= tx % snapX; + if (snapY > 0) ty -= ty % snapY; + + return [tx, ty]; + } + + function _handleStart() { + if (!this._active) return; + if (this._positionState.isActive()) this._positionState.halt(); + this.eventOutput.emit('start', {position : this.getPosition()}); + } + + function _handleMove(event) { + if (!this._active) return; + + var options = this.options; + this._differential = event.position; + var newDifferential = _mapDifferential.call(this, this._differential); + + //buffer the differential if snapping is set + this._differential[0] -= newDifferential[0]; + this._differential[1] -= newDifferential[1]; + + var pos = this.getPosition(); + + //modify position, retain reference + pos[0] += newDifferential[0]; + pos[1] += newDifferential[1]; + + //handle bounding box + if (options.xRange){ + var xRange = [options.xRange[0] + 0.5 * options.snapX, options.xRange[1] - 0.5 * options.snapX]; + pos[0] = _clamp(pos[0], xRange); + } + + if (options.yRange){ + var yRange = [options.yRange[0] + 0.5 * options.snapY, options.yRange[1] - 0.5 * options.snapY]; + pos[1] = _clamp(pos[1], yRange); + } + + this.eventOutput.emit('update', {position : pos}); + } + + function _handleEnd() { + if (!this._active) return; + this.eventOutput.emit('end', {position : this.getPosition()}); + } + + function _bindEvents() { + this.sync.on('start', _handleStart.bind(this)); + this.sync.on('update', _handleMove.bind(this)); + this.sync.on('end', _handleEnd.bind(this)); + } + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options. See constructor. + */ + Draggable.prototype.setOptions = function setOptions(options) { + var currentOptions = this.options; + if (options.projection !== undefined) { + var proj = options.projection; + this.options.projection = 0; + ['x', 'y'].forEach(function(val) { + if (proj.indexOf(val) !== -1) currentOptions.projection |= _direction[val]; + }); + } + if (options.scale !== undefined) { + currentOptions.scale = options.scale; + this.sync.setOptions({ + scale: options.scale + }); + } + if (options.xRange !== undefined) currentOptions.xRange = options.xRange; + if (options.yRange !== undefined) currentOptions.yRange = options.yRange; + if (options.snapX !== undefined) currentOptions.snapX = options.snapX; + if (options.snapY !== undefined) currentOptions.snapY = options.snapY; + }; + + /** + * Get current delta in position from where this draggable started. + * + * @method getPosition + * + * @return {array} [x, y] position delta from start. + */ + Draggable.prototype.getPosition = function getPosition() { + return this._positionState.get(); + }; + + /** + * Transition the element to the desired relative position via provided transition. + * For example, calling this with [0,0] will not change the position. + * Callback will be executed on completion. + * + * @method setRelativePosition + * + * @param {array} position end state to which we interpolate + * @param {transition} transition transition object specifying how object moves to new position + * @param {function} callback zero-argument function to call on observed completion + */ + Draggable.prototype.setRelativePosition = function setRelativePosition(position, transition, callback) { + var currPos = this.getPosition(); + var relativePosition = [currPos[0] + position[0], currPos[1] + position[1]]; + this.setPosition(relativePosition, transition, callback); + }; + + /** + * Transition the element to the desired absolute position via provided transition. + * Callback will be executed on completion. + * + * @method setPosition + * + * @param {array} position end state to which we interpolate + * @param {transition} transition transition object specifying how object moves to new position + * @param {function} callback zero-argument function to call on observed completion + */ + Draggable.prototype.setPosition = function setPosition(position, transition, callback) { + if (this._positionState.isActive()) this._positionState.halt(); + this._positionState.set(position, transition, callback); + }; + + /** + * Set this draggable to respond to user input. + * + * @method activate + * + */ + Draggable.prototype.activate = function activate() { + this._active = true; + }; + + /** + * Set this draggable to ignore user input. + * + * @method deactivate + * + */ + Draggable.prototype.deactivate = function deactivate() { + this._active = false; + }; + + /** + * Switch the input response stage between active and inactive. + * + * @method toggle + * + */ + Draggable.prototype.toggle = function toggle() { + this._active = !this._active; + }; + + /** + * Return render spec for this Modifier, applying to the provided + * target component. This is similar to render() for Surfaces. + * + * @private + * @method modify + * + * @param {Object} target (already rendered) render spec to + * which to apply the transform. + * @return {Object} render spec for this Modifier, including the + * provided target + */ + Draggable.prototype.modify = function modify(target) { + var pos = this.getPosition(); + return { + transform: Transform.translate(pos[0], pos[1]), + target: target + }; + }; + + module.exports = Draggable; +}); + +define('famous/modifiers/Fader',['require','exports','module','../transitions/Transitionable','../core/OptionsManager'],function(require, exports, module) { + var Transitionable = require('../transitions/Transitionable'); + var OptionsManager = require('../core/OptionsManager'); + + /** + * Modifier that allows you to fade the opacity of affected renderables in and out. + * @class Fader + * @constructor + * @param {Object} [options] options configuration object. + * @param {Boolean} [options.cull=false] Stops returning affected renderables up the tree when they're fully faded when true. + * @param {Transition} [options.transition=true] The main transition for showing and hiding. + * @param {Transition} [options.pulseInTransition=true] Controls the transition to a pulsed state when the Fader instance's pulse + * method is called. + * @param {Transition} [options.pulseOutTransition=true]Controls the transition back from a pulsed state when the Fader instance's pulse + * method is called. + * + */ + function Fader(options, startState) { + this.options = Object.create(Fader.DEFAULT_OPTIONS); + this._optionsManager = new OptionsManager(this.options); + + if (options) this.setOptions(options); + + if (!startState) startState = 0; + this.transitionHelper = new Transitionable(startState); + } + + Fader.DEFAULT_OPTIONS = { + cull: false, + transition: true, + pulseInTransition: true, + pulseOutTransition: true + }; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options. See constructor. + */ + Fader.prototype.setOptions = function setOptions(options) { + return this._optionsManager.setOptions(options); + }; + + /** + * Fully displays the Fader instance's associated renderables. + * + * @method show + * @param {Transition} [transition] The transition that coordinates setting to the new state. + * @param {Function} [callback] A callback that executes once you've transitioned to the fully shown state. + */ + Fader.prototype.show = function show(transition, callback) { + transition = transition || this.options.transition; + this.set(1, transition, callback); + }; + + /** + * Fully fades the Fader instance's associated renderables. + * + * @method hide + * @param {Transition} [transition] The transition that coordinates setting to the new state. + * @param {Function} [callback] A callback that executes once you've transitioned to the fully faded state. + */ + Fader.prototype.hide = function hide(transition, callback) { + transition = transition || this.options.transition; + this.set(0, transition, callback); + }; + + /** + * Manually sets the opacity state of the fader to the passed-in one. Executes with an optional + * transition and callback. + * + * @method set + * @param {Number} state A number from zero to one: the amount of opacity you want to set to. + * @param {Transition} [transition] The transition that coordinates setting to the new state. + * @param {Function} [callback] A callback that executes once you've finished executing the pulse. + */ + Fader.prototype.set = function set(state, transition, callback) { + this.halt(); + this.transitionHelper.set(state, transition, callback); + }; + + /** + * Halt the transition + * + * @method halt + */ + Fader.prototype.halt = function halt() { + this.transitionHelper.halt(); + }; + + /** + * Tells you if your Fader instance is above its visibility threshold. + * + * @method isVisible + * @return {Boolean} Whether or not your Fader instance is visible. + */ + Fader.prototype.isVisible = function isVisible() { + return (this.transitionHelper.get() > 0); + }; + + /** + * Return render spec for this Modifier, applying to the provided + * target component. This is similar to render() for Surfaces. + * + * @private + * @method modify + * + * @param {Object} target (already rendered) render spec to + * which to apply the transform. + * @return {Object} render spec for this Modifier, including the + * provided target + */ + Fader.prototype.modify = function modify(target) { + var currOpacity = this.transitionHelper.get(); + if (this.options.cull && !currOpacity) return undefined; + else return {opacity: currOpacity, target: target}; + }; + + module.exports = Fader; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/modifiers/ModifierChain',['require','exports','module'],function(require, exports, module) { + + /** + * A class to add and remove a chain of modifiers + * at a single point in the render tree + * + * @class ModifierChain + * @constructor + */ + function ModifierChain() { + this._chain = []; + if (arguments.length) this.addModifier.apply(this, arguments); + } + + /** + * Add a modifier, or comma separated modifiers, to the modifier chain. + * + * @method addModifier + * + * @param {...Modifier*} varargs args list of Modifiers + */ + ModifierChain.prototype.addModifier = function addModifier(varargs) { + Array.prototype.push.apply(this._chain, arguments); + }; + + /** + * Remove a modifier from the modifier chain. + * + * @method removeModifier + * + * @param {Modifier} modifier + */ + ModifierChain.prototype.removeModifier = function removeModifier(modifier) { + var index = this._chain.indexOf(modifier); + if (index < 0) return; + this._chain.splice(index, 1); + }; + + /** + * Return render spec for this Modifier, applying to the provided + * target component. This is similar to render() for Surfaces. + * + * @private + * @method modify + * + * @param {Object} input (already rendered) render spec to + * which to apply the transform. + * @return {Object} render spec for this Modifier, including the + * provided target + */ + ModifierChain.prototype.modify = function modify(input) { + var chain = this._chain; + var result = input; + for (var i = 0; i < chain.length; i++) { + result = chain[i].modify(result); + } + return result; + }; + + module.exports = ModifierChain; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/modifiers/StateModifier',['require','exports','module','../core/Modifier','../core/Transform','../transitions/Transitionable','../transitions/TransitionableTransform'],function(require, exports, module) { + var Modifier = require('../core/Modifier'); + var Transform = require('../core/Transform'); + var Transitionable = require('../transitions/Transitionable'); + var TransitionableTransform = require('../transitions/TransitionableTransform'); + + /** + * A collection of visual changes to be + * applied to another renderable component, strongly coupled with the state that defines + * those changes. This collection includes a + * transform matrix, an opacity constant, a size, an origin specifier, and an alignment specifier. + * StateModifier objects can be added to any RenderNode or object + * capable of displaying renderables. The StateModifier's children and descendants + * are transformed by the amounts specified in the modifier's properties. + * + * @class StateModifier + * @constructor + * @param {Object} [options] overrides of default options + * @param {Transform} [options.transform] affine transformation matrix + * @param {Number} [options.opacity] + * @param {Array.Number} [options.origin] origin adjustment + * @param {Array.Number} [options.align] align adjustment + * @param {Array.Number} [options.size] size to apply to descendants + * @param {Array.Number} [options.propportions] proportions to apply to descendants + */ + function StateModifier(options) { + this._transformState = new TransitionableTransform(Transform.identity); + this._opacityState = new Transitionable(1); + this._originState = new Transitionable([0, 0]); + this._alignState = new Transitionable([0, 0]); + this._sizeState = new Transitionable([0, 0]); + this._proportionsState = new Transitionable([0, 0]); + + this._modifier = new Modifier({ + transform: this._transformState, + opacity: this._opacityState, + origin: null, + align: null, + size: null, + proportions: null + }); + + this._hasOrigin = false; + this._hasAlign = false; + this._hasSize = false; + this._hasProportions = false; + + if (options) { + if (options.transform) this.setTransform(options.transform); + if (options.opacity !== undefined) this.setOpacity(options.opacity); + if (options.origin) this.setOrigin(options.origin); + if (options.align) this.setAlign(options.align); + if (options.size) this.setSize(options.size); + if (options.proportions) this.setProportions(options.proportions); + } + } + + /** + * Set the transform matrix of this modifier, either statically or + * through a provided Transitionable. + * + * @method setTransform + * + * @param {Transform} transform Transform to transition to. + * @param {Transitionable} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {Function} [callback] callback to call after transition completes + * @return {StateModifier} this + */ + StateModifier.prototype.setTransform = function setTransform(transform, transition, callback) { + this._transformState.set(transform, transition, callback); + return this; + }; + + /** + * Set the opacity of this modifier, either statically or + * through a provided Transitionable. + * + * @method setOpacity + * + * @param {Number} opacity Opacity value to transition to. + * @param {Transitionable} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {Function} callback callback to call after transition completes + * @return {StateModifier} this + */ + StateModifier.prototype.setOpacity = function setOpacity(opacity, transition, callback) { + this._opacityState.set(opacity, transition, callback); + return this; + }; + + /** + * Set the origin of this modifier, either statically or + * through a provided Transitionable. + * + * @method setOrigin + * + * @param {Array.Number} origin two element array with values between 0 and 1. + * @param {Transitionable} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {Function} callback callback to call after transition completes + * @return {StateModifier} this + */ + StateModifier.prototype.setOrigin = function setOrigin(origin, transition, callback) { + if (origin === null) { + if (this._hasOrigin) { + this._modifier.originFrom(null); + this._hasOrigin = false; + } + return this; + } + else if (!this._hasOrigin) { + this._hasOrigin = true; + this._modifier.originFrom(this._originState); + } + this._originState.set(origin, transition, callback); + return this; + }; + + /** + * Set the alignment of this modifier, either statically or + * through a provided Transitionable. + * + * @method setAlign + * + * @param {Array.Number} align two element array with values between 0 and 1. + * @param {Transitionable} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {Function} callback callback to call after transition completes + * @return {StateModifier} this + */ + StateModifier.prototype.setAlign = function setOrigin(align, transition, callback) { + if (align === null) { + if (this._hasAlign) { + this._modifier.alignFrom(null); + this._hasAlign = false; + } + return this; + } + else if (!this._hasAlign) { + this._hasAlign = true; + this._modifier.alignFrom(this._alignState); + } + this._alignState.set(align, transition, callback); + return this; + }; + + /** + * Set the size of this modifier, either statically or + * through a provided Transitionable. + * + * @method setSize + * + * @param {Array.Number} size two element array of [width, height] + * @param {Transitionable} transition object of type {duration: number, curve: + * f[0,1] -> [0,1] or name}. If transition is omitted, change will be + * instantaneous. + * @param {Function} callback callback to call after transition completes + * @return {StateModifier} this + */ + StateModifier.prototype.setSize = function setSize(size, transition, callback) { + if (size === null) { + if (this._hasSize) { + this._modifier.sizeFrom(null); + this._hasSize = false; + } + return this; + } + else if (!this._hasSize) { + this._hasSize = true; + this._modifier.sizeFrom(this._sizeState); + } + this._sizeState.set(size, transition, callback); + return this; + }; + + /** + * Set the proportions of this modifier, either statically or + * through a provided Transitionable. + * + * @method setProportions + * + * @param {Array.Number} proportions two element array with values between 0 and 1. + * @param {Transitionable} transition Valid transitionable object + * @param {Function} callback callback to call after transition completes + * @return {StateModifier} this + */ + StateModifier.prototype.setProportions = function setSize(proportions, transition, callback) { + if (proportions === null) { + if (this._hasProportions) { + this._modifier.proportionsFrom(null); + this._hasProportions = false; + } + return this; + } + else if (!this._hasProportions) { + this._hasProportions = true; + this._modifier.proportionsFrom(this._proportionsState); + } + this._proportionsState.set(proportions, transition, callback); + return this; + }; + + /** + * Stop the transition. + * + * @method halt + */ + StateModifier.prototype.halt = function halt() { + this._transformState.halt(); + this._opacityState.halt(); + this._originState.halt(); + this._alignState.halt(); + this._sizeState.halt(); + this._proportionsState.halt(); + }; + + /** + * Get the current state of the transform matrix component. + * + * @method getTransform + * @return {Object} transform provider object + */ + StateModifier.prototype.getTransform = function getTransform() { + return this._transformState.get(); + }; + + /** + * Get the destination state of the transform component. + * + * @method getFinalTransform + * @return {Transform} transform matrix + */ + StateModifier.prototype.getFinalTransform = function getFinalTransform() { + return this._transformState.getFinal(); + }; + + /** + * Get the current state of the opacity component. + * + * @method getOpacity + * @return {Object} opacity provider object + */ + StateModifier.prototype.getOpacity = function getOpacity() { + return this._opacityState.get(); + }; + + /** + * Get the current state of the origin component. + * + * @method getOrigin + * @return {Object} origin provider object + */ + StateModifier.prototype.getOrigin = function getOrigin() { + return this._hasOrigin ? this._originState.get() : null; + }; + + /** + * Get the current state of the align component. + * + * @method getAlign + * @return {Object} align provider object + */ + StateModifier.prototype.getAlign = function getAlign() { + return this._hasAlign ? this._alignState.get() : null; + }; + + /** + * Get the current state of the size component. + * + * @method getSize + * @return {Object} size provider object + */ + StateModifier.prototype.getSize = function getSize() { + return this._hasSize ? this._sizeState.get() : null; + }; + + /** + * Get the current state of the propportions component. + * + * @method getProportions + * @return {Object} size provider object + */ + StateModifier.prototype.getProportions = function getProportions() { + return this._hasProportions ? this._proportionsState.get() : null; + }; + + /** + * Return render spec for this StateModifier, applying to the provided + * target component. This is similar to render() for Surfaces. + * + * @private + * @method modify + * + * @param {Object} target (already rendered) render spec to + * which to apply the transform. + * @return {Object} render spec for this StateModifier, including the + * provided target + */ + StateModifier.prototype.modify = function modify(target) { + return this._modifier.modify(target); + }; + + module.exports = StateModifier; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ +define('famous/physics/PhysicsEngine',['require','exports','module','../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../core/EventHandler'); + + /** + * The Physics Engine is responsible for mediating bodies with their + * interaction with forces and constraints (agents). Specifically, it + * is responsible for: + * + * - adding and removing bodies + * - updating a body's state over time + * - attaching and detaching agents + * - sleeping upon equillibrium and waking upon excitation + * + * @class PhysicsEngine + * @constructor + * @param options {Object} options + */ + function PhysicsEngine(options) { + this.options = Object.create(PhysicsEngine.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + this._particles = []; //list of managed particles + this._bodies = []; //list of managed bodies + this._agentData = {}; //hash of managed agent data + this._forces = []; //list of Ids of agents that are forces + this._constraints = []; //list of Ids of agents that are constraints + + this._buffer = 0.0; + this._prevTime = now(); + this._isSleeping = false; + this._eventHandler = null; + this._currAgentId = 0; + this._hasBodies = false; + this._eventHandler = null; + } + + /** const */ + var TIMESTEP = 17; + var MIN_TIME_STEP = 1000 / 120; + var MAX_TIME_STEP = 17; + + var now = Date.now; + + // Catalogue of outputted events + var _events = { + start : 'start', + update : 'update', + end : 'end' + }; + + /** + * @property PhysicsEngine.DEFAULT_OPTIONS + * @type Object + * @protected + * @static + */ + PhysicsEngine.DEFAULT_OPTIONS = { + + /** + * The number of iterations the engine takes to resolve constraints + * @attribute constraintSteps + * @type Number + */ + constraintSteps : 1, + + /** + * The energy threshold required for the Physics Engine to update + * @attribute sleepTolerance + * @type Number + */ + sleepTolerance : 1e-7, + + /** + * The maximum velocity magnitude of a physics body + * Range : [0, Infinity] + * @attribute velocityCap + * @type Number + */ + velocityCap : undefined, + + /** + * The maximum angular velocity magnitude of a physics body + * Range : [0, Infinity] + * @attribute angularVelocityCap + * @type Number + */ + angularVelocityCap : undefined + }; + + /** + * Options setter + * + * @method setOptions + * @param opts {Object} + */ + PhysicsEngine.prototype.setOptions = function setOptions(opts) { + for (var key in opts) if (this.options[key]) this.options[key] = opts[key]; + }; + + /** + * Method to add a physics body to the engine. Necessary to update the + * body over time. + * + * @method addBody + * @param body {Body} + * @return body {Body} + */ + PhysicsEngine.prototype.addBody = function addBody(body) { + body._engine = this; + if (body.isBody) { + this._bodies.push(body); + this._hasBodies = true; + } + else this._particles.push(body); + body.on('start', this.wake.bind(this)); + return body; + }; + + /** + * Remove a body from the engine. Detaches body from all forces and + * constraints. + * + * TODO: Fix for in loop + * + * @method removeBody + * @param body {Body} + */ + PhysicsEngine.prototype.removeBody = function removeBody(body) { + var array = (body.isBody) ? this._bodies : this._particles; + var index = array.indexOf(body); + if (index > -1) { + for (var agentKey in this._agentData) { + if (this._agentData.hasOwnProperty(agentKey)) { + this.detachFrom(this._agentData[agentKey].id, body); + } + } + array.splice(index,1); + } + if (this.getBodies().length === 0) this._hasBodies = false; + }; + + function _mapAgentArray(agent) { + if (agent.applyForce) return this._forces; + if (agent.applyConstraint) return this._constraints; + } + + function _attachOne(agent, targets, source) { + if (targets === undefined) targets = this.getParticlesAndBodies(); + if (!(targets instanceof Array)) targets = [targets]; + + agent.on('change', this.wake.bind(this)); + + this._agentData[this._currAgentId] = { + agent : agent, + id : this._currAgentId, + targets : targets, + source : source + }; + + _mapAgentArray.call(this, agent).push(this._currAgentId); + return this._currAgentId++; + } + + /** + * Attaches a force or constraint to a Body. Returns an AgentId of the + * attached agent which can be used to detach the agent. + * + * @method attach + * @param agents {Agent|Array.Agent} A force, constraint, or array of them. + * @param [targets=All] {Body|Array.Body} The Body or Bodies affected by the agent + * @param [source] {Body} The source of the agent + * @return AgentId {Number} + */ + PhysicsEngine.prototype.attach = function attach(agents, targets, source) { + this.wake(); + + if (agents instanceof Array) { + var agentIDs = []; + for (var i = 0; i < agents.length; i++) + agentIDs[i] = _attachOne.call(this, agents[i], targets, source); + return agentIDs; + } + else return _attachOne.call(this, agents, targets, source); + }; + + /** + * Append a body to the targets of a previously defined physics agent. + * + * @method attachTo + * @param agentID {AgentId} The agentId of a previously defined agent + * @param target {Body} The Body affected by the agent + */ + PhysicsEngine.prototype.attachTo = function attachTo(agentID, target) { + _getAgentData.call(this, agentID).targets.push(target); + }; + + /** + * Undoes PhysicsEngine.attach. Removes an agent and its associated + * effect on its affected Bodies. + * + * @method detach + * @param id {AgentId} The agentId of a previously defined agent + */ + PhysicsEngine.prototype.detach = function detach(id) { + // detach from forces/constraints array + var agent = this.getAgent(id); + var agentArray = _mapAgentArray.call(this, agent); + var index = agentArray.indexOf(id); + agentArray.splice(index,1); + + // detach agents array + delete this._agentData[id]; + }; + + /** + * Remove a single Body from a previously defined agent. + * + * @method detach + * @param id {AgentId} The agentId of a previously defined agent + * @param target {Body} The body to remove from the agent + */ + PhysicsEngine.prototype.detachFrom = function detachFrom(id, target) { + var boundAgent = _getAgentData.call(this, id); + if (boundAgent.source === target) this.detach(id); + else { + var targets = boundAgent.targets; + var index = targets.indexOf(target); + if (index > -1) targets.splice(index,1); + } + }; + + /** + * A convenience method to give the Physics Engine a clean slate of + * agents. Preserves all added Body objects. + * + * @method detachAll + */ + PhysicsEngine.prototype.detachAll = function detachAll() { + this._agentData = {}; + this._forces = []; + this._constraints = []; + this._currAgentId = 0; + }; + + function _getAgentData(id) { + return this._agentData[id]; + } + + /** + * Returns the corresponding agent given its agentId. + * + * @method getAgent + * @param id {AgentId} + */ + PhysicsEngine.prototype.getAgent = function getAgent(id) { + return _getAgentData.call(this, id).agent; + }; + + /** + * Returns all particles that are currently managed by the Physics Engine. + * + * @method getParticles + * @return particles {Array.Particles} + */ + PhysicsEngine.prototype.getParticles = function getParticles() { + return this._particles; + }; + + /** + * Returns all bodies, except particles, that are currently managed by the Physics Engine. + * + * @method getBodies + * @return bodies {Array.Bodies} + */ + PhysicsEngine.prototype.getBodies = function getBodies() { + return this._bodies; + }; + + /** + * Returns all bodies that are currently managed by the Physics Engine. + * + * @method getBodies + * @return bodies {Array.Bodies} + */ + PhysicsEngine.prototype.getParticlesAndBodies = function getParticlesAndBodies() { + return this.getParticles().concat(this.getBodies()); + }; + + /** + * Iterates over every Particle and applies a function whose first + * argument is the Particle + * + * @method forEachParticle + * @param fn {Function} Function to iterate over + * @param [dt] {Number} Delta time + */ + PhysicsEngine.prototype.forEachParticle = function forEachParticle(fn, dt) { + var particles = this.getParticles(); + for (var index = 0, len = particles.length; index < len; index++) + fn.call(this, particles[index], dt); + }; + + /** + * Iterates over every Body that isn't a Particle and applies + * a function whose first argument is the Body + * + * @method forEachBody + * @param fn {Function} Function to iterate over + * @param [dt] {Number} Delta time + */ + PhysicsEngine.prototype.forEachBody = function forEachBody(fn, dt) { + if (!this._hasBodies) return; + var bodies = this.getBodies(); + for (var index = 0, len = bodies.length; index < len; index++) + fn.call(this, bodies[index], dt); + }; + + /** + * Iterates over every Body and applies a function whose first + * argument is the Body + * + * @method forEach + * @param fn {Function} Function to iterate over + * @param [dt] {Number} Delta time + */ + PhysicsEngine.prototype.forEach = function forEach(fn, dt) { + this.forEachParticle(fn, dt); + this.forEachBody(fn, dt); + }; + + function _updateForce(index) { + var boundAgent = _getAgentData.call(this, this._forces[index]); + boundAgent.agent.applyForce(boundAgent.targets, boundAgent.source); + } + + function _updateForces() { + for (var index = this._forces.length - 1; index > -1; index--) + _updateForce.call(this, index); + } + + function _updateConstraint(index, dt) { + var boundAgent = this._agentData[this._constraints[index]]; + return boundAgent.agent.applyConstraint(boundAgent.targets, boundAgent.source, dt); + } + + function _updateConstraints(dt) { + var iteration = 0; + while (iteration < this.options.constraintSteps) { + for (var index = this._constraints.length - 1; index > -1; index--) + _updateConstraint.call(this, index, dt); + iteration++; + } + } + + function _updateVelocities(body, dt) { + body.integrateVelocity(dt); + if (this.options.velocityCap) + body.velocity.cap(this.options.velocityCap).put(body.velocity); + } + + function _updateAngularVelocities(body, dt) { + body.integrateAngularMomentum(dt); + body.updateAngularVelocity(); + if (this.options.angularVelocityCap) + body.angularVelocity.cap(this.options.angularVelocityCap).put(body.angularVelocity); + } + + function _updateOrientations(body, dt) { + body.integrateOrientation(dt); + } + + function _updatePositions(body, dt) { + body.integratePosition(dt); + body.emit(_events.update, body); + } + + function _integrate(dt) { + _updateForces.call(this, dt); + this.forEach(_updateVelocities, dt); + this.forEachBody(_updateAngularVelocities, dt); + _updateConstraints.call(this, dt); + this.forEachBody(_updateOrientations, dt); + this.forEach(_updatePositions, dt); + } + + function _getParticlesEnergy() { + var energy = 0.0; + var particleEnergy = 0.0; + this.forEach(function(particle) { + particleEnergy = particle.getEnergy(); + energy += particleEnergy; + }); + return energy; + } + + function _getAgentsEnergy() { + var energy = 0; + for (var id in this._agentData) + energy += this.getAgentEnergy(id); + return energy; + } + + /** + * Calculates the potential energy of an agent, like a spring, by its Id + * + * @method getAgentEnergy + * @param agentId {Number} The attached agent Id + * @return energy {Number} + */ + PhysicsEngine.prototype.getAgentEnergy = function(agentId) { + var agentData = _getAgentData.call(this, agentId); + return agentData.agent.getEnergy(agentData.targets, agentData.source); + }; + + /** + * Calculates the kinetic energy of all Body objects and potential energy + * of all attached agents. + * + * TODO: implement. + * @method getEnergy + * @return energy {Number} + */ + PhysicsEngine.prototype.getEnergy = function getEnergy() { + return _getParticlesEnergy.call(this) + _getAgentsEnergy.call(this); + }; + + /** + * Updates all Body objects managed by the physics engine over the + * time duration since the last time step was called. + * + * @method step + */ + PhysicsEngine.prototype.step = function step() { + if (this.isSleeping()) return; + + //set current frame's time + var currTime = now(); + + //milliseconds elapsed since last frame + var dtFrame = currTime - this._prevTime; + + this._prevTime = currTime; + + if (dtFrame < MIN_TIME_STEP) return; + if (dtFrame > MAX_TIME_STEP) dtFrame = MAX_TIME_STEP; + + //robust integration +// this._buffer += dtFrame; +// while (this._buffer > this._timestep){ +// _integrate.call(this, this._timestep); +// this._buffer -= this._timestep; +// }; +// _integrate.call(this, this._buffer); +// this._buffer = 0.0; + + _integrate.call(this, TIMESTEP); + + this.emit(_events.update, this); + + if (this.getEnergy() < this.options.sleepTolerance) this.sleep(); + }; + + /** + * Tells whether the Physics Engine is sleeping or awake. + * + * @method isSleeping + * @return {Boolean} + */ + PhysicsEngine.prototype.isSleeping = function isSleeping() { + return this._isSleeping; + }; + + /** + * Tells whether the Physics Engine is sleeping or awake. + * + * @method isActive + * @return {Boolean} + */ + PhysicsEngine.prototype.isActive = function isSleeping() { + return !this._isSleeping; + }; + + /** + * Stops the Physics Engine update loop. Emits an 'end' event. + * + * @method sleep + */ + PhysicsEngine.prototype.sleep = function sleep() { + if (this._isSleeping) return; + this.forEach(function(body) { + body.sleep(); + }); + this.emit(_events.end, this); + this._isSleeping = true; + }; + + /** + * Restarts the Physics Engine update loop. Emits an 'start' event. + * + * @method wake + */ + PhysicsEngine.prototype.wake = function wake() { + if (!this._isSleeping) return; + this._prevTime = now(); + this.emit(_events.start, this); + this._isSleeping = false; + }; + + PhysicsEngine.prototype.emit = function emit(type, data) { + if (this._eventHandler === null) return; + this._eventHandler.emit(type, data); + }; + + PhysicsEngine.prototype.on = function on(event, fn) { + if (this._eventHandler === null) this._eventHandler = new EventHandler(); + this._eventHandler.on(event, fn); + }; + + module.exports = PhysicsEngine; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/integrators/SymplecticEuler',['require','exports','module'],function(require, exports, module) { + + /** + * Ordinary Differential Equation (ODE) Integrator. + * Manages updating a physics body's state over time. + * + * p = position, v = velocity, m = mass, f = force, dt = change in time + * + * v <- v + dt * f / m + * p <- p + dt * v + * + * q = orientation, w = angular velocity, L = angular momentum + * + * L <- L + dt * t + * q <- q + dt/2 * q * w + * + * @class SymplecticEuler + * @constructor + * @param {Object} options Options to set + */ + var SymplecticEuler = {}; + + /* + * Updates the velocity of a physics body from its accumulated force. + * v <- v + dt * f / m + * + * @method integrateVelocity + * @param {Body} physics body + * @param {Number} dt delta time + */ + SymplecticEuler.integrateVelocity = function integrateVelocity(body, dt) { + var v = body.velocity; + var w = body.inverseMass; + var f = body.force; + + if (f.isZero()) return; + + v.add(f.mult(dt * w)).put(v); + f.clear(); + }; + + /* + * Updates the position of a physics body from its velocity. + * p <- p + dt * v + * + * @method integratePosition + * @param {Body} physics body + * @param {Number} dt delta time + */ + SymplecticEuler.integratePosition = function integratePosition(body, dt) { + var p = body.position; + var v = body.velocity; + + p.add(v.mult(dt)).put(p); + }; + + /* + * Updates the angular momentum of a physics body from its accumuled torque. + * L <- L + dt * t + * + * @method integrateAngularMomentum + * @param {Body} physics body (except a particle) + * @param {Number} dt delta time + */ + SymplecticEuler.integrateAngularMomentum = function integrateAngularMomentum(body, dt) { + var L = body.angularMomentum; + var t = body.torque; + + if (t.isZero()) return; + + L.add(t.mult(dt)).put(L); + t.clear(); + }; + + /* + * Updates the orientation of a physics body from its angular velocity. + * q <- q + dt/2 * q * w + * + * @method integrateOrientation + * @param {Body} physics body (except a particle) + * @param {Number} dt delta time + */ + SymplecticEuler.integrateOrientation = function integrateOrientation(body, dt) { + var q = body.orientation; + var w = body.angularVelocity; + + if (w.isZero()) return; + q.add(q.multiply(w).scalarMultiply(0.5 * dt)).put(q); +// q.normalize.put(q); + }; + + module.exports = SymplecticEuler; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/bodies/Particle',['require','exports','module','../../math/Vector','../../core/Transform','../../core/EventHandler','../integrators/SymplecticEuler'],function(require, exports, module) { + var Vector = require('../../math/Vector'); + var Transform = require('../../core/Transform'); + var EventHandler = require('../../core/EventHandler'); + var Integrator = require('../integrators/SymplecticEuler'); + + /** + * A point body that is controlled by the Physics Engine. A particle has + * position and velocity states that are updated by the Physics Engine. + * Ultimately, a particle is a special type of modifier, and can be added to + * the Famo.us Scene Graph like any other modifier. + * + * @class Particle + * @uses EventHandler + * @extensionfor Body + * + * @param [options] {Options} An object of configurable options. + * @param [options.position] {Array} The position of the particle. + * @param [options.velocity] {Array} The velocity of the particle. + * @param [options.mass] {Number} The mass of the particle. + */ + function Particle(options) { + options = options || {}; + var defaults = Particle.DEFAULT_OPTIONS; + + // registers + this.position = new Vector(); + this.velocity = new Vector(); + this.force = new Vector(); + + // state variables + this._engine = null; + this._isSleeping = true; + this._eventOutput = null; + + // set scalars + this.mass = (options.mass !== undefined) + ? options.mass + : defaults.mass; + + this.inverseMass = 1 / this.mass; + + // set vectors + this.setPosition(options.position || defaults.position); + this.setVelocity(options.velocity || defaults.velocity); + this.force.set(options.force || [0,0,0]); + + this.transform = Transform.identity.slice(); + + // cached _spec + this._spec = { + size : [true, true], + target : { + transform : this.transform, + origin : [0.5, 0.5], + target : null + } + }; + } + + Particle.DEFAULT_OPTIONS = { + position : [0, 0, 0], + velocity : [0, 0, 0], + mass : 1 + }; + + //Catalogue of outputted events + var _events = { + start : 'start', + update : 'update', + end : 'end' + }; + + // Cached timing function + var now = Date.now; + + /** + * @attribute isBody + * @type Boolean + * @static + */ + Particle.prototype.isBody = false; + + /** + * Determines if particle is active + * + * @method isActive + * @return {Boolean} + */ + Particle.prototype.isActive = function isActive() { + return !this._isSleeping; + }; + + /** + * Stops the particle from updating + * + * @method sleep + */ + Particle.prototype.sleep = function sleep() { + if (this._isSleeping) return; + this.emit(_events.end, this); + this._isSleeping = true; + }; + + /** + * Starts the particle update + * + * @method wake + */ + Particle.prototype.wake = function wake() { + if (!this._isSleeping) return; + this.emit(_events.start, this); + this._isSleeping = false; + this._prevTime = now(); + if (this._engine) this._engine.wake(); + }; + + /** + * Basic setter for position + * + * @method setPosition + * @param position {Array|Vector} + */ + Particle.prototype.setPosition = function setPosition(position) { + this.position.set(position); + }; + + /** + * 1-dimensional setter for position + * + * @method setPosition1D + * @param x {Number} + */ + Particle.prototype.setPosition1D = function setPosition1D(x) { + this.position.x = x; + }; + + /** + * Basic getter function for position + * + * @method getPosition + * @return position {Array} + */ + Particle.prototype.getPosition = function getPosition() { + this._engine.step(); + return this.position.get(); + }; + + /** + * 1-dimensional getter for position + * + * @method getPosition1D + * @return value {Number} + */ + Particle.prototype.getPosition1D = function getPosition1D() { + this._engine.step(); + return this.position.x; + }; + + /** + * Basic setter function for velocity Vector + * + * @method setVelocity + * @function + */ + Particle.prototype.setVelocity = function setVelocity(velocity) { + this.velocity.set(velocity); + if (!(velocity[0] === 0 && velocity[1] === 0 && velocity[2] === 0)) + this.wake(); + }; + + /** + * 1-dimensional setter for velocity + * + * @method setVelocity1D + * @param x {Number} + */ + Particle.prototype.setVelocity1D = function setVelocity1D(x) { + this.velocity.x = x; + if (x !== 0) this.wake(); + }; + + /** + * Basic getter function for velocity Vector + * + * @method getVelocity + * @return velocity {Array} + */ + Particle.prototype.getVelocity = function getVelocity() { + return this.velocity.get(); + }; + + /** + * Basic setter function for force Vector + * + * @method setForce + * @return force {Array} + */ + Particle.prototype.setForce = function setForce(force) { + this.force.set(force); + this.wake(); + }; + + /** + * 1-dimensional getter for velocity + * + * @method getVelocity1D + * @return velocity {Number} + */ + Particle.prototype.getVelocity1D = function getVelocity1D() { + return this.velocity.x; + }; + + /** + * Basic setter function for mass quantity + * + * @method setMass + * @param mass {Number} mass + */ + Particle.prototype.setMass = function setMass(mass) { + this.mass = mass; + this.inverseMass = 1 / mass; + }; + + /** + * Basic getter function for mass quantity + * + * @method getMass + * @return mass {Number} + */ + Particle.prototype.getMass = function getMass() { + return this.mass; + }; + + /** + * Reset position and velocity + * + * @method reset + * @param position {Array|Vector} + * @param velocity {Array|Vector} + */ + Particle.prototype.reset = function reset(position, velocity) { + this.setPosition(position || [0,0,0]); + this.setVelocity(velocity || [0,0,0]); + }; + + /** + * Add force vector to existing internal force Vector + * + * @method applyForce + * @param force {Vector} + */ + Particle.prototype.applyForce = function applyForce(force) { + if (force.isZero()) return; + this.force.add(force).put(this.force); + this.wake(); + }; + + /** + * Add impulse (change in velocity) Vector to this Vector's velocity. + * + * @method applyImpulse + * @param impulse {Vector} + */ + Particle.prototype.applyImpulse = function applyImpulse(impulse) { + if (impulse.isZero()) return; + var velocity = this.velocity; + velocity.add(impulse.mult(this.inverseMass)).put(velocity); + }; + + /** + * Update a particle's velocity from its force accumulator + * + * @method integrateVelocity + * @param dt {Number} Time differential + */ + Particle.prototype.integrateVelocity = function integrateVelocity(dt) { + Integrator.integrateVelocity(this, dt); + }; + + /** + * Update a particle's position from its velocity + * + * @method integratePosition + * @param dt {Number} Time differential + */ + Particle.prototype.integratePosition = function integratePosition(dt) { + Integrator.integratePosition(this, dt); + }; + + /** + * Update the position and velocity of the particle + * + * @method _integrate + * @protected + * @param dt {Number} Time differential + */ + Particle.prototype._integrate = function _integrate(dt) { + this.integrateVelocity(dt); + this.integratePosition(dt); + }; + + /** + * Get kinetic energy of the particle. + * + * @method getEnergy + * @function + */ + Particle.prototype.getEnergy = function getEnergy() { + return 0.5 * this.mass * this.velocity.normSquared(); + }; + + /** + * Generate transform from the current position state + * + * @method getTransform + * @return Transform {Transform} + */ + Particle.prototype.getTransform = function getTransform() { + this._engine.step(); + + var position = this.position; + var transform = this.transform; + + transform[12] = position.x; + transform[13] = position.y; + transform[14] = position.z; + return transform; + }; + + /** + * The modify interface of a Modifier + * + * @method modify + * @param target {Spec} + * @return Spec {Spec} + */ + Particle.prototype.modify = function modify(target) { + var _spec = this._spec.target; + _spec.transform = this.getTransform(); + _spec.target = target; + return this._spec; + }; + + // private + function _createEventOutput() { + this._eventOutput = new EventHandler(); + this._eventOutput.bindThis(this); + EventHandler.setOutputHandler(this, this._eventOutput); + } + + Particle.prototype.emit = function emit(type, data) { + if (!this._eventOutput) return; + this._eventOutput.emit(type, data); + }; + + Particle.prototype.on = function on() { + _createEventOutput.call(this); + return this.on.apply(this, arguments); + }; + + Particle.prototype.removeListener = function removeListener() { + _createEventOutput.call(this); + return this.removeListener.apply(this, arguments); + }; + + Particle.prototype.pipe = function pipe() { + _createEventOutput.call(this); + return this.pipe.apply(this, arguments); + }; + + Particle.prototype.unpipe = function unpipe() { + _createEventOutput.call(this); + return this.unpipe.apply(this, arguments); + }; + + module.exports = Particle; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/bodies/Body',['require','exports','module','./Particle','../../core/Transform','../../math/Vector','../../math/Quaternion','../../math/Matrix','../integrators/SymplecticEuler'],function(require, exports, module) { + var Particle = require('./Particle'); + var Transform = require('../../core/Transform'); + var Vector = require('../../math/Vector'); + var Quaternion = require('../../math/Quaternion'); + var Matrix = require('../../math/Matrix'); + var Integrator = require('../integrators/SymplecticEuler'); + + /** + * A unit controlled by the physics engine which extends the zero-dimensional + * Particle to include geometry. In addition to maintaining the state + * of a Particle its state includes orientation, angular velocity + * and angular momentum and responds to torque forces. + * + * @class Body + * @extends Particle + * @constructor + */ + function Body(options) { + Particle.call(this, options); + options = options || {}; + + this.orientation = new Quaternion(); + this.angularVelocity = new Vector(); + this.angularMomentum = new Vector(); + this.torque = new Vector(); + + if (options.orientation) this.orientation.set(options.orientation); + if (options.angularVelocity) this.angularVelocity.set(options.angularVelocity); + if (options.angularMomentum) this.angularMomentum.set(options.angularMomentum); + if (options.torque) this.torque.set(options.torque); + + this.angularVelocity.w = 0; //quaternify the angular velocity + this.setMomentsOfInertia(); + + // registers + this.pWorld = new Vector(); //placeholder for world space position + } + + Body.DEFAULT_OPTIONS = Particle.DEFAULT_OPTIONS; + Body.DEFAULT_OPTIONS.orientation = [0, 0, 0, 1]; + Body.DEFAULT_OPTIONS.angularVelocity = [0, 0, 0]; + + Body.prototype = Object.create(Particle.prototype); + Body.prototype.constructor = Body; + + Body.prototype.isBody = true; + + Body.prototype.setMass = function setMass() { + Particle.prototype.setMass.apply(this, arguments); + this.setMomentsOfInertia(); + }; + + /** + * Setter for moment of inertia, which is necessary to give proper + * angular inertia depending on the geometry of the body. + * + * @method setMomentsOfInertia + */ + Body.prototype.setMomentsOfInertia = function setMomentsOfInertia() { + this.inertia = new Matrix(); + this.inverseInertia = new Matrix(); + }; + + /** + * Update the angular velocity from the angular momentum state. + * + * @method updateAngularVelocity + */ + Body.prototype.updateAngularVelocity = function updateAngularVelocity() { + this.angularVelocity.set(this.inverseInertia.vectorMultiply(this.angularMomentum)); + }; + + /** + * Determine world coordinates from the local coordinate system. Useful + * if the Body has rotated in space. + * + * @method toWorldCoordinates + * @param localPosition {Vector} local coordinate vector + * @return global coordinate vector {Vector} + */ + Body.prototype.toWorldCoordinates = function toWorldCoordinates(localPosition) { + return this.pWorld.set(this.orientation.rotateVector(localPosition)); + }; + + /** + * Calculates the kinetic and intertial energy of a body. + * + * @method getEnergy + * @return energy {Number} + */ + Body.prototype.getEnergy = function getEnergy() { + return Particle.prototype.getEnergy.call(this) + + 0.5 * this.inertia.vectorMultiply(this.angularVelocity).dot(this.angularVelocity); + }; + + /** + * Extends Particle.reset to reset orientation, angular velocity + * and angular momentum. + * + * @method reset + * @param [p] {Array|Vector} position + * @param [v] {Array|Vector} velocity + * @param [q] {Array|Quaternion} orientation + * @param [L] {Array|Vector} angular momentum + */ + Body.prototype.reset = function reset(p, v, q, L) { + Particle.prototype.reset.call(this, p, v); + this.angularVelocity.clear(); + this.setOrientation(q || [1,0,0,0]); + this.setAngularMomentum(L || [0,0,0]); + }; + + /** + * Setter for orientation + * + * @method setOrientation + * @param q {Array|Quaternion} orientation + */ + Body.prototype.setOrientation = function setOrientation(q) { + this.orientation.set(q); + }; + + /** + * Setter for angular velocity + * + * @method setAngularVelocity + * @param w {Array|Vector} angular velocity + */ + Body.prototype.setAngularVelocity = function setAngularVelocity(w) { + this.wake(); + this.angularVelocity.set(w); + }; + + /** + * Setter for angular momentum + * + * @method setAngularMomentum + * @param L {Array|Vector} angular momentum + */ + Body.prototype.setAngularMomentum = function setAngularMomentum(L) { + this.wake(); + this.angularMomentum.set(L); + }; + + /** + * Extends Particle.applyForce with an optional argument + * to apply the force at an off-centered location, resulting in a torque. + * + * @method applyForce + * @param force {Vector} force + * @param [location] {Vector} off-center location on the body + */ + Body.prototype.applyForce = function applyForce(force, location) { + Particle.prototype.applyForce.call(this, force); + if (location !== undefined) this.applyTorque(location.cross(force)); + }; + + /** + * Applied a torque force to a body, inducing a rotation. + * + * @method applyTorque + * @param torque {Vector} torque + */ + Body.prototype.applyTorque = function applyTorque(torque) { + this.wake(); + this.torque.set(this.torque.add(torque)); + }; + + /** + * Extends Particle.getTransform to include a rotational component + * derived from the particle's orientation. + * + * @method getTransform + * @return transform {Transform} + */ + Body.prototype.getTransform = function getTransform() { + return Transform.thenMove( + this.orientation.getTransform(), + Transform.getTranslate(Particle.prototype.getTransform.call(this)) + ); + }; + + /** + * Extends Particle._integrate to also update the rotational states + * of the body. + * + * @method getTransform + * @protected + * @param dt {Number} delta time + */ + Body.prototype._integrate = function _integrate(dt) { + Particle.prototype._integrate.call(this, dt); + this.integrateAngularMomentum(dt); + this.updateAngularVelocity(dt); + this.integrateOrientation(dt); + }; + + /** + * Updates the angular momentum via the its integrator. + * + * @method integrateAngularMomentum + * @param dt {Number} delta time + */ + Body.prototype.integrateAngularMomentum = function integrateAngularMomentum(dt) { + Integrator.integrateAngularMomentum(this, dt); + }; + + /** + * Updates the orientation via the its integrator. + * + * @method integrateOrientation + * @param dt {Number} delta time + */ + Body.prototype.integrateOrientation = function integrateOrientation(dt) { + Integrator.integrateOrientation(this, dt); + }; + + module.exports = Body; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/bodies/Circle',['require','exports','module','./Body','../../math/Matrix'],function(require, exports, module) { + var Body = require('./Body'); + var Matrix = require('../../math/Matrix'); + + /** + * Implements a circle, or spherical, geometry for a Body with + * radius. + * + * @class Circle + * @extends Body + * @constructor + */ + function Circle(options) { + options = options || {}; + this.setRadius(options.radius || 0); + Body.call(this, options); + } + + Circle.prototype = Object.create(Body.prototype); + Circle.prototype.constructor = Circle; + + /** + * Basic setter for radius. + * @method setRadius + * @param r {Number} radius + */ + Circle.prototype.setRadius = function setRadius(r) { + this.radius = r; + this.size = [2*this.radius, 2*this.radius]; + this.setMomentsOfInertia(); + }; + + Circle.prototype.setMomentsOfInertia = function setMomentsOfInertia() { + var m = this.mass; + var r = this.radius; + + this.inertia = new Matrix([ + [0.25 * m * r * r, 0, 0], + [0, 0.25 * m * r * r, 0], + [0, 0, 0.5 * m * r * r] + ]); + + this.inverseInertia = new Matrix([ + [4 / (m * r * r), 0, 0], + [0, 4 / (m * r * r), 0], + [0, 0, 2 / (m * r * r)] + ]); + }; + + module.exports = Circle; + +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/bodies/Rectangle',['require','exports','module','./Body','../../math/Matrix'],function(require, exports, module) { + var Body = require('./Body'); + var Matrix = require('../../math/Matrix'); + + /** + * Implements a rectangular geometry for an Body with + * size = [width, height]. + * + * @class Rectangle + * @extends Body + * @constructor + */ + function Rectangle(options) { + options = options || {}; + this.size = options.size || [0,0]; + Body.call(this, options); + } + + Rectangle.prototype = Object.create(Body.prototype); + Rectangle.prototype.constructor = Rectangle; + + /** + * Basic setter for size. + * @method setSize + * @param size {Array} size = [width, height] + */ + Rectangle.prototype.setSize = function setSize(size) { + this.size = size; + this.setMomentsOfInertia(); + }; + + Rectangle.prototype.setMomentsOfInertia = function setMomentsOfInertia() { + var m = this.mass; + var w = this.size[0]; + var h = this.size[1]; + + this.inertia = new Matrix([ + [m * h * h / 12, 0, 0], + [0, m * w * w / 12, 0], + [0, 0, m * (w * w + h * h) / 12] + ]); + + this.inverseInertia = new Matrix([ + [12 / (m * h * h), 0, 0], + [0, 12 / (m * w * w), 0], + [0, 0, 12 / (m * (w * w + h * h))] + ]); + }; + + module.exports = Rectangle; + +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Constraint',['require','exports','module','../../core/EventHandler'],function(require, exports, module) { + var EventHandler = require('../../core/EventHandler'); + + /** + * Allows for two circular bodies to collide and bounce off each other. + * + * @class Constraint + * @constructor + * @uses EventHandler + * @param options {Object} + */ + function Constraint() { + this.options = this.options || {}; + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); + } + + /* + * Setter for options. + * + * @method setOptions + * @param options {Objects} + */ + Constraint.prototype.setOptions = function setOptions(options) { + this._eventOutput.emit('change', options); + }; + + /** + * Adds an impulse to a physics body's velocity due to the constraint + * + * @method applyConstraint + */ + Constraint.prototype.applyConstraint = function applyConstraint() {}; + + /** + * Getter for energy + * + * @method getEnergy + * @return energy {Number} + */ + Constraint.prototype.getEnergy = function getEnergy() { + return 0.0; + }; + + module.exports = Constraint; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Collision',['require','exports','module','./Constraint','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Vector = require('../../math/Vector'); + + /** + * Allows for two circular bodies to collide and bounce off each other. + * + * @class Collision + * @constructor + * @extends Constraint + * @param {Options} [options] An object of configurable options. + * @param {Number} [options.restitution] The energy ratio lost in a collision (0 = stick, 1 = elastic) Range : [0, 1] + * @param {Number} [options.drift] Baumgarte stabilization parameter. Makes constraints "loosely" (0) or "tightly" (1) enforced. Range : [0, 1] + * @param {Number} [options.slop] Amount of penetration in pixels to ignore before collision event triggers + * + */ + function Collision(options) { + this.options = Object.create(Collision.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.normal = new Vector(); + this.pDiff = new Vector(); + this.vDiff = new Vector(); + this.impulse1 = new Vector(); + this.impulse2 = new Vector(); + + Constraint.call(this); + } + + Collision.prototype = Object.create(Constraint.prototype); + Collision.prototype.constructor = Collision; + + Collision.DEFAULT_OPTIONS = { + restitution : 0.5, + drift : 0.5, + slop : 0 + }; + + function _normalVelocity(particle1, particle2) { + return particle1.velocity.dot(particle2.velocity); + } + + /* + * Setter for options. + * + * @method setOptions + * @param options {Objects} + */ + Collision.prototype.setOptions = function setOptions(options) { + for (var key in options) this.options[key] = options[key]; + }; + + /** + * Adds an impulse to a physics body's velocity due to the constraint + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply the constraint to + * @param source {Body} The source of the constraint + * @param dt {Number} Delta time + */ + Collision.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + if (source === undefined) return; + + var v1 = source.velocity; + var p1 = source.position; + var w1 = source.inverseMass; + var r1 = source.radius; + + var options = this.options; + var drift = options.drift; + var slop = -options.slop; + var restitution = options.restitution; + + var n = this.normal; + var pDiff = this.pDiff; + var vDiff = this.vDiff; + var impulse1 = this.impulse1; + var impulse2 = this.impulse2; + + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + + if (target === source) continue; + + var v2 = target.velocity; + var p2 = target.position; + var w2 = target.inverseMass; + var r2 = target.radius; + + pDiff.set(p2.sub(p1)); + vDiff.set(v2.sub(v1)); + + var dist = pDiff.norm(); + var overlap = dist - (r1 + r2); + var effMass = 1/(w1 + w2); + var gamma = 0; + + if (overlap < 0) { + + n.set(pDiff.normalize()); + + if (this._eventOutput) { + var collisionData = { + target : target, + source : source, + overlap : overlap, + normal : n + }; + + this._eventOutput.emit('preCollision', collisionData); + this._eventOutput.emit('collision', collisionData); + } + + var lambda = (overlap <= slop) + ? ((1 + restitution) * n.dot(vDiff) + drift/dt * (overlap - slop)) / (gamma + dt/effMass) + : ((1 + restitution) * n.dot(vDiff)) / (gamma + dt/effMass); + + n.mult(dt*lambda).put(impulse1); + impulse1.mult(-1).put(impulse2); + + source.applyImpulse(impulse1); + target.applyImpulse(impulse2); + + //source.setPosition(p1.add(n.mult(overlap/2))); + //target.setPosition(p2.sub(n.mult(overlap/2))); + + if (this._eventOutput) this._eventOutput.emit('postCollision', collisionData); + + } + } + }; + + module.exports = Collision; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Curve',['require','exports','module','./Constraint','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Vector = require('../../math/Vector'); + + /** + * A constraint that keeps a physics body on a given implicit curve + * regardless of other physical forces are applied to it. + * + * A curve constraint is two surface constraints in disguise, as a curve is + * the intersection of two surfaces, and is essentially constrained to both + * + * @class Curve + * @constructor + * @extends Constraint + * @param {Options} [options] An object of configurable options. + * @param {Function} [options.equation] An implicitly defined surface f(x,y,z) = 0 that body is constrained to e.g. function(x,y,z) { x*x + y*y - r*r } corresponds to a circle of radius r pixels + * @param {Function} [options.plane] An implicitly defined second surface that the body is constrained to + * @param {Number} [options.period] The spring-like reaction when the constraint is violated + * @param {Number} [options.number] The damping-like reaction when the constraint is violated + */ + function Curve(options) { + this.options = Object.create(Curve.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.J = new Vector(); + this.impulse = new Vector(); + + Constraint.call(this); + } + + Curve.prototype = Object.create(Constraint.prototype); + Curve.prototype.constructor = Curve; + + /** @const */ var epsilon = 1e-7; + /** @const */ var pi = Math.PI; + + Curve.DEFAULT_OPTIONS = { + equation : function(x,y,z) { + return 0; + }, + plane : function(x,y,z) { + return z; + }, + period : 0, + dampingRatio : 0 + }; + + /** + * Basic options setter + * + * @method setOptions + * @param options {Objects} + */ + Curve.prototype.setOptions = function setOptions(options) { + for (var key in options) this.options[key] = options[key]; + }; + + /** + * Adds a curve impulse to a physics body. + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply force to. + * @param source {Body} Not applicable + * @param dt {Number} Delta time + */ + Curve.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var options = this.options; + var impulse = this.impulse; + var J = this.J; + + var f = options.equation; + var g = options.plane; + var dampingRatio = options.dampingRatio; + var period = options.period; + + for (var i = 0; i < targets.length; i++) { + var body = targets[i]; + + var v = body.velocity; + var p = body.position; + var m = body.mass; + + var gamma; + var beta; + + if (period === 0) { + gamma = 0; + beta = 1; + } + else { + var c = 4 * m * pi * dampingRatio / period; + var k = 4 * m * pi * pi / (period * period); + + gamma = 1 / (c + dt*k); + beta = dt*k / (c + dt*k); + } + + var x = p.x; + var y = p.y; + var z = p.z; + + var f0 = f(x, y, z); + var dfx = (f(x + epsilon, y, z) - f0) / epsilon; + var dfy = (f(x, y + epsilon, z) - f0) / epsilon; + var dfz = (f(x, y, z + epsilon) - f0) / epsilon; + + var g0 = g(x, y, z); + var dgx = (g(x + epsilon, y, z) - g0) / epsilon; + var dgy = (g(x, y + epsilon, z) - g0) / epsilon; + var dgz = (g(x, y, z + epsilon) - g0) / epsilon; + + J.setXYZ(dfx + dgx, dfy + dgy, dfz + dgz); + + var antiDrift = beta/dt * (f0 + g0); + var lambda = -(J.dot(v) + antiDrift) / (gamma + dt * J.normSquared() / m); + + impulse.set(J.mult(dt*lambda)); + body.applyImpulse(impulse); + } + }; + + module.exports = Curve; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Distance',['require','exports','module','./Constraint','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Vector = require('../../math/Vector'); + + /** + * A constraint that keeps a physics body a given distance away from a given + * anchor, or another attached body. + * + * + * @class Distance + * @constructor + * @extends Constraint + * @param {Options} [options] An object of configurable options. + * @param {Array} [options.anchor] The location of the anchor + * @param {Number} [options.length] The amount of distance from the anchor the constraint should enforce + * @param {Number} [options.minLength] The minimum distance before the constraint is activated. Use this property for a "rope" effect. + * @param {Number} [options.period] The spring-like reaction when the constraint is broken. + * @param {Number} [options.dampingRatio] The damping-like reaction when the constraint is broken. + * + */ + function Distance(options) { + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.impulse = new Vector(); + this.normal = new Vector(); + this.diffP = new Vector(); + this.diffV = new Vector(); + + Constraint.call(this); + } + + Distance.prototype = Object.create(Constraint.prototype); + Distance.prototype.constructor = Distance; + + Distance.DEFAULT_OPTIONS = { + anchor : null, + length : 0, + minLength : 0, + period : 0, + dampingRatio : 0 + }; + + /** @const */ var pi = Math.PI; + + /** + * Basic options setter + * + * @method setOptions + * @param options {Objects} + */ + Distance.prototype.setOptions = function setOptions(options) { + if (options.anchor) { + if (options.anchor.position instanceof Vector) this.options.anchor = options.anchor.position; + if (options.anchor instanceof Vector) this.options.anchor = options.anchor; + if (options.anchor instanceof Array) this.options.anchor = new Vector(options.anchor); + } + if (options.length !== undefined) this.options.length = options.length; + if (options.dampingRatio !== undefined) this.options.dampingRatio = options.dampingRatio; + if (options.period !== undefined) this.options.period = options.period; + if (options.minLength !== undefined) this.options.minLength = options.minLength; + }; + + function _calcError(impulse, body) { + return body.mass * impulse.norm(); + } + + /** + * Set the anchor position + * + * @method setOptions + * @param anchor {Array} + */ + Distance.prototype.setAnchor = function setAnchor(anchor) { + if (!this.options.anchor) this.options.anchor = new Vector(); + this.options.anchor.set(anchor); + }; + + /** + * Adds an impulse to a physics body's velocity due to the constraint + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply the constraint to + * @param source {Body} The source of the constraint + * @param dt {Number} Delta time + */ + Distance.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var n = this.normal; + var diffP = this.diffP; + var diffV = this.diffV; + var impulse = this.impulse; + var options = this.options; + + var dampingRatio = options.dampingRatio; + var period = options.period; + var minLength = options.minLength; + + var p2; + var w2; + + if (source) { + var v2 = source.velocity; + p2 = source.position; + w2 = source.inverseMass; + } + else { + p2 = this.options.anchor; + w2 = 0; + } + + var length = this.options.length; + + for (var i = 0; i < targets.length; i++) { + var body = targets[i]; + + var v1 = body.velocity; + var p1 = body.position; + var w1 = body.inverseMass; + + diffP.set(p1.sub(p2)); + n.set(diffP.normalize()); + + var dist = diffP.norm() - length; + + //rope effect + if (Math.abs(dist) < minLength) return; + + if (source) diffV.set(v1.sub(v2)); + else diffV.set(v1); + + var effMass = 1 / (w1 + w2); + var gamma; + var beta; + + if (period === 0) { + gamma = 0; + beta = 1; + } + else { + var c = 4 * effMass * pi * dampingRatio / period; + var k = 4 * effMass * pi * pi / (period * period); + + gamma = 1 / (c + dt*k); + beta = dt*k / (c + dt*k); + } + + var antiDrift = beta/dt * dist; + var lambda = -(n.dot(diffV) + antiDrift) / (gamma + dt/effMass); + + impulse.set(n.mult(dt*lambda)); + body.applyImpulse(impulse); + + if (source) source.applyImpulse(impulse.mult(-1)); + } + }; + + module.exports = Distance; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Snap',['require','exports','module','./Constraint','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Vector = require('../../math/Vector'); + + /** + * A spring constraint is like a spring force, except that it is always + * numerically stable (even for low periods), at the expense of introducing + * damping (even with dampingRatio set to 0). + * + * Use this if you need fast spring-like behavior, e.g., snapping + * + * @class Snap + * @constructor + * @extends Constraint + * @param {Options} [options] An object of configurable options. + * @param {Number} [options.period] The amount of time in milliseconds taken for one complete oscillation when there is no damping. Range : [150, Infinity] + * @param {Number} [options.dampingRatio] Additional damping of the spring. Range : [0, 1]. At 0 this spring will still be damped, at 1 the spring will be critically damped (the spring will never oscillate) + * @param {Number} [options.length] The rest length of the spring. Range: [0, Infinity]. + * @param {Array} [options.anchor] The location of the spring's anchor, if not another physics body. + * + */ + function Snap(options) { + Constraint.call(this); + + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.pDiff = new Vector(); + this.vDiff = new Vector(); + this.impulse1 = new Vector(); + this.impulse2 = new Vector(); + } + + Snap.prototype = Object.create(Constraint.prototype); + Snap.prototype.constructor = Snap; + + Snap.DEFAULT_OPTIONS = { + period : 300, + dampingRatio : 0.1, + length : 0, + anchor : undefined + }; + + /** const */ var pi = Math.PI; + + /** + * Basic options setter + * + * @method setOptions + * @param options {Objects} options + */ + Snap.prototype.setOptions = function setOptions(options) { + if (options.anchor !== undefined) { + if (options.anchor instanceof Vector) this.options.anchor = options.anchor; + if (options.anchor.position instanceof Vector) this.options.anchor = options.anchor.position; + if (options.anchor instanceof Array) this.options.anchor = new Vector(options.anchor); + } + if (options.length !== undefined) this.options.length = options.length; + if (options.dampingRatio !== undefined) this.options.dampingRatio = options.dampingRatio; + if (options.period !== undefined) this.options.period = options.period; + Constraint.prototype.setOptions.call(this, options); + }; + + /** + * Calculates energy of spring + * + * @method getEnergy + * @param targets {Body} target physics body + * @param source {Body} source physics body + * @return energy {Number} + */ + Snap.prototype.getEnergy = function getEnergy(targets, source) { + var options = this.options; + var restLength = options.length; + var anchor = options.anchor || source.position; + var strength = Math.pow(2 * pi / options.period, 2); + + var energy = 0.0; + for (var i = 0; i < targets.length; i++){ + var target = targets[i]; + var dist = anchor.sub(target.position).norm() - restLength; + energy += 0.5 * strength * dist * dist; + } + return energy; + }; + + /** + * Adds a spring impulse to a physics body's velocity due to the constraint + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply the constraint to + * @param source {Body} The source of the constraint + * @param dt {Number} Delta time + */ + Snap.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var options = this.options; + var pDiff = this.pDiff; + var vDiff = this.vDiff; + var impulse1 = this.impulse1; + var impulse2 = this.impulse2; + var length = options.length; + var anchor = options.anchor || source.position; + var period = options.period; + var dampingRatio = options.dampingRatio; + + for (var i = 0; i < targets.length ; i++) { + var target = targets[i]; + + var p1 = target.position; + var v1 = target.velocity; + var m1 = target.mass; + var w1 = target.inverseMass; + + pDiff.set(p1.sub(anchor)); + var dist = pDiff.norm() - length; + var effMass; + + if (source) { + var w2 = source.inverseMass; + var v2 = source.velocity; + vDiff.set(v1.sub(v2)); + effMass = 1 / (w1 + w2); + } + else { + vDiff.set(v1); + effMass = m1; + } + + var gamma; + var beta; + + if (this.options.period === 0) { + gamma = 0; + beta = 1; + } + else { + var k = 4 * effMass * pi * pi / (period * period); + var c = 4 * effMass * pi * dampingRatio / period; + + beta = dt * k / (c + dt * k); + gamma = 1 / (c + dt*k); + } + + var antiDrift = beta/dt * dist; + pDiff.normalize(-antiDrift) + .sub(vDiff) + .mult(dt / (gamma + dt/effMass)) + .put(impulse1); + + // var n = new Vector(); + // n.set(pDiff.normalize()); + // var lambda = -(n.dot(vDiff) + antiDrift) / (gamma + dt/effMass); + // impulse2.set(n.mult(dt*lambda)); + + target.applyImpulse(impulse1); + + if (source) { + impulse1.mult(-1).put(impulse2); + source.applyImpulse(impulse2); + } + } + }; + + module.exports = Snap; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Surface',['require','exports','module','./Constraint','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Vector = require('../../math/Vector'); + + /** + * A constraint that keeps a physics body on a given implicit surface + * regardless of other physical forces are applied to it. + * + * @class Surface + * @constructor + * @extends Constraint + * @param {Options} [options] An object of configurable options. + * @param {Function} [options.equation] An implicitly defined surface f(x,y,z) = 0 that body is constrained to e.g. function(x,y,z) { x*x + y*y + z*z - r*r } corresponds to a sphere of radius r pixels. + * @param {Number} [options.period] The spring-like reaction when the constraint is violated. + * @param {Number} [options.dampingRatio] The damping-like reaction when the constraint is violated. + */ + function Surface(options) { + this.options = Object.create(Surface.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + this.J = new Vector(); + this.impulse = new Vector(); + + Constraint.call(this); + } + + Surface.prototype = Object.create(Constraint.prototype); + Surface.prototype.constructor = Surface; + + Surface.DEFAULT_OPTIONS = { + equation : undefined, + period : 0, + dampingRatio : 0 + }; + + /** @const */ var epsilon = 1e-7; + /** @const */ var pi = Math.PI; + + /** + * Basic options setter + * + * @method setOptions + * @param options {Objects} + */ + Surface.prototype.setOptions = function setOptions(options) { + for (var key in options) this.options[key] = options[key]; + }; + + /** + * Adds a surface impulse to a physics body. + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply force to. + * @param source {Body} Not applicable + * @param dt {Number} Delta time + */ + Surface.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var impulse = this.impulse; + var J = this.J; + var options = this.options; + + var f = options.equation; + var dampingRatio = options.dampingRatio; + var period = options.period; + + for (var i = 0; i < targets.length; i++) { + var particle = targets[i]; + + var v = particle.velocity; + var p = particle.position; + var m = particle.mass; + + var gamma; + var beta; + + if (period === 0) { + gamma = 0; + beta = 1; + } + else { + var c = 4 * m * pi * dampingRatio / period; + var k = 4 * m * pi * pi / (period * period); + + gamma = 1 / (c + dt*k); + beta = dt*k / (c + dt*k); + } + + var x = p.x; + var y = p.y; + var z = p.z; + + var f0 = f(x, y, z); + var dfx = (f(x + epsilon, p, p) - f0) / epsilon; + var dfy = (f(x, y + epsilon, p) - f0) / epsilon; + var dfz = (f(x, y, p + epsilon) - f0) / epsilon; + J.setXYZ(dfx, dfy, dfz); + + var antiDrift = beta/dt * f0; + var lambda = -(J.dot(v) + antiDrift) / (gamma + dt * J.normSquared() / m); + + impulse.set(J.mult(dt*lambda)); + particle.applyImpulse(impulse); + } + }; + + module.exports = Surface; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Wall',['require','exports','module','./Constraint','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Vector = require('../../math/Vector'); + + /** + * A wall describes an infinite two-dimensional plane that physics bodies + * can collide with. To define a wall, you must give it a distance (from + * the center of the physics engine's origin, and a normal defining the plane + * of the wall. + * + * (wall) + * | + * | (normal) (origin) + * | ---> * + * | + * | (distance) + * ................... + * (100px) + * + * e.g., Wall({normal : [1,0,0], distance : 100}) + * would be a wall 100 pixels to the left, whose normal points right + * + * @class Wall + * @constructor + * @extends Constraint + * @param {Options} [options] An object of configurable options. + * @param {Number} [options.restitution] The energy ratio lost in a collision (0 = stick, 1 = elastic). Range : [0, 1] + * @param {Number} [options.drift] Baumgarte stabilization parameter. Makes constraints "loosely" (0) or "tightly" (1) enforced. Range : [0, 1] + * @param {Number} [options.slop] Amount of penetration in pixels to ignore before collision event triggers. + * @param {Array} [options.normal] The normal direction to the wall. + * @param {Number} [options.distance] The distance from the origin that the wall is placed. + * @param {onContact} [options.onContact] How to handle collision against the wall. + * + */ + function Wall(options) { + this.options = Object.create(Wall.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.diff = new Vector(); + this.impulse = new Vector(); + + Constraint.call(this); + } + + Wall.prototype = Object.create(Constraint.prototype); + Wall.prototype.constructor = Wall; + + /** + * @property Wall.ON_CONTACT + * @type Object + * @protected + * @static + */ + Wall.ON_CONTACT = { + + /** + * Physical bodies bounce off the wall + * @attribute REFLECT + */ + REFLECT : 0, + + /** + * Physical bodies are unaffected. Usecase is to fire events on contact. + * @attribute SILENT + */ + SILENT : 1 + }; + + Wall.DEFAULT_OPTIONS = { + restitution : 0.5, + drift : 0.5, + slop : 0, + normal : [1, 0, 0], + distance : 0, + onContact : Wall.ON_CONTACT.REFLECT + }; + + /* + * Setter for options. + * + * @method setOptions + * @param options {Objects} + */ + Wall.prototype.setOptions = function setOptions(options) { + if (options.normal !== undefined) { + if (options.normal instanceof Vector) this.options.normal = options.normal.clone(); + if (options.normal instanceof Array) this.options.normal = new Vector(options.normal); + } + if (options.restitution !== undefined) this.options.restitution = options.restitution; + if (options.drift !== undefined) this.options.drift = options.drift; + if (options.slop !== undefined) this.options.slop = options.slop; + if (options.distance !== undefined) this.options.distance = options.distance; + if (options.onContact !== undefined) this.options.onContact = options.onContact; + }; + + function _getNormalVelocity(n, v) { + return v.dot(n); + } + + function _getDistanceFromOrigin(p) { + var n = this.options.normal; + var d = this.options.distance; + return p.dot(n) + d; + } + + function _onEnter(particle, overlap, dt) { + var p = particle.position; + var v = particle.velocity; + var m = particle.mass; + var n = this.options.normal; + var action = this.options.onContact; + var restitution = this.options.restitution; + var impulse = this.impulse; + + var drift = this.options.drift; + var slop = -this.options.slop; + var gamma = 0; + + if (this._eventOutput) { + var data = {particle : particle, wall : this, overlap : overlap, normal : n}; + this._eventOutput.emit('preCollision', data); + this._eventOutput.emit('collision', data); + } + + switch (action) { + case Wall.ON_CONTACT.REFLECT: + var lambda = (overlap < slop) + ? -((1 + restitution) * n.dot(v) + drift / dt * (overlap - slop)) / (m * dt + gamma) + : -((1 + restitution) * n.dot(v)) / (m * dt + gamma); + + impulse.set(n.mult(dt * lambda)); + particle.applyImpulse(impulse); + particle.setPosition(p.add(n.mult(-overlap))); + break; + } + + if (this._eventOutput) this._eventOutput.emit('postCollision', data); + } + + function _onExit(particle, overlap, dt) { + var action = this.options.onContact; + var p = particle.position; + var n = this.options.normal; + + if (action === Wall.ON_CONTACT.REFLECT) { + particle.setPosition(p.add(n.mult(-overlap))); + } + } + + /** + * Adds an impulse to a physics body's velocity due to the wall constraint + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply the constraint to + * @param source {Body} The source of the constraint + * @param dt {Number} Delta time + */ + Wall.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + var n = this.options.normal; + + for (var i = 0; i < targets.length; i++) { + var particle = targets[i]; + var p = particle.position; + var v = particle.velocity; + var r = particle.radius || 0; + + var overlap = _getDistanceFromOrigin.call(this, p.add(n.mult(-r))); + var nv = _getNormalVelocity.call(this, n, v); + + if (overlap <= 0) { + if (nv < 0) _onEnter.call(this, particle, overlap, dt); + else _onExit.call(this, particle, overlap, dt); + } + } + }; + + module.exports = Wall; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/constraints/Walls',['require','exports','module','./Constraint','./Wall','../../math/Vector'],function(require, exports, module) { + var Constraint = require('./Constraint'); + var Wall = require('./Wall'); + var Vector = require('../../math/Vector'); + + /** + * Walls combines one or more Wall primitives and exposes a simple API to + * interact with several walls at once. A common use case would be to set up + * a bounding box for a physics body, that would collide with each side. + * + * @class Walls + * @constructor + * @extends Constraint + * @uses Wall + * @param {Options} [options] An object of configurable options. + * @param {Array} [options.sides] An array of sides e.g., [Walls.LEFT, Walls.TOP] + * @param {Array} [options.size] The size of the bounding box of the walls. + * @param {Array} [options.origin] The center of the wall relative to the size. + * @param {Array} [options.drift] Baumgarte stabilization parameter. Makes constraints "loosely" (0) or "tightly" (1) enforced. Range : [0, 1] + * @param {Array} [options.slop] Amount of penetration in pixels to ignore before collision event triggers. + * @param {Array} [options.restitution] The energy ratio lost in a collision (0 = stick, 1 = elastic) The energy ratio lost in a collision (0 = stick, 1 = elastic) + * @param {Array} [options.onContact] How to handle collision against the wall. + */ + function Walls(options) { + this.options = Object.create(Walls.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + _createComponents.call(this, options.sides || this.options.sides); + + Constraint.call(this); + } + + Walls.prototype = Object.create(Constraint.prototype); + Walls.prototype.constructor = Walls; + /** + * @property Walls.ON_CONTACT + * @type Object + * @extends Wall.ON_CONTACT + * @static + */ + Walls.ON_CONTACT = Wall.ON_CONTACT; + + /** + * An enumeration of common types of walls + * LEFT, RIGHT, TOP, BOTTOM, FRONT, BACK + * TWO_DIMENSIONAL, THREE_DIMENSIONAL + * + * @property Walls.SIDES + * @type Object + * @final + * @static + */ + Walls.SIDES = { + LEFT : 0, + RIGHT : 1, + TOP : 2, + BOTTOM : 3, + FRONT : 4, + BACK : 5, + TWO_DIMENSIONAL : [0, 1, 2, 3], + THREE_DIMENSIONAL : [0, 1, 2, 3, 4, 5] + }; + + Walls.DEFAULT_OPTIONS = { + sides : Walls.SIDES.TWO_DIMENSIONAL, + size : [window.innerWidth, window.innerHeight, 0], + origin : [.5, .5, .5], + drift : 0.5, + slop : 0, + restitution : 0.5, + onContact : Walls.ON_CONTACT.REFLECT + }; + + var _SIDE_NORMALS = { + 0 : new Vector(1, 0, 0), + 1 : new Vector(-1, 0, 0), + 2 : new Vector(0, 1, 0), + 3 : new Vector(0,-1, 0), + 4 : new Vector(0, 0, 1), + 5 : new Vector(0, 0,-1) + }; + + function _getDistance(side, size, origin) { + var distance; + var SIDES = Walls.SIDES; + switch (parseInt(side)) { + case SIDES.LEFT: + distance = size[0] * origin[0]; + break; + case SIDES.TOP: + distance = size[1] * origin[1]; + break; + case SIDES.FRONT: + distance = size[2] * origin[2]; + break; + case SIDES.RIGHT: + distance = size[0] * (1 - origin[0]); + break; + case SIDES.BOTTOM: + distance = size[1] * (1 - origin[1]); + break; + case SIDES.BACK: + distance = size[2] * (1 - origin[2]); + break; + } + return distance; + } + + /* + * Setter for options. + * + * @method setOptions + * @param options {Objects} + */ + Walls.prototype.setOptions = function setOptions(options) { + var resizeFlag = false; + if (options.restitution !== undefined) _setOptionsForEach.call(this, {restitution : options.restitution}); + if (options.drift !== undefined) _setOptionsForEach.call(this, {drift : options.drift}); + if (options.slop !== undefined) _setOptionsForEach.call(this, {slop : options.slop}); + if (options.onContact !== undefined) _setOptionsForEach.call(this, {onContact : options.onContact}); + if (options.size !== undefined) resizeFlag = true; + if (options.sides !== undefined) this.options.sides = options.sides; + if (options.origin !== undefined) resizeFlag = true; + if (resizeFlag) this.setSize(options.size, options.origin); + }; + + function _createComponents(sides) { + this.components = {}; + var components = this.components; + + for (var i = 0; i < sides.length; i++) { + var side = sides[i]; + components[i] = new Wall({ + normal : _SIDE_NORMALS[side].clone(), + distance : _getDistance(side, this.options.size, this.options.origin) + }); + } + } + + /* + * Setter for size. + * + * @method setOptions + * @param options {Objects} + */ + Walls.prototype.setSize = function setSize(size, origin) { + origin = origin || this.options.origin; + if (origin.length < 3) origin[2] = 0.5; + + this.forEach(function(wall, side) { + var d = _getDistance(side, size, origin); + wall.setOptions({distance : d}); + }); + + this.options.size = size; + this.options.origin = origin; + }; + + function _setOptionsForEach(options) { + this.forEach(function(wall) { + wall.setOptions(options); + }); + for (var key in options) this.options[key] = options[key]; + } + + /** + * Adds an impulse to a physics body's velocity due to the walls constraint + * + * @method applyConstraint + * @param targets {Array.Body} Array of bodies to apply the constraint to + * @param source {Body} The source of the constraint + * @param dt {Number} Delta time + */ + Walls.prototype.applyConstraint = function applyConstraint(targets, source, dt) { + this.forEach(function(wall) { + wall.applyConstraint(targets, source, dt); + }); + }; + + /** + * Apply a method to each wall making up the walls + * + * @method applyConstraint + * @param fn {Function} Function that takes in a wall as its first parameter + */ + Walls.prototype.forEach = function forEach(fn) { + var sides = this.options.sides; + for (var key in this.sides) fn(sides[key], key); + }; + + /** + * Rotates the walls by an angle in the XY-plane + * + * @method applyConstraint + * @param angle {Function} + */ + Walls.prototype.rotateZ = function rotateZ(angle) { + this.forEach(function(wall) { + var n = wall.options.normal; + n.rotateZ(angle).put(n); + }); + }; + + /** + * Rotates the walls by an angle in the YZ-plane + * + * @method applyConstraint + * @param angle {Function} + */ + Walls.prototype.rotateX = function rotateX(angle) { + this.forEach(function(wall) { + var n = wall.options.normal; + n.rotateX(angle).put(n); + }); + }; + + /** + * Rotates the walls by an angle in the XZ-plane + * + * @method applyConstraint + * @param angle {Function} + */ + Walls.prototype.rotateY = function rotateY(angle) { + this.forEach(function(wall) { + var n = wall.options.normal; + n.rotateY(angle).put(n); + }); + }; + + /** + * Resets the walls to their starting oritentation + */ + Walls.prototype.reset = function reset() { + var sides = this.options.sides; + for (var i in sides) { + var component = this.components[i]; + component.options.normal.set(_SIDE_NORMALS[i]); + } + }; + + module.exports = Walls; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/forces/Force',['require','exports','module','../../math/Vector','../../core/EventHandler'],function(require, exports, module) { + var Vector = require('../../math/Vector'); + var EventHandler = require('../../core/EventHandler'); + + /** + * Force base class. + * + * @class Force + * @uses EventHandler + * @constructor + */ + function Force(force) { + this.force = new Vector(force); + this._eventOutput = new EventHandler(); + EventHandler.setOutputHandler(this, this._eventOutput); + } + + /** + * Basic setter for options + * + * @method setOptions + * @param options {Objects} + */ + Force.prototype.setOptions = function setOptions(options) { + this._eventOutput.emit('change', options); + }; + + /** + * Adds a force to a physics body's force accumulator. + * + * @method applyForce + * @param targets {Array.Body} Array of bodies to apply a force to. + */ + Force.prototype.applyForce = function applyForce(targets) { + var length = targets.length; + while (length--) { + targets[length].applyForce(this.force); + } + }; + + /** + * Getter for a force's potential energy. + * + * @method getEnergy + * @return energy {Number} + */ + Force.prototype.getEnergy = function getEnergy() { + return 0.0; + }; + + module.exports = Force; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/forces/Drag',['require','exports','module','./Force'],function(require, exports, module) { + var Force = require('./Force'); + + /** + * Drag is a force that opposes velocity. Attach it to the physics engine + * to slow down a physics body in motion. + * + * @class Drag + * @constructor + * @extends Force + * @param {Object} options options to set on drag + */ + function Drag(options) { + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + Force.call(this); + } + + Drag.prototype = Object.create(Force.prototype); + Drag.prototype.constructor = Drag; + + /** + * @property Drag.FORCE_FUNCTIONS + * @type Object + * @protected + * @static + */ + Drag.FORCE_FUNCTIONS = { + + /** + * A drag force proportional to the velocity + * @attribute LINEAR + * @type Function + * @param {Vector} velocity + * @return {Vector} drag force + */ + LINEAR : function(velocity) { + return velocity; + }, + + /** + * A drag force proportional to the square of the velocity + * @attribute QUADRATIC + * @type Function + * @param {Vector} velocity + * @return {Vector} drag force + */ + QUADRATIC : function(velocity) { + return velocity.mult(velocity.norm()); + } + }; + + /** + * @property Drag.DEFAULT_OPTIONS + * @type Object + * @protected + * @static + */ + Drag.DEFAULT_OPTIONS = { + + /** + * The strength of the force + * Range : [0, 0.1] + * @attribute strength + * @type Number + * @default 0.01 + */ + strength : 0.01, + + /** + * The type of opposing force + * @attribute forceFunction + * @type Function + */ + forceFunction : Drag.FORCE_FUNCTIONS.LINEAR + }; + + /** + * Adds a drag force to a physics body's force accumulator. + * + * @method applyForce + * @param targets {Array.Body} Array of bodies to apply drag force to. + */ + Drag.prototype.applyForce = function applyForce(targets) { + var strength = this.options.strength; + var forceFunction = this.options.forceFunction; + var force = this.force; + var index; + var particle; + + for (index = 0; index < targets.length; index++) { + particle = targets[index]; + forceFunction(particle.velocity).mult(-strength).put(force); + particle.applyForce(force); + } + }; + + /** + * Basic options setter + * + * @method setOptions + * @param {Objects} options + */ + Drag.prototype.setOptions = function setOptions(options) { + for (var key in options) this.options[key] = options[key]; + }; + + module.exports = Drag; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/forces/Repulsion',['require','exports','module','./Force','../../math/Vector'],function(require, exports, module) { + var Force = require('./Force'); + var Vector = require('../../math/Vector'); + + /** + * Repulsion is a force that repels (attracts) bodies away (towards) + * each other. A repulsion of negative strength is attractive. + * + * @class Repulsion + * @constructor + * @extends Force + * @param {Object} options overwrites default options + */ + function Repulsion(options) { + this.options = Object.create(Repulsion.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.disp = new Vector(); + + Force.call(this); + } + + Repulsion.prototype = Object.create(Force.prototype); + Repulsion.prototype.constructor = Repulsion; + /** + * @property Repulsion.DECAY_FUNCTIONS + * @type Object + * @protected + * @static + */ + Repulsion.DECAY_FUNCTIONS = { + + /** + * A linear decay function + * @attribute LINEAR + * @type Function + * @param {Number} r distance from the source body + * @param {Number} cutoff the effective radius of influence + */ + LINEAR : function(r, cutoff) { + return Math.max(1 - (1 / cutoff) * r, 0); + }, + + /** + * A Morse potential decay function (http://en.wikipedia.org/wiki/Morse_potential) + * @attribute MORSE + * @type Function + * @param {Number} r distance from the source body + * @param {Number} cutoff the minimum radius of influence + */ + MORSE : function(r, cutoff) { + var r0 = (cutoff === 0) ? 100 : cutoff; + var rShifted = r + r0 * (1 - Math.log(2)); //shift by x-intercept + return Math.max(1 - Math.pow(1 - Math.exp(rShifted/r0 - 1), 2), 0); + }, + + /** + * An inverse distance decay function + * @attribute INVERSE + * @type Function + * @param {Number} r distance from the source body + * @param {Number} cutoff a distance shift to avoid singularities + */ + INVERSE : function(r, cutoff) { + return 1 / (1 - cutoff + r); + }, + + /** + * An inverse squared distance decay function + * @attribute GRAVITY + * @type Function + * @param {Number} r distance from the source body + * @param {Number} cutoff a distance shift to avoid singularities + */ + GRAVITY : function(r, cutoff) { + return 1 / (1 - cutoff + r*r); + } + }; + + /** + * @property Repulsion.DEFAULT_OPTIONS + * @type Object + * @protected + * @static + */ + Repulsion.DEFAULT_OPTIONS = { + + /** + * The strength of the force + * Range : [0, 100] + * @attribute strength + * @type Number + * @default 1 + */ + strength : 1, + + /** + * The location of the force, if not another physics body + * + * @attribute anchor + * @type Number + * @default 0.01 + * @optional + */ + anchor : undefined, + + /** + * The range of the repulsive force + * @attribute radii + * @type Array + * @default [0, Infinity] + */ + range : [0, Infinity], + + /** + * A normalization for the force to avoid singularities at the origin + * @attribute cutoff + * @type Number + * @default 0 + */ + cutoff : 0, + + /** + * The maximum magnitude of the force + * Range : [0, Infinity] + * @attribute cap + * @type Number + * @default Infinity + */ + cap : Infinity, + + /** + * The type of decay the repulsive force should have + * @attribute decayFunction + * @type Function + */ + decayFunction : Repulsion.DECAY_FUNCTIONS.GRAVITY + }; + + /* + * Setter for options. + * + * @method setOptions + * @param {Objects} options + */ + Repulsion.prototype.setOptions = function setOptions(options) { + if (options.anchor !== undefined) { + if (options.anchor.position instanceof Vector) this.options.anchor = options.anchor.position; + if (options.anchor instanceof Array) this.options.anchor = new Vector(options.anchor); + delete options.anchor; + } + for (var key in options) this.options[key] = options[key]; + }; + + /** + * Adds a drag force to a physics body's force accumulator. + * + * @method applyForce + * @param targets {Array.Body} Array of bodies to apply force to. + * @param source {Body} The source of the force + */ + Repulsion.prototype.applyForce = function applyForce(targets, source) { + var options = this.options; + var force = this.force; + var disp = this.disp; + + var strength = options.strength; + var anchor = options.anchor || source.position; + var cap = options.cap; + var cutoff = options.cutoff; + var rMin = options.range[0]; + var rMax = options.range[1]; + var decayFn = options.decayFunction; + + if (strength === 0) return; + + var length = targets.length; + var particle; + var m1; + var p1; + var r; + + while (length--) { + particle = targets[length]; + + if (particle === source) continue; + + m1 = particle.mass; + p1 = particle.position; + + disp.set(p1.sub(anchor)); + r = disp.norm(); + + if (r < rMax && r > rMin) { + force.set(disp.normalize(strength * m1 * decayFn(r, cutoff)).cap(cap)); + particle.applyForce(force); + } + } + + }; + + module.exports = Repulsion; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/forces/RotationalDrag',['require','exports','module','./Drag'],function(require, exports, module) { + var Drag = require('./Drag'); + + /** + * Rotational drag is a force that opposes angular velocity. + * Attach it to a physics body to slow down its rotation. + * + * @class RotationalDrag + * @constructor + * @extends Force + * @param {Object} options options to set on drag + */ + function RotationalDrag(options) { + Drag.call(this, options); + } + + RotationalDrag.prototype = Object.create(Drag.prototype); + RotationalDrag.prototype.constructor = RotationalDrag; + + RotationalDrag.DEFAULT_OPTIONS = Drag.DEFAULT_OPTIONS; + RotationalDrag.FORCE_FUNCTIONS = Drag.FORCE_FUNCTIONS; + + /** + * @property Repulsion.FORCE_FUNCTIONS + * @type Object + * @protected + * @static + */ + RotationalDrag.FORCE_FUNCTIONS = { + + /** + * A drag force proprtional to the angular velocity + * @attribute LINEAR + * @type Function + * @param {Vector} angularVelocity + * @return {Vector} drag force + */ + LINEAR : function(angularVelocity) { + return angularVelocity; + }, + + /** + * A drag force proprtional to the square of the angular velocity + * @attribute QUADRATIC + * @type Function + * @param {Vector} angularVelocity + * @return {Vector} drag force + */ + QUADRATIC : function(angularVelocity) { + return angularVelocity.mult(angularVelocity.norm()); + } + }; + + /** + * Adds a rotational drag force to a physics body's torque accumulator. + * + * @method applyForce + * @param targets {Array.Body} Array of bodies to apply drag force to. + */ + RotationalDrag.prototype.applyForce = function applyForce(targets) { + var strength = this.options.strength; + var forceFunction = this.options.forceFunction; + var force = this.force; + + //TODO: rotational drag as function of inertia + + var index; + var particle; + + for (index = 0; index < targets.length; index++) { + particle = targets[index]; + forceFunction(particle.angularVelocity).mult(-100*strength).put(force); + particle.applyTorque(force); + } + }; + + /* + * Setter for options. + * + * @method setOptions + * @param {Objects} options + */ + RotationalDrag.prototype.setOptions = function setOptions(options) { + for (var key in options) this.options[key] = options[key]; + }; + + module.exports = RotationalDrag; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +/*global console */ + +define('famous/physics/forces/Spring',['require','exports','module','./Force','../../math/Vector'],function(require, exports, module) { + var Force = require('./Force'); + var Vector = require('../../math/Vector'); + + /** + * A force that moves a physics body to a location with a spring motion. + * The body can be moved to another physics body, or an anchor point. + * + * @class Spring + * @constructor + * @extends Force + * @param {Object} options options to set on drag + */ + function Spring(options) { + Force.call(this); + + this.options = Object.create(this.constructor.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.disp = new Vector(0,0,0); + + _init.call(this); + } + + Spring.prototype = Object.create(Force.prototype); + Spring.prototype.constructor = Spring; + + /** @const */ + var pi = Math.PI; + var MIN_PERIOD = 150; + + /** + * @property Spring.FORCE_FUNCTIONS + * @type Object + * @protected + * @static + */ + Spring.FORCE_FUNCTIONS = { + + /** + * A FENE (Finitely Extensible Nonlinear Elastic) spring force + * see: http://en.wikipedia.org/wiki/FENE + * @attribute FENE + * @type Function + * @param {Number} dist current distance target is from source body + * @param {Number} rMax maximum range of influence + * @return {Number} unscaled force + */ + FENE : function(dist, rMax) { + var rMaxSmall = rMax * .99; + var r = Math.max(Math.min(dist, rMaxSmall), -rMaxSmall); + return r / (1 - r * r/(rMax * rMax)); + }, + + /** + * A Hookean spring force, linear in the displacement + * see: http://en.wikipedia.org/wiki/Hooke's_law + * @attribute FENE + * @type Function + * @param {Number} dist current distance target is from source body + * @return {Number} unscaled force + */ + HOOK : function(dist) { + return dist; + } + }; + + /** + * @property Spring.DEFAULT_OPTIONS + * @type Object + * @protected + * @static + */ + Spring.DEFAULT_OPTIONS = { + + /** + * The amount of time in milliseconds taken for one complete oscillation + * when there is no damping + * Range : [150, Infinity] + * @attribute period + * @type Number + * @default 300 + */ + period : 300, + + /** + * The damping of the spring. + * Range : [0, 1] + * 0 = no damping, and the spring will oscillate forever + * 1 = critically damped (the spring will never oscillate) + * @attribute dampingRatio + * @type Number + * @default 0.1 + */ + dampingRatio : 0.1, + + /** + * The rest length of the spring + * Range : [0, Infinity] + * @attribute length + * @type Number + * @default 0 + */ + length : 0, + + /** + * The maximum length of the spring (for a FENE spring) + * Range : [0, Infinity] + * @attribute length + * @type Number + * @default Infinity + */ + maxLength : Infinity, + + /** + * The location of the spring's anchor, if not another physics body + * + * @attribute anchor + * @type Array + * @optional + */ + anchor : undefined, + + /** + * The type of spring force + * @attribute forceFunction + * @type Function + */ + forceFunction : Spring.FORCE_FUNCTIONS.HOOK + }; + + function _calcStiffness() { + var options = this.options; + options.stiffness = Math.pow(2 * pi / options.period, 2); + } + + function _calcDamping() { + var options = this.options; + options.damping = 4 * pi * options.dampingRatio / options.period; + } + + function _init() { + _calcStiffness.call(this); + _calcDamping.call(this); + } + + /** + * Basic options setter + * + * @method setOptions + * @param options {Object} + */ + Spring.prototype.setOptions = function setOptions(options) { + // TODO fix no-console error + /* eslint no-console: 0 */ + + if (options.anchor !== undefined) { + if (options.anchor.position instanceof Vector) this.options.anchor = options.anchor.position; + if (options.anchor instanceof Vector) this.options.anchor = options.anchor; + if (options.anchor instanceof Array) this.options.anchor = new Vector(options.anchor); + } + + if (options.period !== undefined){ + if (options.period < MIN_PERIOD) { + options.period = MIN_PERIOD; + console.warn('The period of a SpringTransition is capped at ' + MIN_PERIOD + ' ms. Use a SnapTransition for faster transitions'); + } + this.options.period = options.period; + } + + if (options.dampingRatio !== undefined) this.options.dampingRatio = options.dampingRatio; + if (options.length !== undefined) this.options.length = options.length; + if (options.forceFunction !== undefined) this.options.forceFunction = options.forceFunction; + if (options.maxLength !== undefined) this.options.maxLength = options.maxLength; + + _init.call(this); + Force.prototype.setOptions.call(this, options); + }; + + /** + * Adds a spring force to a physics body's force accumulator. + * + * @method applyForce + * @param targets {Array.Body} Array of bodies to apply force to. + */ + Spring.prototype.applyForce = function applyForce(targets, source) { + var force = this.force; + var disp = this.disp; + var options = this.options; + + var stiffness = options.stiffness; + var damping = options.damping; + var restLength = options.length; + var maxLength = options.maxLength; + var anchor = options.anchor || source.position; + var forceFunction = options.forceFunction; + + var i; + var target; + var p2; + var v2; + var dist; + var m; + + for (i = 0; i < targets.length; i++) { + target = targets[i]; + p2 = target.position; + v2 = target.velocity; + + anchor.sub(p2).put(disp); + dist = disp.norm() - restLength; + + if (dist === 0) return; + + //if dampingRatio specified, then override strength and damping + m = target.mass; + stiffness *= m; + damping *= m; + + disp.normalize(stiffness * forceFunction(dist, maxLength)) + .put(force); + + if (damping) + if (source) force.add(v2.sub(source.velocity).mult(-damping)).put(force); + else force.add(v2.mult(-damping)).put(force); + + target.applyForce(force); + if (source) source.applyForce(force.mult(-1)); + } + }; + + /** + * Calculates the potential energy of the spring. + * + * @method getEnergy + * @param [targets] target The physics body attached to the spring + * @return {source} The potential energy of the spring + */ + Spring.prototype.getEnergy = function getEnergy(targets, source) { + var options = this.options; + var restLength = options.length; + var anchor = (source) ? source.position : options.anchor; + var strength = options.stiffness; + + var energy = 0.0; + for (var i = 0; i < targets.length; i++){ + var target = targets[i]; + var dist = anchor.sub(target.position).norm() - restLength; + energy += 0.5 * strength * dist * dist; + } + return energy; + }; + + module.exports = Spring; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +//TODO: test inheritance +define('famous/physics/forces/RotationalSpring',['require','exports','module','./Force','./Spring','../../math/Quaternion'],function(require, exports, module) { + var Force = require('./Force'); + var Spring = require('./Spring'); + var Quaternion = require('../../math/Quaternion'); + + /** + * A force that rotates a physics body back to target Euler angles. + * Just as a spring translates a body to a particular X, Y, Z, location, + * a rotational spring rotates a body to a particular X, Y, Z Euler angle. + * Note: there is no physical agent that does this in the "real world" + * + * @class RotationalSpring + * @constructor + * @extends Spring + * @param {Object} options options to set on drag + */ + function RotationalSpring(options) { + Spring.call(this, options); + } + + RotationalSpring.prototype = Object.create(Spring.prototype); + RotationalSpring.prototype.constructor = RotationalSpring; + + RotationalSpring.DEFAULT_OPTIONS = Spring.DEFAULT_OPTIONS; + RotationalSpring.FORCE_FUNCTIONS = Spring.FORCE_FUNCTIONS; + + /** @const */ + var pi = Math.PI; + + function _calcStiffness() { + var options = this.options; + options.stiffness = Math.pow(2 * pi / options.period, 2); + } + + function _calcDamping() { + var options = this.options; + options.damping = 4 * pi * options.dampingRatio / options.period; + } + + function _init() { + _calcStiffness.call(this); + _calcDamping.call(this); + } + + RotationalSpring.prototype.setOptions = function setOptions(options) { + // TODO fix no-console error + /* eslint no-console: 0 */ + + if (options.anchor !== undefined) { + if (options.anchor instanceof Quaternion) this.options.anchor = options.anchor; + if (options.anchor instanceof Array) this.options.anchor = new Quaternion(options.anchor); + } + + if (options.period !== undefined){ + this.options.period = options.period; + } + + if (options.dampingRatio !== undefined) this.options.dampingRatio = options.dampingRatio; + if (options.length !== undefined) this.options.length = options.length; + if (options.forceFunction !== undefined) this.options.forceFunction = options.forceFunction; + if (options.maxLength !== undefined) this.options.maxLength = options.maxLength; + + _init.call(this); + Force.prototype.setOptions.call(this, options); + }; + + /** + * Adds a torque force to a physics body's torque accumulator. + * + * @method applyForce + * @param targets {Array.Body} Array of bodies to apply torque to. + */ + RotationalSpring.prototype.applyForce = function applyForce(targets) { + var force = this.force; + var options = this.options; + var disp = this.disp; + + var stiffness = options.stiffness; + var damping = options.damping; + var restLength = options.length; + var anchor = options.anchor; + var forceFunction = options.forceFunction; + var maxLength = options.maxLength; + + var i; + var target; + var dist; + var m; + + for (i = 0; i < targets.length; i++) { + target = targets[i]; + + disp.set(anchor.sub(target.orientation)); + dist = disp.norm() - restLength; + + if (dist === 0) return; + + //if dampingRatio specified, then override strength and damping + m = target.mass; + stiffness *= m; + damping *= m; + + force.set(disp.normalize(stiffness * forceFunction(dist, maxLength))); + + if (damping) force.add(target.angularVelocity.mult(-damping)).put(force); + + target.applyTorque(force); + } + }; + + /** + * Calculates the potential energy of the rotational spring. + * + * @method getEnergy + * @param [targets] target The physics body attached to the spring + */ + RotationalSpring.prototype.getEnergy = function getEnergy(targets) { + var options = this.options; + var restLength = options.length; + var anchor = options.anchor; + var strength = options.stiffness; + + var energy = 0.0; + for (var i = 0; i < targets.length; i++) { + var target = targets[i]; + var dist = anchor.sub(target.orientation).norm() - restLength; + energy += 0.5 * strength * dist * dist; + } + return energy; + }; + + module.exports = RotationalSpring; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: david@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/physics/forces/VectorField',['require','exports','module','./Force','../../math/Vector'],function(require, exports, module) { + var Force = require('./Force'); + var Vector = require('../../math/Vector'); + + /** + * A force that moves a physics body to a location with a spring motion. + * The body can be moved to another physics body, or an anchor point. + * + * @class VectorField + * @constructor + * @extends Force + * @param {Object} options options to set on drag + */ + function VectorField(options) { + Force.call(this); + + this.options = Object.create(VectorField.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + + //registers + this.evaluation = new Vector(); + } + + VectorField.prototype = Object.create(Force.prototype); + VectorField.prototype.constructor = VectorField; + + /** + * @property Spring.FORCE_FUNCTIONS + * @type Object + * @protected + * @static + */ + VectorField.FIELDS = { + /** + * Constant force, e.g., gravity + * @attribute CONSTANT + * @type Function + * @param v {Vector} Current position of physics body + * @param options {Object} The direction of the force + * Pass a {direction : Vector} into the VectorField options + * @return {Number} unscaled force + */ + CONSTANT : function(v, options) { + options.direction.put(this.evaluation); + }, + + /** + * Linear force + * @attribute LINEAR + * @type Function + * @param v {Vector} Current position of physics body + * @return {Vector} unscaled force + */ + LINEAR : function(v) { + v.put(this.evaluation); + }, + + /** + * Radial force, e.g., Hookean spring + * @attribute RADIAL + * @type Function + * @param v {Vector} Current position of physics body + * @return {Vector} unscaled force + */ + RADIAL : function(v) { + v.mult(-1).put(this.evaluation); + }, + + /** + * Point attractor force, e.g., Hookean spring with an anchor + * @attribute POINT_ATTRACTOR + * @type Function + * @param v {Vector} Current position of physics body + * @param options {Object} And object with the position of the attractor + * Pass a {position : Vector} into the VectorField options + * @return {Vector} unscaled force + */ + POINT_ATTRACTOR : function(v, options) { + options.position.sub(v).put(this.evaluation); + } + }; + + /** + * @property VectorField.DEFAULT_OPTIONS + * @type Object + * @protected + * @static + */ + VectorField.DEFAULT_OPTIONS = { + + /** + * The strength of the force + * Range : [0, 10] + * @attribute strength + * @type Number + * @default .01 + */ + strength : .01, + + /** + * Type of vectorfield + * Range : [0, 100] + * @attribute field + * @type Function + */ + field : VectorField.FIELDS.CONSTANT + }; + + /** + * Basic options setter + * + * @method setOptions + * @param {Objects} options + */ + VectorField.prototype.setOptions = function setOptions(options) { + if (options.strength !== undefined) this.options.strength = options.strength; + if (options.direction !== undefined) this.options.direction = options.direction; + if (options.field !== undefined) { + this.options.field = options.field; + _setFieldOptions.call(this, this.options.field); + } + }; + + function _setFieldOptions(field) { + var FIELDS = VectorField.FIELDS; + + switch (field) { + case FIELDS.CONSTANT: + if (!this.options.direction) this.options.direction = new Vector(0,1,0); + else if (this.options.direction instanceof Array) this.options.direction = new Vector(this.options.direction); + break; + case FIELDS.POINT_ATTRACTOR: + if (!this.options.position) this.options.position = new Vector(0,0,0); + else if (this.options.position instanceof Array) this.options.position = new Vector(this.options.position); + break; + } + } + + /** + * Adds the VectorField's force to a physics body's force accumulator. + * + * @method applyForce + * @param targets {Array.body} Array of bodies to apply force to. + */ + VectorField.prototype.applyForce = function applyForce(targets) { + var force = this.force; + var strength = this.options.strength; + var field = this.options.field; + + var i; + var target; + + for (i = 0; i < targets.length; i++) { + target = targets[i]; + field.call(this, target.position, this.options); + this.evaluation.mult(target.mass * strength).put(force); + target.applyForce(force); + } + }; + + VectorField.prototype.getEnergy = function getEnergy(targets) { + var field = this.options.field; + var FIELDS = VectorField.FIELDS; + + var energy = 0; + + var i; + var target; + switch (field) { + case FIELDS.CONSTANT: + energy = targets.length * this.options.direction.norm(); + break; + case FIELDS.RADIAL: + for (i = 0; i < targets.length; i++){ + target = targets[i]; + energy += target.position.norm(); + } + break; + case FIELDS.POINT_ATTRACTOR: + for (i = 0; i < targets.length; i++){ + target = targets[i]; + energy += target.position.sub(this.options.position).norm(); + } + break; + } + energy *= this.options.strength; + return energy; + }; + + module.exports = VectorField; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/surfaces/CanvasSurface',['require','exports','module','../core/Surface'],function(require, exports, module) { + var Surface = require('../core/Surface'); + + /** + * A surface containing an HTML5 Canvas element. + * This extends the Surface class. + * + * @class CanvasSurface + * @extends Surface + * @constructor + * @param {Object} [options] overrides of default options + * @param {Array.Number} [options.canvasSize] [width, height] for document element + */ + function CanvasSurface(options) { + if (options && options.canvasSize) this._canvasSize = options.canvasSize; + Surface.apply(this, arguments); + if (!this._canvasSize) this._canvasSize = this.getSize(); + this._backBuffer = document.createElement('canvas'); + if (this._canvasSize) { + this._backBuffer.width = this._canvasSize[0]; + this._backBuffer.height = this._canvasSize[1]; + } + this._contextId = undefined; + } + + CanvasSurface.prototype = Object.create(Surface.prototype); + CanvasSurface.prototype.constructor = CanvasSurface; + CanvasSurface.prototype.elementType = 'canvas'; + CanvasSurface.prototype.elementClass = 'famous-surface'; + + /** + * Set inner document content. Note that this is a noop for CanvasSurface. + * + * @method setContent + * + */ + CanvasSurface.prototype.setContent = function setContent() {}; + + /** + * Place the document element this component manages into the document. + * This will draw the content to the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + CanvasSurface.prototype.deploy = function deploy(target) { + if (this._canvasSize) { + target.width = this._canvasSize[0]; + target.height = this._canvasSize[1]; + } + if (this._contextId === '2d') { + target.getContext(this._contextId).drawImage(this._backBuffer, 0, 0); + this._backBuffer.width = 0; + this._backBuffer.height = 0; + } + }; + + /** + * Remove this component and contained content from the document + * + * @private + * @method recall + * + * @param {Node} target node to which the component was deployed + */ + CanvasSurface.prototype.recall = function recall(target) { + var size = this.getSize(); + + this._backBuffer.width = target.width; + this._backBuffer.height = target.height; + + if (this._contextId === '2d') { + this._backBuffer.getContext(this._contextId).drawImage(target, 0, 0); + target.width = 0; + target.height = 0; + } + }; + + /** + * Returns the canvas element's context + * + * @method getContext + * @param {string} contextId context identifier + */ + CanvasSurface.prototype.getContext = function getContext(contextId) { + this._contextId = contextId; + return this._currentTarget ? this._currentTarget.getContext(contextId) : this._backBuffer.getContext(contextId); + }; + + /** + * Set the size of the surface and canvas element. + * + * @method setSize + * @param {Array.number} size [width, height] of surface + * @param {Array.number} canvasSize [width, height] of canvas surface + */ + CanvasSurface.prototype.setSize = function setSize(size, canvasSize) { + Surface.prototype.setSize.apply(this, arguments); + if (canvasSize) this._canvasSize = [canvasSize[0], canvasSize[1]]; + if (this._currentTarget) { + this._currentTarget.width = this._canvasSize[0]; + this._currentTarget.height = this._canvasSize[1]; + } + }; + + module.exports = CanvasSurface; +}); + + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/surfaces/ContainerSurface',['require','exports','module','../core/Surface','../core/Context'],function(require, exports, module) { + var Surface = require('../core/Surface'); + var Context = require('../core/Context'); + + /** + * ContainerSurface is an object designed to contain surfaces and + * set properties to be applied to all of them at once. + * This extends the Surface class. + * A container surface will enforce these properties on the + * surfaces it contains: + * + * size (clips contained surfaces to its own width and height); + * + * origin; + * + * its own opacity and transform, which will be automatically + * applied to all Surfaces contained directly and indirectly. + * + * @class ContainerSurface + * @extends Surface + * @constructor + * @param {Array.Number} [options.size] [width, height] in pixels + * @param {Array.string} [options.classes] CSS classes to set on all inner content + * @param {Array} [options.properties] string dictionary of HTML attributes to set on target div + * @param {string} [options.content] inner (HTML) content of surface (should not be used) + */ + function ContainerSurface(options) { + Surface.call(this, options); + this._container = document.createElement('div'); + this._container.classList.add('famous-group'); + this._container.classList.add('famous-container-group'); + this._shouldRecalculateSize = false; + this.context = new Context(this._container); + this.setContent(this._container); + } + + ContainerSurface.prototype = Object.create(Surface.prototype); + ContainerSurface.prototype.constructor = ContainerSurface; + ContainerSurface.prototype.elementType = 'div'; + ContainerSurface.prototype.elementClass = 'famous-surface'; + + /** + * Add renderables to this object's render tree + * + * @method add + * + * @param {Object} obj renderable object + * @return {RenderNode} RenderNode wrapping this object, if not already a RenderNode + */ + ContainerSurface.prototype.add = function add() { + return this.context.add.apply(this.context, arguments); + }; + + /** + * Return spec for this surface. Note: Can result in a size recalculation. + * + * @private + * @method render + * + * @return {Object} render spec for this surface (spec id) + */ + ContainerSurface.prototype.render = function render() { + if (this._sizeDirty) this._shouldRecalculateSize = true; + return Surface.prototype.render.apply(this, arguments); + }; + + /** + * Place the document element this component manages into the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + ContainerSurface.prototype.deploy = function deploy() { + this._shouldRecalculateSize = true; + return Surface.prototype.deploy.apply(this, arguments); + }; + + /** + * Apply changes from this component to the corresponding document element. + * This includes changes to classes, styles, size, content, opacity, origin, + * and matrix transforms. + * + * @private + * @method commit + * @param {Context} context commit context + * @param {Transform} transform unused TODO + * @param {Number} opacity unused TODO + * @param {Array.Number} origin unused TODO + * @param {Array.Number} size unused TODO + * @return {undefined} TODO returns an undefined value + */ + ContainerSurface.prototype.commit = function commit(context, transform, opacity, origin, size) { + var previousSize = this._size ? [this._size[0], this._size[1]] : null; + var result = Surface.prototype.commit.apply(this, arguments); + if (this._shouldRecalculateSize || (previousSize && (this._size[0] !== previousSize[0] || this._size[1] !== previousSize[1]))) { + this.context.setSize(); + this._shouldRecalculateSize = false; + } + this.context.update(); + return result; + }; + + module.exports = ContainerSurface; +}); + +define('famous/surfaces/FormContainerSurface',['require','exports','module','./ContainerSurface'],function(require, exports, module) { + var ContainerSurface = require('./ContainerSurface'); + + function FormContainerSurface(options) { + if (options) this._method = options.method || ''; + ContainerSurface.apply(this, arguments); + } + + FormContainerSurface.prototype = Object.create(ContainerSurface.prototype); + FormContainerSurface.prototype.constructor = FormContainerSurface; + + FormContainerSurface.prototype.elementType = 'form'; + + FormContainerSurface.prototype.deploy = function deploy(target) { + if (this._method) target.method = this._method; + return ContainerSurface.prototype.deploy.apply(this, arguments); + }; + + module.exports = FormContainerSurface; +}); + + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/surfaces/ImageSurface',['require','exports','module','../core/Surface'],function(require, exports, module) { + var Surface = require('../core/Surface'); + + /** + * A surface containing image content. + * This extends the Surface class. + * + * @class ImageSurface + * + * @extends Surface + * @constructor + * @param {Object} [options] overrides of default options + */ + function ImageSurface(options) { + this._imageUrl = undefined; + Surface.apply(this, arguments); + } + + var urlCache = []; + var countCache = []; + var nodeCache = []; + var cacheEnabled = true; + + ImageSurface.enableCache = function enableCache() { + cacheEnabled = true; + }; + + ImageSurface.disableCache = function disableCache() { + cacheEnabled = false; + }; + + ImageSurface.clearCache = function clearCache() { + urlCache = []; + countCache = []; + nodeCache = []; + }; + + ImageSurface.getCache = function getCache() { + return { + urlCache: urlCache, + countCache: countCache, + nodeCache: nodeCache + }; + }; + + ImageSurface.prototype = Object.create(Surface.prototype); + ImageSurface.prototype.constructor = ImageSurface; + ImageSurface.prototype.elementType = 'img'; + ImageSurface.prototype.elementClass = 'famous-surface'; + + /** + * Set content URL. This will cause a re-rendering. + * @method setContent + * @param {string} imageUrl + */ + ImageSurface.prototype.setContent = function setContent(imageUrl) { + var urlIndex = urlCache.indexOf(this._imageUrl); + if (urlIndex !== -1) { + if (countCache[urlIndex] === 1) { + urlCache.splice(urlIndex, 1); + countCache.splice(urlIndex, 1); + nodeCache.splice(urlIndex, 1); + } else { + countCache[urlIndex]--; + } + } + + urlIndex = urlCache.indexOf(imageUrl); + if (urlIndex === -1) { + urlCache.push(imageUrl); + countCache.push(1); + } + else { + countCache[urlIndex]++; + } + + this._imageUrl = imageUrl; + this._contentDirty = true; + }; + + /** + * Place the document element that this component manages into the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + ImageSurface.prototype.deploy = function deploy(target) { + var urlIndex = urlCache.indexOf(this._imageUrl); + if (nodeCache[urlIndex] === undefined && cacheEnabled) { + var img = new Image(); + img.src = this._imageUrl || ''; + nodeCache[urlIndex] = img; + } + + target.src = this._imageUrl || ''; + }; + + /** + * Remove this component and contained content from the document + * + * @private + * @method recall + * + * @param {Node} target node to which the component was deployed + */ + ImageSurface.prototype.recall = function recall(target) { + target.src = ''; + }; + + module.exports = ImageSurface; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/surfaces/InputSurface',['require','exports','module','../core/Surface'],function(require, exports, module) { + var Surface = require('../core/Surface'); + + /** + * A Famo.us surface in the form of an HTML input element. + * This extends the Surface class. + * + * @class InputSurface + * @extends Surface + * @constructor + * @param {Object} [options] overrides of default options + * @param {string} [options.placeholder] placeholder text hint that describes the expected value of an element + * @param {string} [options.type] specifies the type of element to display (e.g. 'datetime', 'text', 'button', etc.) + * @param {string} [options.value] value of text + */ + function InputSurface(options) { + this._placeholder = options.placeholder || ''; + this._value = options.value || ''; + this._type = options.type || 'text'; + this._name = options.name || ''; + + Surface.apply(this, arguments); + + this.on('click', this.focus.bind(this)); + window.addEventListener('click', function(event) { + if (event.target !== this._currentTarget) this.blur(); + }.bind(this)); + } + InputSurface.prototype = Object.create(Surface.prototype); + InputSurface.prototype.constructor = InputSurface; + + InputSurface.prototype.elementType = 'input'; + InputSurface.prototype.elementClass = 'famous-surface'; + + /** + * Set placeholder text. Note: Triggers a repaint. + * + * @method setPlaceholder + * @param {string} str Value to set the placeholder to. + * @return {InputSurface} this, allowing method chaining. + */ + InputSurface.prototype.setPlaceholder = function setPlaceholder(str) { + this._placeholder = str; + this._contentDirty = true; + return this; + }; + + /** + * Focus on the current input, pulling up the keyboard on mobile. + * + * @method focus + * @return {InputSurface} this, allowing method chaining. + */ + InputSurface.prototype.focus = function focus() { + if (this._currentTarget) this._currentTarget.focus(); + return this; + }; + + /** + * Blur the current input, hiding the keyboard on mobile. + * + * @method blur + * @return {InputSurface} this, allowing method chaining. + */ + InputSurface.prototype.blur = function blur() { + if (this._currentTarget) this._currentTarget.blur(); + return this; + }; + + /** + * Set the placeholder conent. + * Note: Triggers a repaint next tick. + * + * @method setValue + * @param {string} str Value to set the main input value to. + * @return {InputSurface} this, allowing method chaining. + */ + InputSurface.prototype.setValue = function setValue(str) { + this._value = str; + this._contentDirty = true; + return this; + }; + + /** + * Set the type of element to display conent. + * Note: Triggers a repaint next tick. + * + * @method setType + * @param {string} str type of the input surface (e.g. 'button', 'text') + * @return {InputSurface} this, allowing method chaining. + */ + InputSurface.prototype.setType = function setType(str) { + this._type = str; + this._contentDirty = true; + return this; + }; + + /** + * Get the value of the inner content of the element (e.g. the entered text) + * + * @method getValue + * @return {string} value of element + */ + InputSurface.prototype.getValue = function getValue() { + if (this._currentTarget) { + return this._currentTarget.value; + } + else { + return this._value; + } + }; + + /** + * Set the name attribute of the element. + * Note: Triggers a repaint next tick. + * + * @method setName + * @param {string} str element name + * @return {InputSurface} this, allowing method chaining. + */ + InputSurface.prototype.setName = function setName(str) { + this._name = str; + this._contentDirty = true; + return this; + }; + + /** + * Get the name attribute of the element. + * + * @method getName + * @return {string} name of element + */ + InputSurface.prototype.getName = function getName() { + return this._name; + }; + + /** + * Place the document element this component manages into the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + InputSurface.prototype.deploy = function deploy(target) { + if (this._placeholder !== '') target.placeholder = this._placeholder; + target.value = this._value; + target.type = this._type; + target.name = this._name; + }; + + module.exports = InputSurface; +}); + +define('famous/surfaces/SubmitInputSurface',['require','exports','module','./InputSurface'],function(require, exports, module) { + var InputSurface = require('./InputSurface'); + + function SubmitInputSurface(options) { + InputSurface.apply(this, arguments); + this._type = 'submit'; + if (options && options.onClick) this.setOnClick(options.onClick); + } + + SubmitInputSurface.prototype = Object.create(InputSurface.prototype); + SubmitInputSurface.prototype.constructor = SubmitInputSurface; + + SubmitInputSurface.prototype.setOnClick = function(onClick) { + this.onClick = onClick; + }; + + SubmitInputSurface.prototype.deploy = function deploy(target) { + if (this.onclick) target.onClick = this.onClick; + InputSurface.prototype.deploy.apply(this, arguments); + }; + + module.exports = SubmitInputSurface; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/surfaces/TextareaSurface',['require','exports','module','../core/Surface'],function(require, exports, module) { + var Surface = require('../core/Surface'); + + /** + * A Famo.us surface in the form of an HTML textarea element. + * This extends the Surface class. + * + * @class TextareaSurface + * @extends Surface + * @constructor + * @param {Object} [options] overrides of default options + * @param {string} [options.placeholder] placeholder text hint that describes the expected value of an textarea element + * @param {string} [options.value] value of text + * @param {string} [options.name] specifies the name of textarea + * @param {string} [options.wrap] specify 'hard' or 'soft' wrap for textarea + * @param {number} [options.cols] number of columns in textarea + * @param {number} [options.rows] number of rows in textarea + */ + function TextareaSurface(options) { + this._placeholder = options.placeholder || ''; + this._value = options.value || ''; + this._name = options.name || ''; + this._wrap = options.wrap || ''; + this._cols = options.cols || ''; + this._rows = options.rows || ''; + + Surface.apply(this, arguments); + this.on('click', this.focus.bind(this)); + } + TextareaSurface.prototype = Object.create(Surface.prototype); + TextareaSurface.prototype.constructor = TextareaSurface; + + TextareaSurface.prototype.elementType = 'textarea'; + TextareaSurface.prototype.elementClass = 'famous-surface'; + + /** + * Set placeholder text. Note: Triggers a repaint. + * + * @method setPlaceholder + * @param {string} str Value to set the placeholder to. + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.setPlaceholder = function setPlaceholder(str) { + this._placeholder = str; + this._contentDirty = true; + return this; + }; + + /** + * Focus on the current input, pulling up the keyboard on mobile. + * + * @method focus + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.focus = function focus() { + if (this._currentTarget) this._currentTarget.focus(); + return this; + }; + + /** + * Blur the current input, hiding the keyboard on mobile. + * + * @method focus + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.blur = function blur() { + if (this._currentTarget) this._currentTarget.blur(); + return this; + }; + + /** + * Set the value of textarea. + * Note: Triggers a repaint next tick. + * + * @method setValue + * @param {string} str Value to set the main textarea value to. + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.setValue = function setValue(str) { + this._value = str; + this._contentDirty = true; + return this; + }; + + /** + * Get the value of the inner content of the textarea (e.g. the entered text) + * + * @method getValue + * @return {string} value of element + */ + TextareaSurface.prototype.getValue = function getValue() { + if (this._currentTarget) { + return this._currentTarget.value; + } + else { + return this._value; + } + }; + + /** + * Set the name attribute of the element. + * Note: Triggers a repaint next tick. + * + * @method setName + * @param {string} str element name + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.setName = function setName(str) { + this._name = str; + this._contentDirty = true; + return this; + }; + + /** + * Get the name attribute of the element. + * + * @method getName + * @return {string} name of element + */ + TextareaSurface.prototype.getName = function getName() { + return this._name; + }; + + /** + * Set the wrap of textarea. + * Note: Triggers a repaint next tick. + * + * @method setWrap + * @param {string} str wrap of the textarea surface (e.g. 'soft', 'hard') + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.setWrap = function setWrap(str) { + this._wrap = str; + this._contentDirty = true; + return this; + }; + + /** + * Set the number of columns visible in the textarea. + * Note: Overridden by surface size; set width to true. (eg. size: [true, *]) + * Triggers a repaint next tick. + * + * @method setColumns + * @param {number} num columns in textarea surface + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.setColumns = function setColumns(num) { + this._cols = num; + this._contentDirty = true; + return this; + }; + + /** + * Set the number of rows visible in the textarea. + * Note: Overridden by surface size; set height to true. (eg. size: [*, true]) + * Triggers a repaint next tick. + * + * @method setRows + * @param {number} num rows in textarea surface + * @return {TextareaSurface} this, allowing method chaining. + */ + TextareaSurface.prototype.setRows = function setRows(num) { + this._rows = num; + this._contentDirty = true; + return this; + }; + + /** + * Place the document element this component manages into the document. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + TextareaSurface.prototype.deploy = function deploy(target) { + if (this._placeholder !== '') target.placeholder = this._placeholder; + if (this._value !== '') target.value = this._value; + if (this._name !== '') target.name = this._name; + if (this._wrap !== '') target.wrap = this._wrap; + if (this._cols !== '') target.cols = this._cols; + if (this._rows !== '') target.rows = this._rows; + }; + + module.exports = TextareaSurface; +}); + +/* This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. + * + * Owner: mark@famo.us + * @license MPL 2.0 + * @copyright Famous Industries, Inc. 2015 + */ + +define('famous/surfaces/VideoSurface',['require','exports','module','../core/Surface'],function(require, exports, module) { + var Surface = require('../core/Surface'); + + /** + * Creates a famous surface containing video content. Currently adding + * controls and manipulating the video are not supported through the + * surface interface, but can be accomplished via standard JavaScript + * manipulation of the video DOM element. + * This extends the Surface class. + * + * @class VideoSurface + * @extends Surface + * @constructor + * @param {Object} [options] default option overrides + * @param {Array.Number} [options.size] [width, height] in pixels + * @param {Array.string} [options.classes] CSS classes to set on inner content + * @param {Array} [options.properties] string dictionary of HTML attributes to set on target div + * @param {String} [options.src] videoUrl URL + * @param {boolean} [options.autoplay] autoplay + */ + function VideoSurface(options) { + Surface.apply(this, arguments); + this._videoUrl = undefined; + this.options = Object.create(VideoSurface.DEFAULT_OPTIONS); + if (options) this.setOptions(options); + } + + VideoSurface.prototype = Object.create(Surface.prototype); + VideoSurface.prototype.constructor = VideoSurface; + + VideoSurface.DEFAULT_OPTIONS = { + autoplay: false + }; + + VideoSurface.prototype.elementType = 'video'; + VideoSurface.prototype.elementClass = 'famous-surface'; + + /** + * Set internal options, overriding any default options + * + * @method setOptions + * + * @param {Object} [options] overrides of default options + * @param {Boolean} [options.autoplay] HTML autoplay + */ + VideoSurface.prototype.setOptions = function setOptions(options) { + if (options.size) this.setSize(options.size); + if (options.classes) this.setClasses(options.classes); + if (options.properties) this.setProperties(options.properties); + if (options.autoplay) this.options.autoplay = options.autoplay; + if (options.src) { + this._videoUrl = options.src; + this._contentDirty = true; + } + }; + + /** + * Set url of the video. + * + * @method setContent + * @param {string} videoUrl URL + */ + VideoSurface.prototype.setContent = function setContent(videoUrl) { + this._videoUrl = videoUrl; + this._contentDirty = true; + }; + + /** + * Place the document element this component manages into the document. + * Note: In the case of VideoSurface, simply changes the options on the target. + * + * @private + * @method deploy + * @param {Node} target document parent of this container + */ + VideoSurface.prototype.deploy = function deploy(target) { + target.src = this._videoUrl; + target.autoplay = this.options.autoplay; + }; + + /** + * Remove this component and contained content from the document. + * Note: This doesn't actually remove the