/** * Ink Core. * @module Ink_1 * This module provides the necessary methods to create and load the modules using Ink. */ ;(function(window, document) { 'use strict'; // skip redefinition of Ink core if ('Ink' in window) { return; } // internal data /* * NOTE: * invoke Ink.setPath('Ink', '/Ink/'); before requiring local modules */ var paths = {}; var modules = {}; var modulesLoadOrder = []; var modulesRequested = {}; var pendingRMs = []; var modulesWaitingForDeps = {}; var apply = Function.prototype.apply; // auxiliary fns var isEmptyObject = function(o) { /*jshint unused:false */ if (typeof o !== 'object') { return false; } for (var k in o) { if (o.hasOwnProperty(k)) { return false; } } return true; }; /** * @namespace Ink_1 */ window.Ink = { VERSION: '3.0.5', _checkPendingRequireModules: function() { var I, F, o, dep, mod, cb, pRMs = []; for (I = 0, F = pendingRMs.length; I < F; ++I) { o = pendingRMs[I]; if (!o) { continue; } for (dep in o.left) { if (o.left.hasOwnProperty(dep)) { mod = modules[dep]; if (mod) { o.args[o.left[dep] ] = mod; delete o.left[dep]; --o.remaining; } } } if (o.remaining > 0) { pRMs.push(o); } else { cb = o.cb; if (!cb) { continue; } delete o.cb; // to make sure I won't call this more than once! cb.apply(false, o.args); } } pendingRMs = pRMs; if (pendingRMs.length > 0) { setTimeout( function() { Ink._checkPendingRequireModules(); }, 0 ); } }, /** * Get the full path of a module. * This method looks up the paths given in setPath (and ultimately the default Ink's path). * * @method getPath * @param {String} key Name of the module you want to get the path * @param {Boolean} [noLib] Flag to skip appending 'lib.js' to the returned path. */ getPath: function(key, noLib) { var split = key.split(/[._]/g); var curKey; var i; var root; var path; // Look for Ink.Dom.Element.1, Ink.Dom.Element, Ink.Dom, Ink in this order. for (i = split.length; i >= 0; i -= 1) { curKey = split.slice(0, i + 1).join('.'); // See comment in setPath if (paths[curKey]) { root = curKey; break; } } if (root in paths) { path = paths[root]; } else { return null; } if (!/\/$/.test(path)) { path += '/'; } if (i < split.length) { // Add the rest of the path. For example, if we found // paths['Ink.Dom'] to be 'http://example.com/Ink/Dom/', // we now add '/Element/' to get the full path. path += split.slice(i + 1).join('/') + '/'; } if (!noLib) { path += 'lib.js'; } return path; }, /** * Sets the URL path for a namespace. * Use this to customize where requireModules and createModule will load dependencies from. * This can be useful to set your own CDN for dynamic module loading or simply to change your module folder structure * * @method setPath * * @param {String} key Module or namespace * @param {String} rootURI Base URL path and schema to be appended to the module or namespace * * @example * Ink.setPath('Ink', 'http://my-cdn/Ink/'); * Ink.setPath('Lol', 'http://my-cdn/Lol/'); * * // Loads from http://my-cdn/Ink/Dom/Whatever/lib.js * Ink.requireModules(['Ink.Dom.Whatever'], function () { ... }); * // Loads from http://my-cdn/Lol/Whatever/lib.js * Ink.requireModules(['Lol.Whatever'], function () { ... }); */ setPath: function(key, rootURI) { // Replacing version separator with dot because the difference // between a submodule and a version doesn't matter here. // It would also overcomplicate the implementation of getPath paths[key.replace(/_/, '.')] = rootURI; }, /** * Loads a script URL. * This creates a `script` tag in the `head` of the document. * Reports errors by listening to 'error' and 'readystatechange' events. * * @method loadScript * @param {String} uri Can be an external URL or a module name * @param {String} [contentType]='text/javascript' The `type` attribute of the new script tag. */ loadScript: function(uri, contentType) { /*jshint evil:true */ if (uri.indexOf('/') === -1) { var givenUri = uri; // For the error message uri = this.getPath(uri); if (uri === null) { throw new Error('Could not load script "' + givenUri + '". ' + 'Path not found in the registry. Did you misspell ' + 'the name, or forgot to call setPath()?'); } } var scriptEl = document.createElement('script'); scriptEl.setAttribute('type', contentType || 'text/javascript'); scriptEl.setAttribute('src', uri); scriptEl.onerror = scriptEl.onreadystatechange = function (ev) { ev = ev || window.event; if (ev.type === 'readystatechange' && scriptEl.readyState !== 'loaded') { // if not readyState == 'loaded' it's not an error. return; } Ink.error(['Failed to load script from ', uri, '.'].join('')); }; // CHECK ON ALL BROWSERS /*if (document.readyState !== 'complete' && !document.body) { document.write( scriptEl.outerHTML ); } else {*/ var aHead = document.getElementsByTagName('head'); if(aHead.length > 0) { aHead[0].appendChild(scriptEl); } //} }, _loadLater: function (dep) { setTimeout(function () { if (modules[dep] || modulesRequested[dep] || modulesWaitingForDeps[dep]) { return; } modulesRequested[dep] = true; Ink.loadScript(dep); }, 0); }, /** * Defines a module namespace. * * @method namespace * @param {String} ns Namespace to define. * @param {Boolean} [returnParentAndKey] Flag to change the return value to an array containing the namespace parent and the namespace key * @return {Object|Array} Returns the created namespace object */ namespace: function(ns, returnParentAndKey) { if (!ns || !ns.length) { return null; } var levels = ns.split('.'); var nsobj = window; var parent; for (var i = 0, f = levels.length; i < f; ++i) { nsobj[ levels[i] ] = nsobj[ levels[i] ] || {}; parent = nsobj; nsobj = nsobj[ levels[i] ]; } if (returnParentAndKey) { return [ parent, levels[i-1] ]; } return nsobj; }, /** * Loads a module. * A synchronous method to get the module from the internal registry. * It assumes the module is defined and loaded already! * * @method getModule * @param {String} mod Module name * @param {Number} [version] Version number of the module * @return {Object|Function} Module object or function, depending how the module is defined */ getModule: function(mod, version) { var key = version ? [mod, '_', version].join('') : mod; return modules[key]; }, /** * Creates a new module. * Use this to wrap your code and benefit from the module loading used throughout the Ink library * * @method createModule * @param {String} mod Module name, separated by dots. Like Ink.Dom.Selector, Ink.UI.Modal * @param {Number} version Version number * @param {Array} deps Array of module names which are dependencies of the module being created. The order in which they are passed here will define the order they will be passed to the callback function. * @param {Function} modFn The callback function to be executed when all the dependencies are resolved. The dependencies are passed as arguments, in the same order they were declared. The function itself should return the module. * @sample Ink_1_createModule.html * */ createModule: function(mod, ver, deps, modFn) { // define if (typeof mod !== 'string') { throw new Error('module name must be a string!'); } // validate version correctness if (!(typeof ver === 'number' || (typeof ver === 'string' && ver.length > 0))) { throw new Error('version number missing!'); } var modAll = [mod, '_', ver].join(''); modulesWaitingForDeps[modAll] = true; var cb = function() { //console.log(['createModule(', mod, ', ', ver, ', [', deps.join(', '), '], ', !!modFn, ')'].join('')); // make sure module in not loaded twice if (modules[modAll]) { //console.warn(['Ink.createModule ', modAll, ': module has been defined already.'].join('')); return; } // delete related pending tasks delete modulesRequested[modAll]; delete modulesRequested[mod]; // run module's supplied factory var args = Array.prototype.slice.call(arguments); var moduleContent = modFn.apply(window, args); modulesLoadOrder.push(modAll); // console.log('** loaded module ' + modAll + '**'); // set version if (typeof moduleContent === 'object') { // Dom.Css Dom.Event moduleContent._version = ver; } else if (typeof moduleContent === 'function') { moduleContent.prototype._version = ver; // if constructor moduleContent._version = ver; // if regular function } // add to global namespace... var isInkModule = mod.indexOf('Ink.') === 0; var t; if (isInkModule) { t = Ink.namespace(mod, true); // for mod 'Ink.Dom.Css', t[0] gets 'Ink.Dom' object and t[1] 'Css' } // versioned modules[ modAll ] = moduleContent; // in modules delete modulesWaitingForDeps[ modAll ]; if (isInkModule) { t[0][ t[1] + '_' + ver ] = moduleContent; // in namespace } // unversioned modules[ mod ] = moduleContent; // in modules if (isInkModule) { if (isEmptyObject( t[0][ t[1] ] )) { t[0][ t[1] ] = moduleContent; // in namespace } // else { // console.warn(['Ink.createModule ', modAll, ': module has been defined already with a different version!'].join('')); // } } if (this) { // there may be pending requires expecting this module, check... Ink._checkPendingRequireModules(); } }; this.requireModules(deps, cb); }, /** * Requires modules asynchronously * Use this to get modules, even if they're not loaded yet * * @method requireModules * @param {Array} deps Array of module names. The order in which they are passed here will define the order they will be passed to the callback function. * @param {Function} cbFn The callback function to be executed when all the dependencies are resolved. The dependencies are passed as arguments, in the same order they were declared. * @sample Ink_1_requireModules.html */ requireModules: function(deps, cbFn) { // require //console.log(['requireModules([', deps.join(', '), '], ', !!cbFn, ')'].join('')); var i, f, o, dep, mod; f = deps && deps.length; o = { args: new Array(f), left: {}, remaining: f, cb: cbFn }; if (!(typeof deps === 'object' && deps.length !== undefined)) { throw new Error('Dependency list should be an array!'); } if (typeof cbFn !== 'function') { throw new Error('Callback should be a function!'); } for (i = 0; i < f; ++i) { if (Ink._moduleRenames[deps[i]]) { Ink.warn(deps[i] + ' was renamed to ' + Ink._moduleRenames[deps[i]]); dep = Ink._moduleRenames[deps[i]]; } else { dep = deps[i]; } // Because trailing commas in oldIE bring us undefined values here if (!dep) { --o.remaining; continue; } mod = modules[dep]; if (mod) { o.args[i] = mod; --o.remaining; continue; } else if (!modulesRequested[dep]) { Ink._loadLater(dep); } o.left[dep] = i; } if (o.remaining > 0) { pendingRMs.push(o); } else { cbFn.apply(true, o.args); } }, _moduleRenames: { 'Ink.UI.Aux_1': 'Ink.UI.Common_1' }, /** * Lists loaded module names. * The list is ordered by loaded time (oldest module comes first) * * @method getModulesLoadOrder * @return {Array} returns the order in which modules were resolved and correctly loaded */ getModulesLoadOrder: function() { return modulesLoadOrder.slice(); }, /** * Builds the markup needed to load the modules. * This method builds the script tags needed to load the currently used modules * * @method getModuleScripts * @uses getModulesLoadOrder * @return {String} The script markup */ getModuleScripts: function() { var mlo = this.getModulesLoadOrder(); mlo.unshift('Ink_1'); mlo = mlo.map(function(m) { return [''].join(''); }); return mlo.join('\n'); }, /** * Creates an Ink.Ext module * * Does exactly the same as createModule but creates the module in the Ink.Ext namespace * * @method createExt * @uses createModule * @param {String} moduleName Extension name * @param {String} version Extension version * @param {Array} dependencies Extension dependencies * @param {Function} modFn Function returning the extension * @sample Ink_1_createExt.html */ createExt: function (moduleName, version, dependencies, modFn) { return Ink.createModule('Ink.Ext.' + moduleName, version, dependencies, modFn); }, /** * Function.prototype.bind alternative. * Creates a new function that, when called, has its this keyword set to the provided value, with a given sequence of arguments preceding any provided when the new function is called. * * @method bind * @param {Function} fn The function * @param {Object} context The value to be passed as the this parameter to the target function when the bound function is called. If used as false, it preserves the original context and just binds the arguments. * @param {Any} [args*] Additional arguments will be sent to the original function as prefix arguments. * @return {Function} * @sample Ink_1_bind.html */ bind: function(fn, context) { var args = Array.prototype.slice.call(arguments, 2); return function() { var innerArgs = Array.prototype.slice.call(arguments); var finalArgs = args.concat(innerArgs); return fn.apply(context === false ? this : context, finalArgs); }; }, /** * Function.prototype.bind alternative for class methods * Creates a new function that, when called, has this k * @method bindMethod * @uses bind * @param {Object} object The object that contains the method to bind * @param {String} methodName The name of the method that will be bound * @param {Any} [args*] Additional arguments will be sent to the new method as prefix arguments. * @return {Function} * @sample Ink_1_bindMethod.html */ bindMethod: function (object, methodName) { return Ink.bind.apply(Ink, [object[methodName], object].concat([].slice.call(arguments, 2))); }, /** * Function.prototype.bind alternative for event handlers. * Same as bind but keeps first argument of the call the original event. * Set "context" to `false` to preserve the original context of the function and just bind the arguments. * * @method bindEvent * @param {Function} fn The function * @param {Object} context The value to be passed as the this parameter to the target * @param {Any} [args*] Additional arguments will be sent to the original function as prefix arguments * @return {Function} * @sample Ink_1_bindEvent.html */ bindEvent: function(fn, context) { var args = Array.prototype.slice.call(arguments, 2); return function(event) { var finalArgs = args.slice(); finalArgs.unshift(event || window.event); return fn.apply(context === false ? this : context, finalArgs); }; }, /** * Alias to document.getElementById * * @method i * @param {String} id Element ID * @return {DOMElement} * @sample Ink_1_i.html */ i: function(id) { if(!id) { throw new Error('Ink.i => id or element must be passed'); } if(typeof(id) === 'string') { return document.getElementById(id); } return id; }, /** * Alias for Ink.Dom.Selector * * @method ss * @uses Ink.Dom.Selector.select * @param {String} rule * @param {DOMElement} [from] * @return {Array} array of DOMElements * @sample Ink_1_ss.html */ ss: function(rule, from) { if(typeof(Ink.Dom) === 'undefined' || typeof(Ink.Dom.Selector) === 'undefined') { throw new Error('This method requires Ink.Dom.Selector'); } return Ink.Dom.Selector.select(rule, (from || document)); }, /** * Alias for Ink.Dom.Selector first result * * @method s * @uses Ink.Dom.Selector.select * @param {String} rule Selector string * @param {DOMElement} [from] Context element. If set to a DOM element, the rule will only look for descendants of this DOM Element. * @return {DOMElement} * @sample Ink_1_s.html */ s: function(rule, from) { if(typeof(Ink.Dom) === 'undefined' || typeof(Ink.Dom.Selector) === 'undefined') { throw new Error('This method requires Ink.Dom.Selector'); } return Ink.Dom.Selector.select(rule, (from || document))[0] || null; }, /** * Extends an object with another * Copy all of the properties in one or more source objects over to the destination object, and return the destination object. It's in-order, so the last source will override properties of the same name in previous arguments. * * @method extendObj * @param {Object} destination The object that will receive the new/updated properties * @param {Object} source The object whose properties will be copied over to the destination object * @param {Object} [args*] Additional source objects. The last source will override properties of the same name in the previous defined sources * @return destination object, enriched with defaults from the sources * @sample Ink_1_extendObj.html */ extendObj: function(destination/*, source... */) { var sources = [].slice.call(arguments, 1); for (var i = 0, len = sources.length; i < len; i++) { if (!sources[i]) { continue; } for (var property in sources[i]) { if(Object.prototype.hasOwnProperty.call(sources[i], property)) { destination[property] = sources[i][property]; } } } return destination; }, /** * Calls native console.log if available. * * @method log * @param {Any} [args*] Arguments to be evaluated * @sample Ink_1_log.html **/ log: function () { // IE does not have console.log.apply in IE10 emulated mode var console = window.console; if (console && console.log) { apply.call(console.log, console, arguments); } }, /** * Calls native console.warn if available. * * @method warn * @param {Any} [args*] Arguments to be evaluated * @sample Ink_1_warn.html **/ warn: function () { // IE does not have console.log.apply in IE10 emulated mode var console = window.console; if (console && console.warn) { apply.call(console.warn, console, arguments); } }, /** * Calls native console.error if available. * * @method error * @param {Any} [args*] Arguments to be evaluated * @sample Ink_1_error.html **/ error: function () { // IE does not have console.log.apply in IE10 emulated mode var console = window.console; if (console && console.error) { apply.call(console.error, console, arguments); } } }; // TODO for debug - to detect pending stuff /* var failCount = {}; // fail count per module name var maxFails = 3; // times var checkDelta = 0.5; //seconds var tmpTmr = setInterval(function() { var mk = Object.keys(modulesRequested); var l = mk.length; if (l > 0) { // console.log('** waiting for modules: ' + mk.join(', ') + ' **'); for (var i = 0, f = mk.length, k, v; i < f; ++i) { k = mk[i]; v = failCount[k]; failCount[k] = (v === undefined) ? 1 : ++v; if (v >= maxFails) { console.error('** Loading of module ' + k + ' failed! **'); delete modulesRequested[k]; } } } else { // console.log('** Module loads complete. **'); clearInterval(tmpTmr); } }, checkDelta*1000); */ }(window, document)); /** * Cross Browser Ajax requests * @module Ink.Net.Ajax_1 * @version 1 */ Ink.createModule('Ink.Net.Ajax', '1', [], function() { 'use strict'; /** * Creates a new XMLHttpRequest object * * @class Ink.Net.Ajax * @constructor * * @param {String} url Request URL * @param {Object} options Request options * @param {Boolean} [options.asynchronous]=true If false, the request synchronous. * @param {Boolean} [options.cors] Flag to activate CORS. Set this to true if you're doing a cross-origin request * @param {String} [options.method]='POST' HTTP request method. POST by default. * @param {Object|String} [options.parameters] Request parameters to be sent with the request * @param {Number} [options.timeout] Request timeout in seconds * @param {Number} [options.delay] Artificial delay. If the request is completed faster than this delay, wait the remaining time before executing the callbacks * @param {String} [options.postBody] POST request body. If not specified, it's filled with the contents from parameters * @param {String} [options.contentType] Content-type header to be sent. Defaults to 'application/x-www-form-urlencoded' * @param {Object} [options.requestHeaders] Key-value pairs for additional request headers * @param {Function} [options.onComplete] Callback executed after the request is completed, regardless of what happened during the request. * @param {Function} [options.onSuccess] Callback executed if the request is successful (requests with 2xx status codes) * @param {Function} [options.onFailure] Callback executed if the request fails (requests with status codes different from 2xx) * @param {Function} [options.onException] Callback executed if an exception occurs. Receives the exception as a parameter. * @param {Function} [options.onCreate] Callback executed after object initialization but before the request is made * @param {Function} [options.onInit] Callback executed before any initialization * @param {Function} [options.onTimeout] Callback executed if the request times out * @param {Boolean|String} [options.evalJS]=true If the request Content-type header is application/json, evaluates the response and populates responseJSON. Use 'force' if you want to force the response evaluation, no matter what Content-type it's using. * @param {Boolean} [options.sanitizeJSON] Flag to sanitize the content of responseText before evaluation * @param {String} [options.xhrProxy] URI for proxy service hosted on the same server as the web app, that can fetch documents from other domains. The service must pipe all input and output untouched (some input sanitization is allowed, like clearing cookies). e.g., requesting http://example.org/doc can become /proxy/http%3A%2F%2Fexample.org%2Fdoc The proxy service will be used for cross-domain requests, if set, else a network error is returned as exception. * * @sample Ink_Net_Ajax_1.html */ var Ajax = function(url, options){ // start of AjaxMock patch - uncomment to enable it /*var AM = SAPO.Communication.AjaxMock; if (AM && !options.inMock) { if (AM.autoRecordThisUrl && AM.autoRecordThisUrl(url)) { return new AM.Record(url, options); } if (AM.mockThisUrl && AM.mockThisUrl(url)) { return new AM.Play(url, options, true); } }*/ // end of AjaxMock patch this.init(url, options); }; /** * Options for all requests. These can then be overriden for individual ones. */ Ajax.globalOptions = { parameters: {}, requestHeaders: {} }; // IE10 does not need XDomainRequest var xMLHttpRequestWithCredentials = 'XMLHttpRequest' in window && 'withCredentials' in (new XMLHttpRequest()); Ajax.prototype = { init: function(url, userOptions) { if (!url) { throw new Error("WRONG_ARGUMENTS_ERR"); } var options = Ink.extendObj({ asynchronous: true, method: 'POST', parameters: null, timeout: 0, delay: 0, postBody: '', contentType: 'application/x-www-form-urlencoded', requestHeaders: null, onComplete: null, onSuccess: null, onFailure: null, onException: null, onHeaders: null, onCreate: null, onInit: null, onTimeout: null, sanitizeJSON: false, evalJS: true, xhrProxy: '', cors: false, debug: false, useCredentials: false, signRequest: false }, Ajax.globalOptions); if (userOptions && typeof userOptions === 'object') { options = Ink.extendObj(options, userOptions); if (typeof userOptions.parameters === 'object') { options.parameters = Ink.extendObj(Ink.extendObj({}, Ajax.globalOptions.parameters), userOptions.parameters); } else if (userOptions.parameters !== null) { var globalParameters = this.paramsObjToStr(Ajax.globalOptions.parameters); if (globalParameters) { options.parameters = userOptions.parameters + '&' + globalParameters; } } options.requestHeaders = Ink.extendObj({}, Ajax.globalOptions.requestHeaders); options.requestHeaders = Ink.extendObj(options.requestHeaders, userOptions.requestHeaders); } this.options = options; this.safeCall('onInit'); this.url = url; var urlLocation = this._locationFromURL(url); this.isHTTP = this._locationIsHTTP(urlLocation); this.isCrossDomain = this._locationIsCrossDomain(urlLocation, location); this.requestHasBody = options.method.search(/^get|head$/i) < 0; if(this.options.cors) { this.isCrossDomain = false; } this.transport = this.getTransport(); this.request(); }, /** * Returns a location object from an URL * * @method _locationFromUrl * @param url * @private **/ _locationFromURL: function (url) { var urlLocation = document.createElementNS ? document.createElementNS('http://www.w3.org/1999/xhtml', 'a') : document.createElement('a'); urlLocation.href = url; return urlLocation; }, /** * Checks whether a location is HTTP or HTTPS * * @method locationIsHttp * @param urlLocation * @private */ _locationIsHTTP: function (urlLocation) { return urlLocation.protocol.match(/^https?:/i) ? true : false; }, /** * Checks whether a location is cross-domain from another * * @method _locationIsCrossDomain * @param urlLocation {Location} * @param otherLocation {Location} */ _locationIsCrossDomain: function (urlLocation, location) { location = location || window.location; if (!Ajax.prototype._locationIsHTTP(urlLocation) || location.protocol === 'widget:' || typeof window.widget === 'object') { return false; } else { return location.protocol !== urlLocation.protocol || location.host.split(':')[0] !== urlLocation.host.split(':')[0]; } }, /** * Creates the appropriate XMLHttpRequest object * * @method getTransport * @return {Object} XMLHttpRequest object */ getTransport: function() { /*global XDomainRequest:false, ActiveXObject:false */ if (!xMLHttpRequestWithCredentials && this.options.cors && 'XDomainRequest' in window) { this.usingXDomainReq = true; return new XDomainRequest(); } else if (typeof XMLHttpRequest !== 'undefined') { return new XMLHttpRequest(); } else if (typeof ActiveXObject !== 'undefined') { try { return new ActiveXObject('Msxml2.XMLHTTP'); } catch (e) { return new ActiveXObject('Microsoft.XMLHTTP'); } } else { return null; } }, /** * Set the necessary headers for an ajax request * * @method setHeaders * @param {String} url The url for the request */ setHeaders: function() { if (this.transport) { try { var headers = { "Accept": "text/javascript,text/xml,application/xml,application/xhtml+xml,text/html,application/json;q=0.9,text/plain;q=0.8,video/x-mng,image/png,image/jpeg,image/gif;q=0.2,*/*;q=0.1", "Accept-Language": navigator.language, "X-Requested-With": "XMLHttpRequest", "X-Ink-Version": "2" }; if (this.options.cors) { if (!this.options.signRequest) { delete headers['X-Requested-With']; } delete headers['X-Ink-Version']; } if (this.options.requestHeaders && typeof this.options.requestHeaders === 'object') { for(var headerReqName in this.options.requestHeaders) { if (this.options.requestHeaders.hasOwnProperty(headerReqName)) { headers[headerReqName] = this.options.requestHeaders[headerReqName]; } } } if (this.transport.overrideMimeType && (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005) { headers.Connection = 'close'; } for (var headerName in headers) { if(headers.hasOwnProperty(headerName)) { this.transport.setRequestHeader(headerName, headers[headerName]); } } } catch(e) {} } }, /** * Converts an object with parameters to a querystring * * @method paramsObjToStr * @param {Object|String} optParams parameters object * @return {String} querystring */ paramsObjToStr: function(optParams) { var k, m, p, a, params = []; if (typeof optParams === 'object') { for (p in optParams){ if (optParams.hasOwnProperty(p)) { a = optParams[p]; if (Object.prototype.toString.call(a) === '[object Array]' && !isNaN(a.length)) { for (k = 0, m = a.length; k < m; k++) { params = params.concat([ encodeURIComponent(p), '[]', '=', encodeURIComponent(a[k]), '&' ]); } } else { params = params.concat([ encodeURIComponent(p), '=', encodeURIComponent(a), '&' ]); } } } if (params.length > 0) { params.pop(); } } else { return optParams; } return params.join(''); }, /** * Set the url parameters for a GET request * * @method setParams */ setParams: function() { var params = null, optParams = this.options.parameters; if(typeof optParams === "object"){ params = this.paramsObjToStr(optParams); } else { params = '' + optParams; } if(params){ if(this.url.indexOf('?') > -1) { this.url = this.url.split('#')[0] + '&' + params; } else { this.url = this.url.split('#')[0] + '?' + params; } } }, /** * Gets an HTTP header from the response * * @method getHeader * @param {String} name Header name * @return {String} header Content */ getHeader: function(name) { if (this.usingXDomainReq && name === 'Content-Type') { return this.transport.contentType; } try{ return this.transport.getResponseHeader(name); } catch(e) { return null; } }, /** * Gets all the HTTP headers from the response * * @method getAllHeaders * @return {String} The headers, each separated by a newline */ getAllHeaders: function() { try { return this.transport.getAllResponseHeaders(); } catch(e) { return null; } }, /** * Sets the response object * * @method getResponse * @return {Object} the response object */ getResponse: function(){ // setup our own stuff var t = this.transport, r = { headerJSON: null, responseJSON: null, getHeader: this.getHeader, getAllHeaders: this.getAllHeaders, request: this, transport: t, timeTaken: new Date() - this.startTime, requestedUrl: this.url }; // setup things expected from the native object r.readyState = t.readyState; try { r.responseText = t.responseText; } catch(e) {} try { r.responseXML = t.responseXML; } catch(e) {} try { r.status = t.status; } catch(e) { r.status = 0; } try { r.statusText = t.statusText; } catch(e) { r.statusText = ''; } return r; }, /** * Aborts the request if still running. No callbacks are called * * @method abort */ abort: function(){ if (this.transport) { clearTimeout(this.delayTimeout); clearTimeout(this.stoTimeout); try { this.transport.abort(); } catch(ex) {} this.finish(); } }, /** * Executes the state changing phase of an ajax request * * @method runStateChange */ runStateChange: function() { var rs = this.transport.readyState; if (rs === 3) { if (this.isHTTP) { this.safeCall('onHeaders'); } } else if (rs === 4 || this.usingXDomainReq) { if (this.options.asynchronous && this.options.delay && (this.startTime + this.options.delay > new Date().getTime())) { this.delayTimeout = setTimeout(Ink.bind(this.runStateChange, this), this.options.delay + this.startTime - new Date().getTime()); return; } var responseJSON, responseContent = this.transport.responseText, response = this.getResponse(), curStatus = this.transport.status; if (this.isHTTP && !this.options.asynchronous) { this.safeCall('onHeaders'); } clearTimeout(this.stoTimeout); if (curStatus === 0) { // Status 0 indicates network error for http requests. // For http less requests, 0 is always returned. if (this.isHTTP) { this.safeCall('onException', this.makeError(18, 'NETWORK_ERR')); } else { curStatus = responseContent ? 200 : 404; } } else if (curStatus === 304) { curStatus = 200; } var isSuccess = this.usingXDomainReq || 200 <= curStatus && curStatus < 300; var headerContentType = this.getHeader('Content-Type') || ''; if (this.options.evalJS && (headerContentType.indexOf("application/json") >= 0 || this.options.evalJS === 'force')){ try { responseJSON = this.evalJSON(responseContent, this.sanitizeJSON); if(responseJSON){ responseContent = response.responseJSON = responseJSON; } } catch(e){ if (isSuccess) { // If the request failed, then this is perhaps an error page // so don't notify error. this.safeCall('onException', e); } } } if (this.usingXDomainReq && headerContentType.indexOf('xml') !== -1 && 'DOMParser' in window) { // http://msdn.microsoft.com/en-us/library/ie/ff975278(v=vs.85).aspx var mimeType; switch (headerContentType) { case 'application/xml': case 'application/xhtml+xml': case 'image/svg+xml': mimeType = headerContentType; break; default: mimeType = 'text/xml'; } var xmlDoc = (new DOMParser()).parseFromString( this.transport.responseText, mimeType); this.transport.responseXML = xmlDoc; response.responseXML = xmlDoc; } if (this.transport.responseXML !== null && response.responseJSON === null && this.transport.responseXML.xml !== ""){ responseContent = this.transport.responseXML; } if (curStatus || this.usingXDomainReq) { if (isSuccess) { this.safeCall('onSuccess', response, responseContent); } else { this.safeCall('onFailure', response, responseContent); } this.safeCall('on'+curStatus, response, responseContent); } this.finish(response, responseContent); } }, /** * Last step after XHR is complete. Call onComplete and cleanup object * * @method finish * @param {Any} response * @param {Any} responseContent */ finish: function(response, responseContent){ if (response) { this.safeCall('onComplete', response, responseContent); } clearTimeout(this.stoTimeout); if (this.transport) { // IE6 sometimes barfs on this one try{ this.transport.onreadystatechange = null; } catch(e){} if (typeof this.transport.destroy === 'function') { // Stuff for Samsung. this.transport.destroy(); } // Let XHR be collected. this.transport = null; } }, /** * Safely calls a callback function. * Verifies that the callback is well defined and traps errors * * @method safeCall * @param {Function} listener */ safeCall: function(listener, first/*, second*/) { function rethrow(exception){ setTimeout(function() { // Rethrow exception so it'll land in // the error console, firebug, whatever. if (exception.message) { exception.message += '\n'+(exception.stacktrace || exception.stack || ''); } throw exception; }, 1); } if (typeof this.options[listener] === 'function') { //SAPO.safeCall(this, this.options[listener], first, second); //return object[listener].apply(object, [].slice.call(arguments, 2)); try { this.options[listener].apply(this, [].slice.call(arguments, 1)); } catch(ex) { rethrow(ex); } } else if (first && window.Error && (first instanceof Error)) { rethrow(first); } }, /** * Sets a new request header for the next http request * * @method setRequestHeader * @param {String} name * @param {String} value */ setRequestHeader: function(name, value){ if (!this.options.requestHeaders) { this.options.requestHeaders = {}; } this.options.requestHeaders[name] = value; }, /** * Executes the request * * @method request */ request: function() { if(this.transport) { var params = null; if(this.requestHasBody) { if(this.options.postBody !== null && this.options.postBody !== '') { params = this.options.postBody; this.setParams(); } else if (this.options.parameters !== null && this.options.parameters !== ''){ params = this.options.parameters; } if (typeof params === "object" && !params.nodeType) { params = this.paramsObjToStr(params); } else if (typeof params !== "object" && params !== null){ params = '' + params; } if(this.options.contentType) { this.setRequestHeader('Content-Type', this.options.contentType); } } else { this.setParams(); } var url = this.url; var method = this.options.method; var crossDomain = this.isCrossDomain; if (crossDomain && this.options.xhrProxy) { this.setRequestHeader('X-Url', url); url = this.options.xhrProxy + encodeURIComponent(url); crossDomain = false; } try { this.transport.open(method, url, this.options.asynchronous); } catch(e) { this.safeCall('onException', e); return this.finish(this.getResponse(), null); } this.setHeaders(); this.safeCall('onCreate'); if(this.options.timeout && !isNaN(this.options.timeout)) { this.stoTimeout = setTimeout(Ink.bind(function() { if(this.options.onTimeout) { this.safeCall('onTimeout'); this.abort(); } }, this), (this.options.timeout * 1000)); } if(this.options.useCredentials && !this.usingXDomainReq) { this.transport.withCredentials = true; } if(this.options.asynchronous && !this.usingXDomainReq) { this.transport.onreadystatechange = Ink.bind(this.runStateChange, this); } else if (this.usingXDomainReq) { this.transport.onload = Ink.bind(this.runStateChange, this); } try { if (crossDomain) { // Need explicit handling because Mozila aborts // the script and Chrome fails silently.per the spec throw this.makeError(18, 'NETWORK_ERR'); } else { this.startTime = new Date().getTime(); this.transport.send(params); } } catch(e) { this.safeCall('onException', e); return this.finish(this.getResponse(), null); } if(!this.options.asynchronous) { this.runStateChange(); } } }, /** * Returns a new exception object that can be thrown * * @method makeError * @param code Error Code * @param message Message * @returns {Object} */ makeError: function(code, message){ if (typeof Error !== 'function') { return {code: code, message: message}; } var e = new Error(message); e.code = code; return e; }, /** * Checks if a given string is valid JSON * * @method isJSON * @param {String} str String to be evaluated * @return {Boolean} True if the string is valid JSON */ isJSON: function(str) { if (typeof str !== "string" || !str){ return false; } str = str.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, ''); return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str); }, /** * Evaluates a given string as JSON * * @method evalJSON * @param {String} str String to be evaluated * @param {Boolean} sanitize Flag to sanitize the content * @return {Object} JSON content as an object */ evalJSON: function(strJSON, sanitize) { if (strJSON && (!sanitize || this.isJSON(strJSON))) { try { if (typeof JSON !== "undefined" && typeof JSON.parse !== 'undefined'){ return JSON.parse(strJSON); } /*jshint evil:true */ return eval('(' + strJSON + ')'); } catch(e) { throw new Error('ERROR: Bad JSON string...'); } } return null; } }; /** * Loads content from a given url through an XMLHttpRequest. * * Shortcut function for simple AJAX use cases. Works with JSON, XML and plain text. * * @method load * @param {String} url Request URL * @param {Function} callback Callback to be executed if the request is successful * @return {Object} XMLHttpRequest object * * @sample Ink_Net_Ajax_load.html */ Ajax.load = function(url, callback){ return new Ajax(url, { method: 'GET', onSuccess: function(response){ callback(response.responseJSON || response.responseText, response); } }); }; /** * Loads content from a given url through an XMLHttpRequest. * Shortcut function for simple AJAX use cases. * * @method ping * @param {String} url Request url * @param {Function} callback Callback to be executed if the request is successful * @return {Object} XMLHttpRequest object */ Ajax.ping = function(url, callback){ return new Ajax(url, { method: 'HEAD', onSuccess: function(response){ if (typeof callback === 'function'){ callback(response); } } }); }; return Ajax; }); /** * Cross Browser JsonP requests * @module Ink.Net.JsonP_1 * @version 1 */ Ink.createModule('Ink.Net.JsonP', '1', [], function() { 'use strict'; /** * Executes a JSONP request * * @class Ink.Net.JsonP * @constructor * * @param {String} uri Request URL * @param {Object} options Request options * @param {Function} options.onSuccess Success callback * @param {Function} [options.onFailure] Failure callback * @param {Object} [options.failureObj] Object to be passed as argument to failure callback * @param {Number} [options.timeout] Timeout for request fail, in seconds. defaults to 10 * @param {Object} [options.params] Object with the parameters and respective values to unfold * @param {String} [options.callbackParam] Parameter to use as callback. defaults to 'jsoncallback' * @param {String} [options.internalCallback] Name of the callback function stored in the Ink.Net.JsonP object. * @param {String} [options.randVar] (Advanced, not recommended unless you know what you're doing) A string to append to the callback name. By default, generate a random number. Use an empty string if you already passed the correct name in the internalCallback option. * * @sample Ink_Net_JsonP_1.html */ var JsonP = function(uri, options) { this.init(uri, options); }; JsonP.prototype = { init: function(uri, options) { this.options = Ink.extendObj( { onSuccess: undefined, onFailure: undefined, failureObj: {}, timeout: 10, params: {}, callbackParam: 'jsoncallback', internalCallback: '_cb', randVar: false }, options || {}); if(this.options.randVar !== false) { this.randVar = this.options.randVar; } else { this.randVar = parseInt(Math.random() * 100000, 10); } this.options.internalCallback += this.randVar; this.uri = uri; // prevent SAPO legacy onComplete - make it onSuccess if(typeof(this.options.onComplete) === 'function') { this.options.onSuccess = this.options.onComplete; } if (typeof this.uri !== 'string') { throw 'Please define an URI'; } if (typeof this.options.onSuccess !== 'function') { throw 'please define a callback function on option onSuccess!'; } Ink.Net.JsonP[this.options.internalCallback] = Ink.bind(function() { window.clearTimeout(this.timeout); delete window.Ink.Net.JsonP[this.options.internalCallback]; this._removeScriptTag(); this.options.onSuccess(arguments[0]); }, this); this._addScriptTag(); }, _addParamsToGet: function(uri, params) { var hasQuestionMark = uri.indexOf('?') !== -1; var sep, pKey, pValue, parts = [uri]; for (pKey in params) { if (params.hasOwnProperty(pKey)) { if (!hasQuestionMark) { sep = '?'; hasQuestionMark = true; } else { sep = '&'; } pValue = params[pKey]; if (typeof pValue !== 'number' && !pValue) { pValue = ''; } parts = parts.concat([sep, pKey, '=', encodeURIComponent(pValue)]); } } return parts.join(''); }, _getScriptContainer: function() { var headEls = document.getElementsByTagName('head'); if (headEls.length === 0) { var scriptEls = document.getElementsByTagName('script'); return scriptEls[0]; } return headEls[0]; }, _addScriptTag: function() { // enrich options will callback and random seed this.options.params[this.options.callbackParam] = 'Ink.Net.JsonP.' + this.options.internalCallback; this.options.params.rnd_seed = this.randVar; this.uri = this._addParamsToGet(this.uri, this.options.params); // create script tag var scriptEl = document.createElement('script'); scriptEl.type = 'text/javascript'; scriptEl.src = this.uri; var scriptCtn = this._getScriptContainer(); scriptCtn.appendChild(scriptEl); this.timeout = setTimeout(Ink.bind(this._requestFailed, this), (this.options.timeout * 1000)); }, _requestFailed : function () { delete Ink.Net.JsonP[this.options.internalCallback]; this._removeScriptTag(); if(typeof this.options.onFailure === 'function'){ this.options.onFailure(this.options.failureObj); } }, _removeScriptTag: function() { var scriptEl; var scriptEls = document.getElementsByTagName('script'); var scriptUri; for (var i = 0, f = scriptEls.length; i < f; ++i) { scriptEl = scriptEls[i]; scriptUri = scriptEl.getAttribute('src') || scriptEl.src; if (scriptUri !== null && scriptUri === this.uri) { scriptEl.parentNode.removeChild(scriptEl); return; } } } }; return JsonP; }); /** * Browser Detection and User Agent sniffing * @module Ink.Dom.Browser_1 * @version 1 */ Ink.createModule('Ink.Dom.Browser', '1', [], function() { 'use strict'; /** * @namespace Ink.Dom.Browser * @version 1 * @static * @example * */ var Browser = { /** * True if the browser is Internet Explorer * * @property IE * @type {Boolean} * @public * @static */ IE: false, /** * True if the browser is Gecko based * * @property GECKO * @type {Boolean} * @public * @static */ GECKO: false, /** * True if the browser is Opera * * @property OPERA * @type {Boolean} * @public * @static */ OPERA: false, /** * True if the browser is Safari * * @property SAFARI * @type {Boolean} * @public * @static */ SAFARI: false, /** * True if the browser is Konqueror * * @property KONQUEROR * @type {Boolean} * @public * @static */ KONQUEROR: false, /** * True if browser is Chrome * * @property CHROME * @type {Boolean} * @public * @static */ CHROME: false, /** * The specific browser model. * False if it is unavailable. * * @property model * @type {Boolean|String} * @public * @static */ model: false, /** * The browser version. * False if it is unavailable. * * @property version * @type {Boolean|String} * @public * @static */ version: false, /** * The user agent string. * False if it is unavailable. * * @property userAgent * @type {Boolean|String} * @public * @static */ userAgent: false, /** * The CSS prefix (-moz-, -webkit-, -ms-, ...) * False if it is unavailable * * @property cssPrefix * @type {Boolean|String} * @public * @static */ cssPrefix: false, /** * The DOM prefix (Moz, Webkit, ms, ...) * False if it is unavailable * @property domPrefix * @type {Boolean|String} * @public * @static */ domPrefix: false, /** * Initialization function for the Browser object. * * Is called automatically when this module is loaded, and calls setDimensions, setBrowser and setReferrer. * * @method init * @public */ init: function() { this.detectBrowser(); this.setDimensions(); this.setReferrer(); }, /** * Retrieves and stores window dimensions in this object. Called automatically when this module is loaded. * * @method setDimensions * @public */ setDimensions: function() { //this.windowWidth=window.innerWidth !== null? window.innerWidth : document.documentElement && document.documentElement.clientWidth ? document.documentElement.clientWidth : document.body !== null ? document.body.clientWidth : null; //this.windowHeight=window.innerHeight != null? window.innerHeight : document.documentElement && document.documentElement.clientHeight ? document.documentElement.clientHeight : document.body != null? document.body.clientHeight : null; var myWidth = 0, myHeight = 0; if ( typeof window.innerWidth=== 'number' ) { myWidth = window.innerWidth; myHeight = window.innerHeight; } else if( document.documentElement && ( document.documentElement.clientWidth || document.documentElement.clientHeight ) ) { myWidth = document.documentElement.clientWidth; myHeight = document.documentElement.clientHeight; } else if( document.body && ( document.body.clientWidth || document.body.clientHeight ) ) { myWidth = document.body.clientWidth; myHeight = document.body.clientHeight; } this.windowWidth = myWidth; this.windowHeight = myHeight; }, /** * Stores the referrer. Called automatically when this module is loaded. * * @method setReferrer * @public */ setReferrer: function() { if (document.referrer && document.referrer.length) { this.referrer = window.escape(document.referrer); } else { this.referrer = false; } }, /** * Detects the browser and stores the found properties. Called automatically when this module is loaded. * * @method detectBrowser * @public */ detectBrowser: function() { this._sniffUserAgent(navigator.userAgent); }, _sniffUserAgent: function (sAgent) { this.userAgent = sAgent; sAgent = sAgent.toLowerCase(); if (/applewebkit\//.test(sAgent)) { this.cssPrefix = '-webkit-'; this.domPrefix = 'Webkit'; if(/(chrome|crios)\//.test(sAgent)) { // Chrome this.CHROME = true; this.model = 'chrome'; this.version = sAgent.replace(/(.*)chrome\/([^\s]+)(.*)/, "$2"); } else { // Safari this.SAFARI = true; this.model = 'safari'; var rVersion = /version\/([^) ]+)/; if (rVersion.test(sAgent)) { this.version = sAgent.match(rVersion)[1]; } else { this.version = sAgent.replace(/(.*)applewebkit\/([^\s]+)(.*)/, "$2"); } } } else if (/opera/.test(sAgent)) { // Opera this.OPERA = true; this.model = 'opera'; this.version = sAgent.replace(/(.*)opera.([^\s$]+)(.*)/, "$2"); this.cssPrefix = '-o-'; this.domPrefix = 'O'; } else if (/konqueror/.test(sAgent)) { // Konqueroh this.KONQUEROR = true; this.model = 'konqueror'; this.version = sAgent.replace(/(.*)konqueror\/([^;]+);(.*)/, "$2"); this.cssPrefix = '-khtml-'; this.domPrefix = 'Khtml'; } else if (/(msie|trident)/i.test(sAgent)) { // MSIE this.IE = true; this.model = 'ie'; if (/rv:((?:\d|\.)+)/.test(sAgent)) { // IE 11 this.version = sAgent.match(/rv:((?:\d|\.)+)/)[1]; } else { this.version = sAgent.replace(/(.*)\smsie\s([^;]+);(.*)/, "$2"); } this.cssPrefix = '-ms-'; this.domPrefix = 'ms'; } else if (/gecko/.test(sAgent)) { // GECKO // Supports only: // Camino, Chimera, Epiphany, Minefield (firefox 3), Firefox, Firebird, Phoenix, Galeon, // Iceweasel, K-Meleon, SeaMonkey, Netscape, Songbird, Sylera, this.cssPrefix = '-moz-'; this.domPrefix = 'Moz'; this.GECKO = true; var re = /(camino|chimera|epiphany|minefield|firefox|firebird|phoenix|galeon|iceweasel|k\-meleon|seamonkey|netscape|songbird|sylera)/; if(re.test(sAgent)) { this.model = sAgent.match(re)[1]; this.version = sAgent.replace(new RegExp("(.*)"+this.model+"\/([^;\\s$]+)(.*)"), "$2"); } else { // probably is mozilla this.model = 'mozilla'; var reVersion = /(.*)rv:([^)]+)(.*)/; if(reVersion.test(sAgent)) { this.version = sAgent.replace(reVersion, "$2"); } } } }, /** * Debug function which displays browser (and Ink.Dom.Browser) information as an alert message. * * @method debug * @public * @sample Ink_Dom_Browser_1_debug.html */ debug: function() { /*global alert:false */ var str = "known browsers: (ie, gecko, opera, safari, konqueror) \n"; str += [this.IE, this.GECKO, this.OPERA, this.SAFARI, this.KONQUEROR] +"\n"; str += "cssPrefix -> "+this.cssPrefix+"\n"; str += "domPrefix -> "+this.domPrefix+"\n"; str += "model -> "+this.model+"\n"; str += "version -> "+this.version+"\n"; str += "\n"; str += "original UA -> "+this.userAgent; alert(str); } }; Browser.init(); return Browser; }); /** * CSS Utilities and toolbox * @module Ink.Dom.Css_1 * @version 1 */ Ink.createModule( 'Ink.Dom.Css', 1, [], function() { 'use strict'; // getComputedStyle feature detection. var getCs = ("defaultView" in document) && ("getComputedStyle" in document.defaultView) ? document.defaultView.getComputedStyle : window.getComputedStyle; /** * @namespace Ink.Dom.Css * @static */ var Css = { /** * Adds of removes a class. * Depending on addRemState, this method either adds a class if it's true or removes if if false. * * @method addRemoveClassName * @param {DOMElement|string} elm DOM element or element id * @param {string} className class name to add or remove. * @param {boolean} addRemState Whether to add or remove. `true` to add, `false` to remove. * @sample Ink_Dom_Css_addRemoveClassName.html */ addRemoveClassName: function(elm, className, addRemState) { if (addRemState) { return this.addClassName(elm, className); } this.removeClassName(elm, className); }, /** * Adds a class to a given element * * @method addClassName * @param {DOMElement|String} elm DOM element or element id * @param {String|Array} className Classes * @sample Ink_Dom_Css_addClassName.html */ addClassName: function(elm, className) { elm = Ink.i(elm); if (!elm || !className) { return null; } className = ('' + className).split(/[, ]+/); var i = 0; var len = className.length; for (; i < len; i++) { // remove whitespace and ignore on empty string if (className[i].replace(/^\s+|\s+$/g, '')) { if (typeof elm.classList !== "undefined") { elm.classList.add(className[i]); } else if (!Css.hasClassName(elm, className[i])) { elm.className += (elm.className ? ' ' : '') + className[i]; } } } }, /** * Removes a class from a given element * * @method removeClassName * @param {DOMElement|String} elm DOM element or element id * @param {String|Array} className Class names to remove. You can either use a space separated string of classnames, comma-separated list or an array * @sample Ink_Dom_Css_removeClassName.html */ removeClassName: function(elm, className) { elm = Ink.i(elm); if (!elm || !className) { return null; } className = ('' + className).split(/[, ]+/); var i = 0; var len = className.length; if (typeof elm.classList !== "undefined"){ for (; i < len; i++) { elm.classList.remove(className[i]); } } else { var elmClassName = elm.className || ''; var re; for (; i < len; i++) { re = new RegExp("(^|\\s+)" + className[i] + "(\\s+|$)"); elmClassName = elmClassName.replace(re, ' '); } elm.className = (elmClassName .replace(/^\s+/, '') .replace(/\s+$/, '')); } }, /** * Alias to addRemoveClassName. * Utility function, saves many if/elses. * * @method setClassName * @uses addRemoveClassName * @param {DOMElement|String} elm DOM element or element id * @param {String|Array} className Class names to add\remove. Comma separated, space separated or simply an Array * @param {Boolean} [add]=false Flag to switch behavior from removal to addition. true to add, false to remove */ setClassName: function(elm, className, add) { this.addRemoveClassName(elm, className, add || false); }, /** * Checks if an element has a class. * This method verifies if an element has ONE of a list of classes. If the last argument is flagged as true, instead checks if the element has ALL the classes * * @method hasClassName * @param {DOMElement|String} elm DOM element or element id * @param {String|Array} className Class names to test * @param {Boolean} [all]=false If flagged as true, it will check if the element contains ALL the CSS classes * @return {Boolean} true if a given class is applied to a given element * @sample Ink_Dom_Css_hasClassName.html */ hasClassName: function(elm, className, all) { elm = Ink.i(elm); if (!elm || !className) { return false; } className = ('' + className).split(/[, ]+/); var i = 0; var len = className.length; var has; var re; for ( ; i < len; i++) { if (typeof elm.classList !== "undefined"){ has = elm.classList.contains(className[i]); } else { var elmClassName = elm.className; if (elmClassName === className[i]) { has = true; } else { re = new RegExp("(^|\\s)" + className[i] + "(\\s|$)"); has = re.test(elmClassName); } } if (has && !all) { return true; } // return if looking for any class if (!has && all) { return false; } // return if looking for all classes } if (all) { // if we got here, all classes were found so far return true; } else { // if we got here with all == false, no class was found return false; } }, /** * Blinks a class from an element * Add and removes the class from the element with a timeout, so it blinks * * @method blinkClass * @uses addRemoveClassName * @param {DOMElement|String} elm DOM element or element id * @param {String|Array} className Class name(s) to blink * @param {Number} timeout timeout in ms between adding and removing, default 100 ms * @param {Boolean} negate is true, class is removed then added * @sample Ink_Dom_Css_blinkClass.html */ blinkClass: function(element, className, timeout, negate){ element = Ink.i(element); Css.addRemoveClassName(element, className, !negate); setTimeout(function() { Css.addRemoveClassName(element, className, negate); }, Number(timeout) || 100); }, /** * Toggles a class name from a given element * * @method toggleClassName * @param {DOMElement|String} elm DOM element or element id * @param {String} className Class name * @param {Boolean} [forceAdd] Flag to force adding the the classe names if they don't exist yet. * @sample Ink_Dom_Css_toggleClassName.html */ toggleClassName: function(elm, className, forceAdd) { if (elm && className){ if (typeof elm.classList !== "undefined" && !/[, ]/.test(className)){ elm = Ink.i(elm); if (elm !== null){ elm.classList.toggle(className); } return true; } } if (typeof forceAdd !== 'undefined') { if (forceAdd === true) { Css.addClassName(elm, className); } else if (forceAdd === false) { Css.removeClassName(elm, className); } } else { if (Css.hasClassName(elm, className)) { Css.removeClassName(elm, className); } else { Css.addClassName(elm, className); } } }, /** * Sets the opacity of given element * * @method setOpacity * @param {DOMElement|String} elm DOM element or element id * @param {Number} value allows 0 to 1(default mode decimal) or percentage (warning using 0 or 1 will reset to default mode) * @sample Ink_Dom_Css_setOpacity.html */ setOpacity: function(elm, value) { elm = Ink.i(elm); if (elm !== null){ var val = 1; if (!isNaN(Number(value))){ if (value <= 0) { val = 0; } else if (value <= 1) { val = value; } else if (value <= 100) { val = value / 100; } else { val = 1; } } if (typeof elm.style.opacity !== 'undefined') { elm.style.opacity = val; } else { elm.style.filter = "alpha(opacity:"+(val*100|0)+")"; } } }, /** * Converts a css property name to a string in camelcase to be used with CSSStyleDeclaration. * @method _camelCase * @private * @param {String} str String to convert * @return {String} Converted string */ _camelCase: function(str) { return str ? str.replace(/-(\w)/g, function (_, $1) { return $1.toUpperCase(); }) : str; }, /** * Gets the value for an element's style attribute * * @method getStyle * @param {DOMElement|String} elm DOM element or element id * @param {String} style Which css attribute to fetch * @return Style value * @sample Ink_Dom_Css_getStyle.html */ getStyle: function(elm, style) { elm = Ink.i(elm); if (elm !== null && elm.style) { style = style === 'float' ? 'cssFloat': this._camelCase(style); var value = elm.style[style]; if (getCs && (!value || value === 'auto')) { var css = getCs(elm, null); value = css ? css[style] : null; } else if (!value && elm.currentStyle) { value = elm.currentStyle[style]; if (value === 'auto' && (style === 'width' || style === 'height')) { value = elm["offset" + style.charAt(0).toUpperCase() + style.slice(1)] + "px"; } } if (style === 'opacity') { return value ? parseFloat(value, 10) : 1.0; } else if (style === 'borderTopWidth' || style === 'borderBottomWidth' || style === 'borderRightWidth' || style === 'borderLeftWidth' ) { if (value === 'thin') { return '1px'; } else if (value === 'medium') { return '3px'; } else if (value === 'thick') { return '5px'; } } return value === 'auto' ? null : value; } }, /** * Adds CSS rules to an element's style attribute. * * @method setStyle * @param {DOMElement|String} elm DOM element or element id * @param {String} style Which css attribute to set * @sample Ink_Dom_Css_setStyle.html */ setStyle: function(elm, style) { elm = Ink.i(elm); if (elm === null) { return; } if (typeof style === 'string') { elm.style.cssText += '; '+style; if (style.indexOf('opacity') !== -1) { this.setOpacity(elm, style.match(/opacity:\s*(\d?\.?\d*)/)[1]); } } else { for (var prop in style) { if (style.hasOwnProperty(prop)){ if (prop === 'opacity') { this.setOpacity(elm, style[prop]); } else if (prop === 'float' || prop === 'cssFloat') { if (typeof elm.style.styleFloat === 'undefined') { elm.style.cssFloat = style[prop]; } else { elm.style.styleFloat = style[prop]; } } else { elm.style[prop] = style[prop]; } } } } }, /** * Shows an element. * Internally it unsets the display property of an element. You can force a specific display property using forceDisplayProperty * * @method show * @param {DOMElement|String} elm DOM element or element id * @param {String} [forceDisplayProperty] Css display property to apply on show * @sample Ink_Dom_Css_show.html */ show: function(elm, forceDisplayProperty) { elm = Ink.i(elm); if (elm !== null) { elm.style.display = (forceDisplayProperty) ? forceDisplayProperty : ''; } }, /** * Hides an element. * * @method hide * @param {DOMElement|String} elm DOM element or element id * @sample Ink_Dom_Css_hide.html */ hide: function(elm) { elm = Ink.i(elm); if (elm !== null) { elm.style.display = 'none'; } }, /** * Shows or hides an element. * If the show parameter is true, it shows the element. Otherwise, hides it. * * @method showHide * @param {DOMElement|String} elm DOM element or element id * @param {boolean} [show]=false Whether to show or hide `elm`. * @sample Ink_Dom_Css_showHide.html */ showHide: function(elm, show) { elm = Ink.i(elm); if (elm) { elm.style.display = show ? '' : 'none'; } }, /** * Toggles an element visibility. * * @method toggle * @param {DOMElement|String} elm DOM element or element id * @param {Boolean} forceShow Forces showing if element is hidden * @sample Ink_Dom_Css_toggle.html */ toggle: function(elm, forceShow) { elm = Ink.i(elm); if (elm !== null) { if (typeof forceShow !== 'undefined') { if (forceShow === true) { this.show(elm); } else { this.hide(elm); } } else { if (this.getStyle(elm,'display').toLowerCase() === 'none') { this.show(elm); } else { this.hide(elm); } } } }, _getRefTag: function(head){ if (head.firstElementChild) { return head.firstElementChild; } for (var child = head.firstChild; child; child = child.nextSibling){ if (child.nodeType === 1){ return child; } } return null; }, /** * Injects style tags with rules to the page. * * @method appendStyleTag * @param {String} selector The css selector for the rule * @param {String} style The content of the style rule * @param {Object} options Options for the tag * @param {String} [options.type]='text/css' File type * @param {Boolean} [options.force]=false If true, the style tag will be appended to end of head * * @sample Ink_Dom_Css_appendStyleTag.html */ appendStyleTag: function(selector, style, options){ options = Ink.extendObj({ type: 'text/css', force: false }, options || {}); var styles = document.getElementsByTagName("style"), oldStyle = false, setStyle = true, i, l; for (i=0, l=styles.length; i= 0) { setStyle = false; } } if (setStyle) { var defStyle = document.createElement("style"), head = document.getElementsByTagName("head")[0], refTag = false, styleStr = ''; defStyle.type = options.type; styleStr += selector +" {"; styleStr += style; styleStr += "} "; if (typeof defStyle.styleSheet !== "undefined") { defStyle.styleSheet.cssText = styleStr; } else { defStyle.appendChild(document.createTextNode(styleStr)); } if (options.force){ head.appendChild(defStyle); } else { refTag = this._getRefTag(head); if (refTag){ head.insertBefore(defStyle, refTag); } } } }, /** * Injects an external link tag. * This method add a stylesheet to the head of a page * * @method appendStylesheet * @param {String} path File path * @param {Object} options Options for the tag * @param {String} [options.media]='screen' Media type * @param {String} [options.type]='text/css' File type * @param {Boolean} [options.force]=false If true, tag will be appended to end of head * @sample Ink_Dom_Css_appendStylesheet.html */ appendStylesheet: function(path, options){ options = Ink.extendObj({ media: 'screen', type: 'text/css', force: false }, options || {}); var refTag, style = document.createElement("link"), head = document.getElementsByTagName("head")[0]; style.media = options.media; style.type = options.type; style.href = path; style.rel = "Stylesheet"; if (options.force){ head.appendChild(style); } else { refTag = this._getRefTag(head); if (refTag){ head.insertBefore(style, refTag); } } }, /** * Injects an external link tag. * Loads CSS via LINK element inclusion in HEAD (skips append if already there) * * Works similarly to appendStylesheet but: * supports optional callback which gets invoked once the CSS has been applied * * @method appendStylesheetCb * @param {String} cssURI URI of the CSS to load, if empty ignores and just calls back directly * @param {Function(cssURI)} [callback] optional callback which will be called once the CSS is loaded * @sample Ink_Dom_Css_appendStylesheetCb.html */ _loadingCSSFiles: {}, _loadedCSSFiles: {}, appendStylesheetCb: function(url, callback) { if (!url) { return callback(url); } if (this._loadedCSSFiles[url]) { return callback(url); } var cbs = this._loadingCSSFiles[url]; if (cbs) { return cbs.push(callback); } this._loadingCSSFiles[url] = [callback]; var linkEl = document.createElement('link'); linkEl.type = 'text/css'; linkEl.rel = 'stylesheet'; linkEl.href = url; var headEl = document.getElementsByTagName('head')[0]; headEl.appendChild(linkEl); var imgEl = document.createElement('img'); /* var _self = this; (function(_url) { imgEl.onerror = function() { //var url = this; var url = _url; _self._loadedCSSFiles[url] = true; var callbacks = _self._loadingCSSFiles[url]; for (var i = 0, f = callbacks.length; i < f; ++i) { callbacks[i](url); } delete _self._loadingCSSFiles[url]; }; })(url); */ imgEl.onerror = Ink.bindEvent(function(event, _url) { //var url = this; var url = _url; this._loadedCSSFiles[url] = true; var callbacks = this._loadingCSSFiles[url]; for (var i = 0, f = callbacks.length; i < f; ++i) { callbacks[i](url); } delete this._loadingCSSFiles[url]; }, this, url); imgEl.src = url; }, /** * Converts decimal to hexadecimal values * Useful to convert colors to their hexadecimal representation. * * @method decToHex * @param {String} dec Either a single decimal value, an rgb(r, g, b) string or an Object with r, g and b properties * @return {String} Hexadecimal value * @sample Ink_Dom_Css_decToHex.html */ decToHex: function(dec) { var normalizeTo2 = function(val) { if (val.length === 1) { val = '0' + val; } val = val.toUpperCase(); return val; }; if (typeof dec === 'object') { var rDec = normalizeTo2(parseInt(dec.r, 10).toString(16)); var gDec = normalizeTo2(parseInt(dec.g, 10).toString(16)); var bDec = normalizeTo2(parseInt(dec.b, 10).toString(16)); return rDec+gDec+bDec; } else { dec += ''; var rgb = dec.match(/\((\d+),\s?(\d+),\s?(\d+)\)/); if (rgb !== null) { return normalizeTo2(parseInt(rgb[1], 10).toString(16)) + normalizeTo2(parseInt(rgb[2], 10).toString(16)) + normalizeTo2(parseInt(rgb[3], 10).toString(16)); } else { return normalizeTo2(parseInt(dec, 10).toString(16)); } } }, /** * Converts hexadecimal values to decimal * Useful to use with CSS colors * * @method hexToDec * @param {String} hex hexadecimal Value with 6, 3, 2 or 1 characters * @return {Number} Object with properties r, g, b if length of number is >= 3 or decimal value instead. * @sample Ink_Dom_Css_hexToDec.html */ hexToDec: function(hex){ if (hex.indexOf('#') === 0) { hex = hex.substr(1); } if (hex.length === 6) { // will return object RGB return { r: parseInt(hex.substr(0,2), 16), g: parseInt(hex.substr(2,2), 16), b: parseInt(hex.substr(4,2), 16) }; } else if (hex.length === 3) { // will return object RGB return { r: parseInt(hex.charAt(0) + hex.charAt(0), 16), g: parseInt(hex.charAt(1) + hex.charAt(1), 16), b: parseInt(hex.charAt(2) + hex.charAt(2), 16) }; } else if (hex.length <= 2) { // will return int return parseInt(hex, 16); } }, /** * Get a single property from a stylesheet. * Use this to obtain the value of a CSS property (searched from loaded CSS documents) * * @method getPropertyFromStylesheet * @param {String} selector a CSS rule. must be an exact match * @param {String} property a CSS property * @return {String} value of the found property, or null if it wasn't matched */ getPropertyFromStylesheet: function(selector, property) { var rule = this.getRuleFromStylesheet(selector); if (rule) { return rule.style[property]; } return null; }, getPropertyFromStylesheet2: function(selector, property) { var rules = this.getRulesFromStylesheet(selector); /* rules.forEach(function(rule) { var x = rule.style[property]; if (x !== null && x !== undefined) { return x; } }); */ var x; for(var i=0, t=rules.length; i < t; i++) { x = rules[i].style[property]; if (x !== null && x !== undefined) { return x; } } return null; }, getRuleFromStylesheet: function(selector) { var sheet, rules, ri, rf, rule; var s = document.styleSheets; if (!s) { return null; } for (var si = 0, sf = document.styleSheets.length; si < sf; ++si) { sheet = document.styleSheets[si]; rules = sheet.rules ? sheet.rules : sheet.cssRules; if (!rules) { return null; } for (ri = 0, rf = rules.length; ri < rf; ++ri) { rule = rules[ri]; if (!rule.selectorText) { continue; } if (rule.selectorText === selector) { return rule; } } } return null; }, getRulesFromStylesheet: function(selector) { var res = []; var sheet, rules, ri, rf, rule; var s = document.styleSheets; if (!s) { return res; } for (var si = 0, sf = document.styleSheets.length; si < sf; ++si) { sheet = document.styleSheets[si]; rules = sheet.rules ? sheet.rules : sheet.cssRules; if (!rules) { return null; } for (ri = 0, rf = rules.length; ri < rf; ++ri) { rule = rules[ri]; if (!rule.selectorText) { continue; } if (rule.selectorText === selector) { res.push(rule); } } } return res; }, getPropertiesFromRule: function(selector) { var rule = this.getRuleFromStylesheet(selector); var props = {}; var prop, i, f; /*if (typeof rule.style.length === 'snumber') { for (i = 0, f = rule.style.length; i < f; ++i) { prop = this._camelCase( rule.style[i] ); props[prop] = rule.style[prop]; } } else { // HANDLES IE 8, FIREFOX RULE JOINING... */ rule = rule.style.cssText; var parts = rule.split(';'); var steps, val, pre, pos; for (i = 0, f = parts.length; i < f; ++i) { if (parts[i].charAt(0) === ' ') { parts[i] = parts[i].substring(1); } steps = parts[i].split(':'); prop = this._camelCase( steps[0].toLowerCase() ); val = steps[1]; if (val) { val = val.substring(1); if (prop === 'padding' || prop === 'margin' || prop === 'borderWidth') { if (prop === 'borderWidth') { pre = 'border'; pos = 'Width'; } else { pre = prop; pos = ''; } if (val.indexOf(' ') !== -1) { val = val.split(' '); props[pre + 'Top' + pos] = val[0]; props[pre + 'Bottom'+ pos] = val[0]; props[pre + 'Left' + pos] = val[1]; props[pre + 'Right' + pos] = val[1]; } else { props[pre + 'Top' + pos] = val; props[pre + 'Bottom'+ pos] = val; props[pre + 'Left' + pos] = val; props[pre + 'Right' + pos] = val; } } else if (prop === 'borderRadius') { if (val.indexOf(' ') !== -1) { val = val.split(' '); props.borderTopLeftRadius = val[0]; props.borderBottomRightRadius = val[0]; props.borderTopRightRadius = val[1]; props.borderBottomLeftRadius = val[1]; } else { props.borderTopLeftRadius = val; props.borderTopRightRadius = val; props.borderBottomLeftRadius = val; props.borderBottomRightRadius = val; } } else { props[prop] = val; } } } //} //console.log(props); return props; }, /** * Change the font size of elements. * Changes the font size of the elements which match the given CSS rule * For this function to work, the CSS file must be in the same domain than the host page, otherwise JS can't access it. * * @method changeFontSize * @param {String} selector CSS selector rule * @param {Number} delta Number of pixels to change on font-size * @param {String} [op] Supported operations are '+' and '*'. defaults to '+' * @param {Number} [minVal] If result gets smaller than minVal, change does not occurr * @param {Number} [maxVal] If result gets bigger than maxVal, change does not occurr */ changeFontSize: function(selector, delta, op, minVal, maxVal) { var that = this; Ink.requireModules(['Ink.Dom.Selector_1'], function(Selector) { var e; if (typeof selector !== 'string') { e = '1st argument must be a CSS selector rule.'; } else if (typeof delta !== 'number') { e = '2nd argument must be a number.'; } else if (op !== undefined && op !== '+' && op !== '*') { e = '3rd argument must be one of "+", "*".'; } else if (minVal !== undefined && (typeof minVal !== 'number' || minVal <= 0)) { e = '4th argument must be a positive number.'; } else if (maxVal !== undefined && (typeof maxVal !== 'number' || maxVal < maxVal)) { e = '5th argument must be a positive number greater than minValue.'; } if (e) { throw new TypeError(e); } var val, el, els = Selector.select(selector); if (minVal === undefined) { minVal = 1; } op = (op === '*') ? function(a,b){return a*b;} : function(a,b){return a+b;}; for (var i = 0, f = els.length; i < f; ++i) { el = els[i]; val = parseFloat( that.getStyle(el, 'fontSize')); val = op(val, delta); if (val < minVal) { continue; } if (typeof maxVal === 'number' && val > maxVal) { continue; } el.style.fontSize = val + 'px'; } }); } }; return Css; }); /** * DOM Traversal and manipulation * @module Ink.Dom.Element_1 * @version 1 */ Ink.createModule('Ink.Dom.Element', 1, [], function() { 'use strict'; var createContextualFragmentSupport = ( typeof document.createRange === 'function' && typeof window.Range.prototype.createContextualFragment === 'function'); var deleteThisTbodyToken = 'Ink.Dom.Element tbody: ' + Math.random(); var browserCreatesTbodies = (function () { var div = document.createElement('div'); div.innerHTML = ''; return div.getElementsByTagName('tbody').length !== 0; }()); function rect(elem){ var dimensions = {}; try { dimensions = elem.getBoundingClientRect(); } catch(e){ dimensions = { top: elem.offsetTop, left: elem.offsetLeft }; } return dimensions; } /** * @namespace Ink.Dom.Element_1 */ var InkElement = { /** * Checks if something is a DOM Element. * * @method isDOMElement * @static * @param {Mixed} o The object to be checked. * @return {Boolean} True if it's a valid DOM Element. * @example * var el = Ink.s('#element'); * if( InkElement.isDOMElement( el ) === true ){ * // It is a DOM Element. * } else { * // It is NOT a DOM Element. * } */ isDOMElement: function(o) { return o !== null && typeof o === 'object' && 'nodeType' in o && o.nodeType === 1; }, /** * Shortcut for `document.getElementById` * * @method get * @param {String|DOMElement} elm Either an ID of an element, or an element. * @return {DOMElement|null} The DOM element with the given id or null when it was not found * @sample Ink_Dom_Element_1_get.html */ get: function(elm) { if(typeof elm !== 'undefined') { if(typeof elm === 'string') { return document.getElementById(elm); } return elm; } return null; }, /** * Creates a DOM element * * @method create * @param {String} tag tag name * @param {Object} properties object with properties to be set on the element. You can also call other functions in Ink.Dom.Element like this * @sample Ink_Dom_Element_1_create.html */ create: function(tag, properties) { var el = document.createElement(tag); //Ink.extendObj(el, properties); for(var property in properties) { if(properties.hasOwnProperty(property)) { if (property in InkElement) { InkElement[property](el, properties[property]); } else { if(property === 'className' || property === 'class') { el.className = properties.className || properties['class']; } else { el.setAttribute(property, properties[property]); } } } } return el; }, /** * Removes a DOM Element * * @method remove * @param {DOMElement} elm The element to remove * @sample Ink_Dom_Element_1_remove.html */ remove: function(el) { el = Ink.i(el); var parEl; if (el && (parEl = el.parentNode)) { parEl.removeChild(el); } }, /** * Scrolls the window to an element * * @method scrollTo * @param {DOMElement|String} elm Element where to scroll * @sample Ink_Dom_Element_1_scrollTo.html */ scrollTo: function(elm) { elm = InkElement.get(elm); if(elm) { if (elm.scrollIntoView) { return elm.scrollIntoView(); } var elmOffset = {}, elmTop = 0, elmLeft = 0; do { elmTop += elm.offsetTop || 0; elmLeft += elm.offsetLeft || 0; elm = elm.offsetParent; } while(elm); elmOffset = {x: elmLeft, y: elmTop}; window.scrollTo(elmOffset.x, elmOffset.y); } }, /** * Gets the top offset of an element * * @method offsetTop * @uses Ink.Dom.Browser * * @param {DOMElement|String} elm Target element * @return {Number} Offset from the target element to the top of the document * @sample Ink_Dom_Element_1_offsetTop.html */ offsetTop: function(elm) { return InkElement.offset(elm)[1]; }, /** * Gets the left offset of an element * * @method offsetLeft * @uses Ink.Dom.Browser * * @param {DOMElement|String} elm Target element * @return {Number} Offset from the target element to the left of the document * @sample Ink_Dom_Element_1_offsetLeft.html */ offsetLeft: function(elm) { return InkElement.offset(elm)[0]; }, /** * Gets the relative offset of an element * * @method positionedOffset * @param {DOMElement|String} elm Target element * @return {Array} Array with the element offsetleft and offsettop relative to the closest positioned ancestor * @sample Ink_Dom_Element_1_positionedOffset.html */ positionedOffset: function(element) { var valueTop = 0, valueLeft = 0; element = InkElement.get(element); do { valueTop += element.offsetTop || 0; valueLeft += element.offsetLeft || 0; element = element.offsetParent; if (element) { if (element.tagName.toLowerCase() === 'body') { break; } var value = element.style.position; if (!value && element.currentStyle) { value = element.currentStyle.position; } if ((!value || value === 'auto') && typeof getComputedStyle !== 'undefined') { var css = getComputedStyle(element, null); value = css ? css.position : null; } if (value === 'relative' || value === 'absolute') { break; } } } while (element); return [valueLeft, valueTop]; }, /** * Gets the cumulative offset for an element * * Returns the top left position of the element on the page * * @method offset * @uses Ink.Dom.Browser * * @method offset * @param {DOMElement|String} elm Target element * @return {[Number, Number]} Array with pixel distance from the target element to the top left corner of the document * @sample Ink_Dom_Element_1_offset.html */ offset: function(el) { /*jshint boss:true */ el = Ink.i(el); var res = [0, 0]; var doc = el.ownerDocument, docElem = doc.documentElement, box = rect(el), body = doc.body, clientTop = docElem.clientTop || body.clientTop || 0, clientLeft = docElem.clientLeft || body.clientLeft || 0, scrollTop = doc.pageYOffset || docElem.scrollTop || body.scrollTop, scrollLeft = doc.pageXOffset || docElem.scrollLeft || body.scrollLeft, top = box.top + scrollTop - clientTop, left = box.left + scrollLeft - clientLeft; res = [left, top]; return res; }, /** * Gets the scroll of the element * * @method scroll * @param {DOMElement|String} [elm] Target element or document.body * @returns {Array} offset values for x and y scroll * @sample Ink_Dom_Element_1_scroll.html */ scroll: function(elm) { elm = elm ? Ink.i(elm) : document.body; return [ ( ( !window.pageXOffset ) ? elm.scrollLeft : window.pageXOffset ), ( ( !window.pageYOffset ) ? elm.scrollTop : window.pageYOffset ) ]; }, _getPropPx: function(cs, prop) { var n, c; var val = cs.getPropertyValue ? cs.getPropertyValue(prop) : cs[prop]; if (!val) { n = 0; } else { c = val.indexOf('px'); if (c === -1) { n = 0; } else { n = parseFloat(val, 10); } } //console.log([prop, ' "', val, '" ', n].join('')); return n; }, /** * Alias for offset() * * @method offset2 * @deprecated Kept for historic reasons. Use offset() instead. */ offset2: function(el) { return InkElement.offset(el); }, /** * Checks if an element has an attribute * * @method hasAttribute * @param {Object} elm Target element * @param {String} attr Attribute name * @return {Boolean} Boolean based on existance of attribute * @sample Ink_Dom_Element_1_hasAttribute.html */ hasAttribute: function(elm, attr){ elm = Ink.i(elm); return elm.hasAttribute ? elm.hasAttribute(attr) : !!elm.getAttribute(attr); }, /** * Inserts an element right after another * * @method insertAfter * @param {DOMElement} newElm Element to be inserted * @param {DOMElement|String} targetElm Key element * @sample Ink_Dom_Element_1_insertAfter.html */ insertAfter: function(newElm, targetElm) { /*jshint boss:true */ if (targetElm = InkElement.get(targetElm)) { if (targetElm.nextSibling !== null) { targetElm.parentNode.insertBefore(newElm, targetElm.nextSibling); } else { targetElm.parentNode.appendChild(newElm); } } }, /** * Inserts an element before another * * @method insertBefore * @param {DOMElement} newElm Element to be inserted * @param {DOMElement|String} targetElm Key element * @sample Ink_Dom_Element_1_insertBefore.html */ insertBefore: function (newElm, targetElm) { /*jshint boss:true */ if ( (targetElm = InkElement.get(targetElm)) ) { targetElm.parentNode.insertBefore(newElm, targetElm); } }, /** * Inserts an element as the first child of another * * @method insertTop * @param {DOMElement} newElm Element to be inserted * @param {DOMElement|String} targetElm Key element * @sample Ink_Dom_Element_1_insertTop.html */ insertTop: function(newElm,targetElm) { /*jshint boss:true */ if (targetElm = InkElement.get(targetElm)) { if (targetElm.firstChild) { targetElm.insertBefore(newElm, targetElm.firstChild); } else { targetElm.appendChild(newElm); } } }, /** * Inserts an element as the last child of another * * @method insertBottom * @param {DOMElement} newElm Element to be inserted * @param {DOMElement|String} targetElm Key element * @sample Ink_Dom_Element_1_insertBottom.html */ insertBottom: function(newElm, targetElm) { /*jshint boss:true */ targetElm = Ink.i(targetElm); targetElm.appendChild(newElm); }, /** * Retrieves textContent from node * * @method textContent * @param {DOMNode} node Where to retreive text from. Can be any node type. * @return {String} the text * @sample Ink_Dom_Element_1_textContent.html */ textContent: function(node){ node = Ink.i(node); var text, k, cs, m; switch(node && node.nodeType) { case 9: /*DOCUMENT_NODE*/ // IE quirks mode does not have documentElement return InkElement.textContent(node.documentElement || node.body && node.body.parentNode || node.body); case 1: /*ELEMENT_NODE*/ text = node.innerText; if (typeof text !== 'undefined') { return text; } /* falls through */ case 11: /*DOCUMENT_FRAGMENT_NODE*/ text = node.textContent; if (typeof text !== 'undefined') { return text; } if (node.firstChild === node.lastChild) { // Common case: 0 or 1 children return InkElement.textContent(node.firstChild); } text = []; cs = node.childNodes; for (k = 0, m = cs.length; k < m; ++k) { text.push( InkElement.textContent( cs[k] ) ); } return text.join(''); case 3: /*TEXT_NODE*/ case 4: /*CDATA_SECTION_NODE*/ return node.nodeValue; } return ''; }, /** * Replaces text content of a DOM Node * This method removes any child node previously present * * @method setTextContent * @param {DOMNode} node node Target node where the text will be added. * @param {String} text text Text to be added on the node. * @sample Ink_Dom_Element_1_setTextContent.html */ setTextContent: function(node, text){ node = Ink.i(node); switch(node && node.nodeType) { case 1: /*ELEMENT_NODE*/ if ('innerText' in node) { node.innerText = text; break; } /* falls through */ case 11: /*DOCUMENT_FRAGMENT_NODE*/ if ('textContent' in node) { node.textContent = text; break; } /* falls through */ case 9: /*DOCUMENT_NODE*/ while(node.firstChild) { node.removeChild(node.firstChild); } if (text !== '') { var doc = node.ownerDocument || node; node.appendChild(doc.createTextNode(text)); } break; case 3: /*TEXT_NODE*/ case 4: /*CDATA_SECTION_NODE*/ node.nodeValue = text; break; } }, /** * Checks if an element is a link * * @method isLink * @param {DOMNode} node Node to check if it's link * @return {Boolean} * @sample Ink_Dom_Element_1_isLink.html */ isLink: function(element){ var b = element && element.nodeType === 1 && ((/^a|area$/i).test(element.tagName) || element.hasAttributeNS && element.hasAttributeNS('http://www.w3.org/1999/xlink','href')); return !!b; }, /** * Checks if a node is an ancestor of another * * @method isAncestorOf * @param {DOMNode} ancestor Ancestor node * @param {DOMNode} node Descendant node * @return {Boolean} * @sample Ink_Dom_Element_1_isAncestorOf.html */ isAncestorOf: function(ancestor, node){ /*jshint boss:true */ if (!node || !ancestor) { return false; } if (node.compareDocumentPosition) { return (ancestor.compareDocumentPosition(node) & 0x10) !== 0;/*Node.DOCUMENT_POSITION_CONTAINED_BY*/ } while (node = node.parentNode){ if (node === ancestor){ return true; } } return false; }, /** * Checks if a node is descendant of another * * @method descendantOf * @param {DOMNode} node The ancestor * @param {DOMNode} descendant The descendant * @return {Boolean} true if 'descendant' is descendant of 'node' * @sample Ink_Dom_Element_1_descendantOf.html */ descendantOf: function(node, descendant){ return node !== descendant && InkElement.isAncestorOf(node, descendant); }, /** * Get first child element of another * @method firstElementChild * @param {DOMElement} elm Parent node * @return {DOMElement} the Element child * @sample Ink_Dom_Element_1_firstElementChild.html */ firstElementChild: function(elm){ if(!elm) { return null; } if ('firstElementChild' in elm) { return elm.firstElementChild; } var child = elm.firstChild; while(child && child.nodeType !== 1) { child = child.nextSibling; } return child; }, /** * Get the last child element of another * @method lastElementChild * @param {DOMElement} elm Parent node * @return {DOMElement} the Element child * @sample Ink_Dom_Element_1_lastElementChild.html */ lastElementChild: function(elm){ if(!elm) { return null; } if ('lastElementChild' in elm) { return elm.lastElementChild; } var child = elm.lastChild; while(child && child.nodeType !== 1) { child = child.previousSibling; } return child; }, /** * Get the first sibling element after the node * * @method nextElementSibling * @param {DOMNode} node The current node * @return {DOMElement|Null} The first sibling element after node or null if none is found * @sample Ink_Dom_Element_1_nextElementSibling.html */ nextElementSibling: function(node){ var sibling = null; if(!node){ return sibling; } if("nextElementSibling" in node){ return node.nextElementSibling; } else { sibling = node.nextSibling; // 1 === Node.ELEMENT_NODE while(sibling && sibling.nodeType !== 1){ sibling = sibling.nextSibling; } return sibling; } }, /** * Get the first sibling element before the node * * @method previousElementSibling * @param {DOMNode} node The current node * @return {DOMElement|Null} The first element sibling before node or null if none is found * @sample Ink_Dom_Element_1_previousElementSibling.html */ previousElementSibling: function(node){ var sibling = null; if(!node){ return sibling; } if("previousElementSibling" in node){ return node.previousElementSibling; } else { sibling = node.previousSibling; // 1 === Node.ELEMENT_NODE while(sibling && sibling.nodeType !== 1){ sibling = sibling.previousSibling; } return sibling; } }, /** * Get an element's width in pixels. * * @method elementWidth * @param {DOMElement|String} element Target DOM element or target ID * @return {Number} The element's width * @sample Ink_Dom_Element_1_elementWidth.html */ elementWidth: function(element) { if(typeof element === "string") { element = document.getElementById(element); } return element.offsetWidth; }, /** * Get an element's height in pixels. * * @method elementHeight * @param {DOMElement|String} element DOM element or target ID * @return {Number} The element's height * @sample Ink_Dom_Element_1_elementHeight.html */ elementHeight: function(element) { if(typeof element === "string") { element = document.getElementById(element); } return element.offsetHeight; }, /** * Deprecated. Alias for offsetLeft() * * @method elementLeft * @param {DOMElement|String} element DOM element or target ID * @return {Number} Element's left position */ elementLeft: function(element) { return InkElement.offsetLeft(element); }, /** * Deprecated. Alias for offsetTop() * * @method elementTop * @param {DOMElement|string} element Target DOM element or target ID * @return {Number} element's top position */ elementTop: function(element) { return InkElement.offsetTop(element); }, /** * Get an element's dimensions in pixels. * * @method elementDimensions * @param {DOMElement|string} element DOM element or target ID * @return {Array} Array with element's width and height * @sample Ink_Dom_Element_1_elementDimensions.html */ elementDimensions: function(element) { element = Ink.i(element); return [element.offsetWidth, element.offsetHeight]; }, /** * Get the outer dimensions of an element in pixels. * * @method outerDimensions * @uses Ink.Dom.Css * * @param {DOMElement} element Target element * @return {Array} Array with element width and height. * @sample Ink_Dom_Element_1_outerDimensions.html */ outerDimensions: function (element) { var bbox = rect(element); var Css = Ink.getModule('Ink.Dom.Css_1'); var getStyle = Ink.bindMethod(Css, 'getStyle', element); return [ bbox.right - bbox.left + parseFloat(getStyle('marginLeft') || 0) + parseFloat(getStyle('marginRight') || 0), // w bbox.bottom - bbox.top + parseFloat(getStyle('marginTop') || 0) + parseFloat(getStyle('marginBottom') || 0) // h ]; }, /** * Check if an element is inside the viewport * * @method inViewport * @param {DOMElement} element DOM Element * @param {Object} [options] Options object. If you pass a Boolean value here, it is interpreted as `options.partial` * @param {Boolean} [options.partial]=false Return `true` even if it is only partially visible. * @param {Number} [options.margin]=0 Consider a margin all around the viewport with `opts.margin` width a dead zone. * @return {Boolean} * @sample Ink_Dom_Element_1_inViewport.html */ inViewport: function (element, opts) { var dims = rect(Ink.i(element)); if (typeof opts === 'boolean') { opts = {partial: opts, margin: 0}; } opts = Ink.extendObj({ partial: false, margin: 0}, opts || {}); if (opts.partial) { return dims.bottom + opts.margin > 0 && // from the top dims.left - opts.margin < InkElement.viewportWidth() && // from the right dims.top - opts.margin < InkElement.viewportHeight() && // from the bottom dims.right + opts.margin > 0; // from the left } else { return dims.top + opts.margin > 0 && // from the top dims.right - opts.margin < InkElement.viewportWidth() && // from the right dims.bottom - opts.margin < InkElement.viewportHeight() && // from the bottom dims.left + opts.margin > 0; // from the left } }, /** * Check if an element is hidden. * Taken from Mootools Element extras ( https://gist.github.com/cheeaun/73342 ) * Does not take into account visibility:hidden * @method isHidden * @param {DOMElement} element Element to check * @return {Boolean} * @sample Ink_Dom_Element_1_isHidden.html */ isHidden: function (element) { var w = element.offsetWidth, h = element.offsetHeight, force = (element.tagName.toLowerCase() === 'tr'); var Css = Ink.getModule('Ink.Dom.Css_1'); return (w===0 && h===0 && !force) ? true : (w!==0 && h!==0 && !force) ? false : Css.getStyle(element, 'display').toLowerCase() === 'none'; }, /** * Check if an element is visible * * @method isVisible * @uses isHidden * @param {DOMElement} element Element to check * @return {Boolean} * @sample Ink_Dom_Element_1_isVisible.html */ isVisible: function (element) { return !this.isHidden(element); }, /** * Clones an element's position to another * * @method clonePosition * @param {DOMElement} cloneTo element to be position cloned * @param {DOMElement} cloneFrom element to get the cloned position * @return {DOMElement} The element with positionClone * @sample Ink_Dom_Element_1_clonePosition.html */ clonePosition: function(cloneTo, cloneFrom){ var pos = InkElement.offset(cloneFrom); cloneTo.style.left = pos[0]+'px'; cloneTo.style.top = pos[1]+'px'; return cloneTo; }, /** * Text-overflow: ellipsis emulation * Slices off a piece of text at the end of the element and adds the ellipsis so all text fits inside. * * @method ellipsizeText * @param {DOMElement} element Element to modify text content * @param {String} [ellipsis]='\u2026' String to append to the chopped text */ ellipsizeText: function(element/*, ellipsis*/){ if ((element = Ink.i(element))) { element.style.overflow = 'hidden'; element.style.whiteSpace = 'nowrap'; element.style.textOverflow = 'ellipsis'; } }, /** * Finds the closest ancestor element matching your test function * * * @method findUpwardsHaving * @param {DOMElement} element Element to base the search from * @param {Function} boolTest Testing function * @return {DOMElement|false} The matched element or false if did not match * @sample Ink_Dom_Element_1_findUpwardsHaving.html */ findUpwardsHaving: function(element, boolTest) { while (element && element.nodeType === 1) { if (boolTest(element)) { return element; } element = element.parentNode; } return false; }, /** * Finds the closest ancestor by class name * * @method findUpwardsByClass * @uses findUpwardsHaving * @param {DOMElement} element Element to base the search from * @param {String} className Class name to search * @returns {DOMElement|false} The matched element or false if did not match * @sample Ink_Dom_Element_1_findUpwardsByClass.html */ findUpwardsByClass: function(element, className) { var re = new RegExp("(^|\\s)" + className + "(\\s|$)"); var tst = function(el) { var cls = el.className; return cls && re.test(cls); }; return InkElement.findUpwardsHaving(element, tst); }, /** * Finds the closest ancestor by tag name * * @method findUpwardsByTag * @param {DOMElement} element Element to base the search from * @param {String} tag Tag to search * @returns {DOMElement|false} the matched element or false if did not match * @sample Ink_Dom_Element_1_findUpwardsByTag.html */ findUpwardsByTag: function(element, tag) { tag = tag.toUpperCase(); var tst = function(el) { return el.nodeName && el.nodeName.toUpperCase() === tag; }; return InkElement.findUpwardsHaving(element, tst); }, /** * Finds the closest ancestor by id * * @method findUpwardsById * @param {HtmlElement} element Element to base the search from * @param {String} id ID to search * @returns {HtmlElement|false} The matched element or false if did not match * @sample Ink_Dom_Element_1_findUpwardsById.html */ findUpwardsById: function(element, id) { var tst = function(el) { return el.id === id; }; return InkElement.findUpwardsHaving(element, tst); }, /** * Finds the closest ancestor by CSS selector * * @method findUpwardsBySelector * @param {HtmlElement} element Element to base the search from * @param {String} sel CSS selector * @returns {HtmlElement|false} The matched element or false if did not match * @sample Ink_Dom_Element_1_findUpwardsBySelector.html */ findUpwardsBySelector: function(element, sel) { var Selector = Ink.getModule('Ink.Dom.Selector', '1'); if (!Selector) { throw new Error('This method requires Ink.Dom.Selector'); } var tst = function(el) { return Selector.matchesSelector(el, sel); }; return InkElement.findUpwardsHaving(element, tst); }, /** * Gets the trimmed text of an element * * @method getChildrenText * @param {DOMElement} el Element to base the search from * @param {Boolean} [removeIt] Flag to remove the text from the element * @return {String} Text found * @sample Ink_Dom_Element_1_getChildrenText.html */ getChildrenText: function(el, removeIt) { var node, j, part, nodes = el.childNodes, jLen = nodes.length, text = ''; if (!el) { return text; } for (j = 0; j < jLen; ++j) { node = nodes[j]; if (!node) { continue; } if (node.nodeType === 3) { // TEXT NODE part = InkElement._trimString( String(node.data) ); if (part.length > 0) { text += part; if (removeIt) { el.removeChild(node); } } else { el.removeChild(node); } } } return text; }, /** * String trim implementation * Used by getChildrenText * * function _trimString * param {String} text * return {String} trimmed text */ _trimString: function(text) { return (String.prototype.trim) ? text.trim() : text.replace(/^\s*/, '').replace(/\s*$/, ''); }, /** * Gets value of a select element * * @method getSelectValues * @param {DOMElement|String} select element * @return {Array} The selected values * @sample Ink_Dom_Element_1_getSelectValues.html */ getSelectValues: function (select) { var selectEl = Ink.i(select); var values = []; for (var i = 0; i < selectEl.options.length; ++i) { values.push( selectEl.options[i].value ); } return values; }, /* used by fills */ _normalizeData: function(data) { var d, data2 = []; for (var i = 0, f = data.length; i < f; ++i) { d = data[i]; if (!(d instanceof Array)) { // if not array, wraps primitive twice: val -> [val, val] d = [d, d]; } else if (d.length === 1) { // if 1 element array: [val] -> [val, val] d.push(d[0]); } data2.push(d); } return data2; }, /** * Fills a select element with options * * @method fillSelect * @param {DOMElement|String} container Select element which will get filled * @param {Array} data Data to populate the component * @param {Boolean} [skipEmpty] Flag to skip empty option * @param {String|Number} [defaultValue] Initial selected value * * @sample Ink_Dom_Element_1_fillSelect.html */ fillSelect: function(container, data, skipEmpty, defaultValue) { var containerEl = Ink.i(container); if (!containerEl) { return; } containerEl.innerHTML = ''; var d, optionEl; if (!skipEmpty) { // add initial empty option optionEl = document.createElement('option'); optionEl.setAttribute('value', ''); containerEl.appendChild(optionEl); } data = InkElement._normalizeData(data); for (var i = 0, f = data.length; i < f; ++i) { d = data[i]; optionEl = document.createElement('option'); optionEl.setAttribute('value', d[0]); if (d.length > 2) { optionEl.setAttribute('extra', d[2]); } optionEl.appendChild( document.createTextNode(d[1]) ); if (d[0] === defaultValue) { optionEl.setAttribute('selected', 'selected'); } containerEl.appendChild(optionEl); } }, /** * Creates a set of radio buttons from an array of data * * @method fillRadios * @param {DOMElement|String} insertAfterEl Element after which the input elements will be created * @param {String} name Name for the form field ([] is added if not present as a suffix) * @param {Array} data Data to populate the component * @param {Boolean} [skipEmpty] Flag to skip creation of empty options * @param {String|Number} [defaultValue] Initial selected value * @param {String} [splitEl] Name of element to add after each input element (example: 'br') * @return {DOMElement} Wrapper element around the radio buttons */ fillRadios: function(insertAfterEl, name, data, skipEmpty, defaultValue, splitEl) { insertAfterEl = Ink.i(insertAfterEl); var containerEl = document.createElement('span'); InkElement.insertAfter(containerEl, insertAfterEl); data = InkElement._normalizeData(data); /* if (name.substring(name.length - 1) !== ']') { name += '[]'; } */ var d, inputEl; if (!skipEmpty) { // add initial empty option inputEl = document.createElement('input'); inputEl.setAttribute('type', 'radio'); inputEl.setAttribute('name', name); inputEl.setAttribute('value', ''); containerEl.appendChild(inputEl); if (splitEl) { containerEl.appendChild( document.createElement(splitEl) ); } } for (var i = 0; i < data.length; ++i) { d = data[i]; inputEl = document.createElement('input'); inputEl.setAttribute('type', 'radio'); inputEl.setAttribute('name', name); inputEl.setAttribute('value', d[0]); containerEl.appendChild(inputEl); containerEl.appendChild( document.createTextNode(d[1]) ); if (splitEl) { containerEl.appendChild( document.createElement(splitEl) ); } if (d[0] === defaultValue) { inputEl.checked = true; } } return containerEl; }, /** * Creates set of checkbox buttons * * @method fillChecks * @param {DOMElement|String} insertAfterEl Element after which the input elements will be created * @param {String} name Name for the form field ([] is added if not present as a suffix) * @param {Array} data Data to populate the component * @param {Boolean} [skipEmpty] Flag to skip creation of empty options * @param {String|Number} [defaultValue] Initial selected value * @param {String} [splitEl] Name of element to add after each input element (example: 'br') * @return {DOMElement} Wrapper element around the checkboxes */ fillChecks: function(insertAfterEl, name, data, defaultValue, splitEl) { insertAfterEl = Ink.i(insertAfterEl); var containerEl = document.createElement('span'); InkElement.insertAfter(containerEl, insertAfterEl); data = InkElement._normalizeData(data); if (name.substring(name.length - 1) !== ']') { name += '[]'; } var d, inputEl; for (var i = 0; i < data.length; ++i) { d = data[i]; inputEl = document.createElement('input'); inputEl.setAttribute('type', 'checkbox'); inputEl.setAttribute('name', name); inputEl.setAttribute('value', d[0]); containerEl.appendChild(inputEl); containerEl.appendChild( document.createTextNode(d[1]) ); if (splitEl) { containerEl.appendChild( document.createElement(splitEl) ); } if (d[0] === defaultValue) { inputEl.checked = true; } } return containerEl; }, /** * Gets the index of an element relative to a parent * * @method parentIndexOf * @param {DOMElement} parentEl Element to parse * @param {DOMElement} childEl Child Element to look for * @return {Number} The index of the childEl inside parentEl. Returns -1 if it's not a direct child * @sample Ink_Dom_Element_1_parentIndexOf.html */ parentIndexOf: function(parentEl, childEl) { var node, idx = 0; for (var i = 0, f = parentEl.childNodes.length; i < f; ++i) { node = parentEl.childNodes[i]; if (node.nodeType === 1) { // ELEMENT if (node === childEl) { return idx; } ++idx; } } return -1; }, /** * Gets the next siblings of an element * * @method nextSiblings * @param {String|DOMElement} elm Element * @return {Array} Array of next sibling elements * @sample Ink_Dom_Element_1_nextSiblings.html */ nextSiblings: function(elm) { elm = Ink.i(elm); if(typeof(elm) === 'object' && elm !== null && elm.nodeType && elm.nodeType === 1) { var elements = [], siblings = elm.parentNode.children, index = InkElement.parentIndexOf(elm.parentNode, elm); for(var i = ++index, len = siblings.length; i
" + deleteThisTbodyToken + "
"; } else { div.innerHTML = "" + html + "
"; } return div.firstChild; }, TBODY: function (div, html) { div.innerHTML = '' + html + '
'; return div.firstChild.getElementsByTagName('tbody')[0]; }, THEAD: function (div, html) { div.innerHTML = '' + html + '
'; return div.firstChild.getElementsByTagName('thead')[0]; }, TFOOT: function (div, html) { div.innerHTML = '' + html + '
'; return div.firstChild.getElementsByTagName('tfoot')[0]; }, TR: function (div, html) { div.innerHTML = '' + html + '
'; return div.firstChild.firstChild.firstChild; } }, /** * Gets a wrapper DIV with a certain HTML content to be inserted inside another element. * This is necessary for appendHTML,prependHTML functions, because they need a container element to copy the children from. * * Works around IE table quirks * @method _getWrapper * @private * @param elm * @param html */ _getWrapper: function (elm, html) { var nodeName = elm.nodeName && elm.nodeName.toUpperCase(); var wrapper = document.createElement('div'); var wrapFunc = InkElement._wrapElements[nodeName]; if ( !wrapFunc ) { wrapper.innerHTML = html; return wrapper; } // special cases wrapper = wrapFunc(wrapper, html); // worst case: tbody auto-creation even when our HTML has a tbody. if (browserCreatesTbodies && nodeName === 'TABLE') { // terrible case. Deal with tbody creation too. var tds = wrapper.getElementsByTagName('td'); for (var i = 0, len = tds.length; i < len; i++) { if (tds[i].innerHTML === deleteThisTbodyToken) { var tbody = tds[i].parentNode.parentNode; tbody.parentNode.removeChild(tbody); } } } return wrapper; }, /** * Appends HTML to an element. * This method parses the html string and doesn't modify its contents * * @method appendHTML * @param {String|DOMElement} elm Element * @param {String} html Markup string * @sample Ink_Dom_Element_1_appendHTML.html */ appendHTML: function(elm, html){ elm = Ink.i(elm); if(elm !== null) { var wrapper = InkElement._getWrapper(elm, html); while (wrapper.firstChild) { elm.appendChild(wrapper.firstChild); } } }, /** * Prepends HTML to an element. * This method parses the html string and doesn't modify its contents * * @method prependHTML * @param {String|DOMElement} elm Element * @param {String} html Markup string * @sample Ink_Dom_Element_1_prependHTML.html */ prependHTML: function(elm, html){ elm = Ink.i(elm); if(elm !== null) { var wrapper = InkElement._getWrapper(elm, html); while (wrapper.lastChild) { elm.insertBefore(wrapper.lastChild, elm.firstChild); } } }, /** * Sets the inner HTML of an element. * * @method setHTML * @param {String|DOMElement} elm Element * @param {String} html Markup string * @sample Ink_Dom_Element_1_setHTML.html */ setHTML: function (elm, html) { elm = Ink.i(elm); if(elm !== null) { try { elm.innerHTML = html; } catch (e) { // Tables in IE7 while (elm.firstChild) { elm.removeChild(elm.firstChild); } InkElement.appendHTML(elm, html); } } }, /** * Wraps an element inside a container. * * The container may or may not be in the document yet. * * @method wrap * @param {String|DOMElement} target Element to be wrapped * @param {String|DOMElement} container Element to wrap the target * @return Container element * @sample Ink_Dom_Element_1_wrap.html * * @example * before: * *
* * call this function to wrap #target with a wrapper div. * * InkElement.wrap('target', InkElement.create('div', {id: 'container'}); * * after: * *
*/ wrap: function (target, container) { target = Ink.i(target); container = Ink.i(container); var nextNode = target.nextSibling; var parent = target.parentNode; container.appendChild(target); if (nextNode !== null) { parent.insertBefore(container, nextNode); } else { parent.appendChild(container); } return container; }, /** * Places an element outside a wrapper. * * @method unwrap * @param {DOMElement} elem The element you're trying to unwrap. This should be an ancestor of the wrapper. * @param {String} [wrapperSelector] CSS Selector for the ancestor. Use this if your wrapper is not the direct parent of elem. * @sample Ink_Dom_Element_1_unwrap.html * * @example * * When you have this: * *
*
*
* * If you do this: * * InkElement.unwrap('unwrapMe'); * * You get this: * *
*
* **/ unwrap: function (elem, wrapperSelector) { elem = Ink.i(elem); var wrapper; if (typeof wrapperSelector === 'string') { wrapper = InkElement.findUpwardsBySelector(elem, wrapperSelector); } else if (typeof wrapperSelector === 'object' && wrapperSelector.tagName) { wrapper = InkElement.findUpwardsHaving(elem, function (ancestor) { return ancestor === wrapperSelector; }); } else { wrapper = elem.parentNode; } if (!wrapper || !wrapper.parentNode) { return; } InkElement.insertBefore(elem, wrapper); }, /** * Replaces an element with another. * * @method replace * @param element The element to be replaced. * @param replacement The new element. * @sample Ink_Dom_Element_1_replace.html * * @example * var newelement1 = InkElement.create('div'); * // ... * replace(Ink.i('element1'), newelement1); */ replace: function (element, replacement) { element = Ink.i(element); if(element !== null) { element.parentNode.replaceChild(replacement, element); } }, /** * Removes direct text children. * Useful to remove nasty layout gaps generated by whitespace on the markup. * * @method removeTextNodeChildren * @param {DOMElement} el Element to remove text from * @sample Ink_Dom_Element_1_removeTextNodeChildren.html */ removeTextNodeChildren: function(el) { el = Ink.i(el); if(el !== null) { var prevEl, toRemove, parent = el; el = el.firstChild; while (el) { toRemove = (el.nodeType === 3); prevEl = el; el = el.nextSibling; if (toRemove) { parent.removeChild(prevEl); } } } }, /** * Creates a documentFragment from an HTML string. * * @method htmlToFragment * @param {String} html HTML string * @return {DocumentFragment} DocumentFragment containing all of the elements from the html string * @sample Ink_Dom_Element_1_htmlToFragment.html */ htmlToFragment: (createContextualFragmentSupport ? function(html){ var range; if(typeof html !== 'string'){ return document.createDocumentFragment(); } range = document.createRange(); // set the context to document.body (firefox does this already, webkit doesn't) range.selectNode(document.body); return range.createContextualFragment(html); } : function (html) { var fragment = document.createDocumentFragment(), tempElement, current; if(typeof html !== 'string'){ return fragment; } tempElement = document.createElement('div'); tempElement.innerHTML = html; // append child removes elements from the original parent while( (current = tempElement.firstChild) ){ // intentional assignment fragment.appendChild(current); } return fragment; }), _camelCase: function(str) { return str ? str.replace(/-(\w)/g, function (_, $1){ return $1.toUpperCase(); }) : str; }, /** * Gets data attributes from an element * * @method data * @param {String|DOMElement} selector Element or CSS selector * @return {Object} Object with the data-* properties. If no data-attributes are present, an empty object is returned. * @sample Ink_Dom_Element_1_data.html */ data: function(selector) { var el; if (typeof selector !== 'object' && typeof selector !== 'string') { throw '[Ink.Dom.Element.data] :: Invalid selector defined'; } if (typeof selector === 'object') { el = selector; } else { var InkDomSelector = Ink.getModule('Ink.Dom.Selector', 1); if (!InkDomSelector) { throw "[Ink.Dom.Element.data] :: this method requires Ink.Dom.Selector - v1"; } el = InkDomSelector.select(selector); if (el.length <= 0) { throw "[Ink.Dom.Element.data] :: Can't find any element with the specified selector"; } el = el[0]; } var dataset = {}; var attrs = el.attributes || []; var curAttr, curAttrName, curAttrValue; if (attrs) { for (var i = 0, total = attrs.length; i < total; ++i) { curAttr = attrs[i]; curAttrName = curAttr.name; curAttrValue = curAttr.value; if (curAttrName && curAttrName.indexOf('data-') === 0) { dataset[InkElement._camelCase(curAttrName.replace('data-', ''))] = curAttrValue; } } } return dataset; }, /** * Move the cursor on an input or textarea element. * @method moveCursorTo * @param {DOMElement} el Input or Textarea element * @param {Number} t Index of the character to move the cursor to * @sample Ink_Dom_Element_1_moveCursorTo.html */ moveCursorTo: function(el, t) { el = Ink.i(el); if(el !== null) { if (el.setSelectionRange) { el.setSelectionRange(t, t); //el.focus(); } else { var range = el.createTextRange(); range.collapse(true); range.moveEnd( 'character', t); range.moveStart('character', t); range.select(); } } }, /** * Get the page's width. * @method pageWidth * @return {Number} Page width in pixels * @sample Ink_Dom_Element_1_pageWidth.html */ pageWidth: function() { var xScroll; if (window.innerWidth && window.scrollMaxX) { xScroll = window.innerWidth + window.scrollMaxX; } else if (document.body.scrollWidth > document.body.offsetWidth){ xScroll = document.body.scrollWidth; } else { xScroll = document.body.offsetWidth; } var windowWidth; if (window.self.innerWidth) { if(document.documentElement.clientWidth){ windowWidth = document.documentElement.clientWidth; } else { windowWidth = window.self.innerWidth; } } else if (document.documentElement && document.documentElement.clientWidth) { windowWidth = document.documentElement.clientWidth; } else if (document.body) { windowWidth = document.body.clientWidth; } if(xScroll < windowWidth){ return xScroll; } else { return windowWidth; } }, /** * Get the page's height. * @method pageHeight * @return {Number} Page height in pixels * @sample Ink_Dom_Element_1_pageHeight.html */ pageHeight: function() { var yScroll; if (window.innerHeight && window.scrollMaxY) { yScroll = window.innerHeight + window.scrollMaxY; } else if (document.body.scrollHeight > document.body.offsetHeight){ yScroll = document.body.scrollHeight; } else { yScroll = document.body.offsetHeight; } var windowHeight; if (window.self.innerHeight) { windowHeight = window.self.innerHeight; } else if (document.documentElement && document.documentElement.clientHeight) { windowHeight = document.documentElement.clientHeight; } else if (document.body) { windowHeight = document.body.clientHeight; } if(yScroll < windowHeight){ return windowHeight; } else { return yScroll; } }, /** * Get the viewport's width. * @method viewportWidth * @return {Number} Viewport width in pixels * @sample Ink_Dom_Element_1_viewportWidth.html */ viewportWidth: function() { if(typeof window.innerWidth !== "undefined") { return window.innerWidth; } if (document.documentElement && typeof document.documentElement.offsetWidth !== "undefined") { return document.documentElement.offsetWidth; } }, /** * Get the viewport's height. * @method viewportHeight * @return {Number} Viewport height in pixels * @sample Ink_Dom_Element_1_viewportHeight.html */ viewportHeight: function() { if (typeof window.innerHeight !== "undefined") { return window.innerHeight; } if (document.documentElement && typeof document.documentElement.offsetHeight !== "undefined") { return document.documentElement.offsetHeight; } }, /** * Get the scroll's width. * @method scrollWidth * @return {Number} Scroll width */ scrollWidth: function() { if (typeof window.self.pageXOffset !== 'undefined') { return window.self.pageXOffset; } if (typeof document.documentElement !== 'undefined' && typeof document.documentElement.scrollLeft !== 'undefined') { return document.documentElement.scrollLeft; } return document.body.scrollLeft; }, /** * Get the scroll's height. * @method scrollHeight * @return {Number} Scroll height */ scrollHeight: function() { if (typeof window.self.pageYOffset !== 'undefined') { return window.self.pageYOffset; } if (typeof document.documentElement !== 'undefined' && typeof document.documentElement.scrollTop !== 'undefined') { return document.documentElement.scrollTop; } return document.body.scrollTop; } }; return InkElement; }); /** * Event management * @module Ink.Dom.Event_1 * @version 1 */ Ink.createModule('Ink.Dom.Event', 1, [], function() { /* jshint asi:true, strict:false, laxcomma:true, eqeqeq:false, laxbreak:true, boss:true, curly:false, expr:true */ /** * @namespace Ink.Dom.Event_1 * @static */ /*! * Bean - copyright (c) Jacob Thornton 2011-2012 * https://github.com/fat/bean * MIT license */ var bean = (function (name, context, definition) { return definition() })('bean', this, function (name, context) { name = name || 'bean' context = context || this var win = window , old = context[name] , namespaceRegex = /[^\.]*(?=\..*)\.|.*/ , nameRegex = /\..*/ , addEvent = 'addEventListener' , removeEvent = 'removeEventListener' , doc = document || {} , root = doc.documentElement || {} , W3C_MODEL = root[addEvent] , eventSupport = W3C_MODEL ? addEvent : 'attachEvent' , ONE = {} // singleton for quick matching making add() do one() , slice = Array.prototype.slice , str2arr = function (s, d) { return s.split(d || ' ') } , isString = function (o) { return typeof o == 'string' } , isFunction = function (o) { return typeof o == 'function' } // events that we consider to be 'native', anything not in this list will // be treated as a custom event , standardNativeEvents = 'click dblclick mouseup mousedown contextmenu ' + // mouse buttons 'mousewheel mousemultiwheel DOMMouseScroll ' + // mouse wheel 'mouseover mouseout mousemove selectstart selectend ' + // mouse movement 'keydown keypress keyup ' + // keyboard 'orientationchange ' + // mobile 'focus blur change reset select submit ' + // form elements 'load unload beforeunload resize move DOMContentLoaded ' + // window 'readystatechange message ' + // window 'error abort scroll ' // misc // element.fireEvent('onXYZ'... is not forgiving if we try to fire an event // that doesn't actually exist, so make sure we only do these on newer browsers , w3cNativeEvents = 'show ' + // mouse buttons 'input invalid ' + // form elements 'touchstart touchmove touchend touchcancel ' + // touch 'gesturestart gesturechange gestureend ' + // gesture 'textinput' + // TextEvent 'readystatechange pageshow pagehide popstate ' + // window 'hashchange offline online ' + // window 'afterprint beforeprint ' + // printing 'dragstart dragenter dragover dragleave drag drop dragend ' + // dnd 'loadstart progress suspend emptied stalled loadmetadata ' + // media 'loadeddata canplay canplaythrough playing waiting seeking ' + // media 'seeked ended durationchange timeupdate play pause ratechange ' + // media 'volumechange cuechange ' + // media 'checking noupdate downloading cached updateready obsolete ' // appcache // convert to a hash for quick lookups , nativeEvents = (function (hash, events, i) { for (i = 0; i < events.length; i++) events[i] && (hash[events[i]] = 1) return hash }({}, str2arr(standardNativeEvents + (W3C_MODEL ? w3cNativeEvents : '')))) // custom events are events that we *fake*, they are not provided natively but // we can use native events to generate them , customEvents = (function () { var isAncestor = 'compareDocumentPosition' in root ? function (element, container) { return container.compareDocumentPosition && (container.compareDocumentPosition(element) & 16) === 16 } : 'contains' in root ? function (element, container) { container = container.nodeType === 9 || container === window ? root : container return container !== element && container.contains(element) } : function (element, container) { while (element = element.parentNode) if (element === container) return 1 return 0 } , check = function (event) { var related = event.relatedTarget return !related ? related == null : (related !== this && related.prefix !== 'xul' && !/document/.test(this.toString()) && !isAncestor(related, this)) } return { mouseenter: { base: 'mouseover', condition: check } , mouseleave: { base: 'mouseout', condition: check } , mousewheel: { base: /Firefox/.test(navigator.userAgent) ? 'DOMMouseScroll' : 'mousewheel' } } }()) // we provide a consistent Event object across browsers by taking the actual DOM // event object and generating a new one from its properties. , Event = (function () { // a whitelist of properties (for different event types) tells us what to check for and copy var commonProps = str2arr('altKey attrChange attrName bubbles cancelable ctrlKey currentTarget ' + 'detail eventPhase getModifierState isTrusted metaKey relatedNode relatedTarget shiftKey ' + 'srcElement target timeStamp type view which propertyName') , mouseProps = commonProps.concat(str2arr('button buttons clientX clientY dataTransfer ' + 'fromElement offsetX offsetY pageX pageY screenX screenY toElement')) , mouseWheelProps = mouseProps.concat(str2arr('wheelDelta wheelDeltaX wheelDeltaY wheelDeltaZ ' + 'axis')) // 'axis' is FF specific , keyProps = commonProps.concat(str2arr('char charCode key keyCode keyIdentifier ' + 'keyLocation location')) , textProps = commonProps.concat(str2arr('data')) , touchProps = commonProps.concat(str2arr('touches targetTouches changedTouches scale rotation')) , messageProps = commonProps.concat(str2arr('data origin source')) , stateProps = commonProps.concat(str2arr('state')) , overOutRegex = /over|out/ // some event types need special handling and some need special properties, do that all here , typeFixers = [ { // key events reg: /key/i , fix: function (event, newEvent) { newEvent.keyCode = event.keyCode || event.which return keyProps } } , { // mouse events reg: /click|mouse(?!(.*wheel|scroll))|menu|drag|drop/i , fix: function (event, newEvent, type) { newEvent.rightClick = event.which === 3 || event.button === 2 newEvent.pos = { x: 0, y: 0 } if (event.pageX || event.pageY) { newEvent.clientX = event.pageX newEvent.clientY = event.pageY } else if (event.clientX || event.clientY) { newEvent.clientX = event.clientX + doc.body.scrollLeft + root.scrollLeft newEvent.clientY = event.clientY + doc.body.scrollTop + root.scrollTop } if (overOutRegex.test(type)) { newEvent.relatedTarget = event.relatedTarget || event[(type == 'mouseover' ? 'from' : 'to') + 'Element'] } return mouseProps } } , { // mouse wheel events reg: /mouse.*(wheel|scroll)/i , fix: function () { return mouseWheelProps } } , { // TextEvent reg: /^text/i , fix: function () { return textProps } } , { // touch and gesture events reg: /^touch|^gesture/i , fix: function () { return touchProps } } , { // message events reg: /^message$/i , fix: function () { return messageProps } } , { // popstate events reg: /^popstate$/i , fix: function () { return stateProps } } , { // everything else reg: /.*/ , fix: function () { return commonProps } } ] , typeFixerMap = {} // used to map event types to fixer functions (above), a basic cache mechanism , Event = function (event, element, isNative) { if (!arguments.length) return event = event || ((element.ownerDocument || element.document || element).parentWindow || win).event this.originalEvent = event this.isNative = isNative this.isBean = true if (!event) return var type = event.type , target = event.target || event.srcElement , i, l, p, props, fixer this.target = target && target.nodeType === 3 ? target.parentNode : target if (isNative) { // we only need basic augmentation on custom events, the rest expensive & pointless fixer = typeFixerMap[type] if (!fixer) { // haven't encountered this event type before, map a fixer function for it for (i = 0, l = typeFixers.length; i < l; i++) { if (typeFixers[i].reg.test(type)) { // guaranteed to match at least one, last is .* typeFixerMap[type] = fixer = typeFixers[i].fix break } } } props = fixer(event, this, type) for (i = props.length; i--;) { if (!((p = props[i]) in this) && p in event) this[p] = event[p] } } } // preventDefault() and stopPropagation() are a consistent interface to those functions // on the DOM, stop() is an alias for both of them together Event.prototype.preventDefault = function () { if (this.originalEvent.preventDefault) this.originalEvent.preventDefault() else this.originalEvent.returnValue = false } Event.prototype.stopPropagation = function () { if (this.originalEvent.stopPropagation) this.originalEvent.stopPropagation() else this.originalEvent.cancelBubble = true } Event.prototype.stop = function () { this.preventDefault() this.stopPropagation() this.stopped = true } // stopImmediatePropagation() has to be handled internally because we manage the event list for // each element // note that originalElement may be a Bean#Event object in some situations Event.prototype.stopImmediatePropagation = function () { if (this.originalEvent.stopImmediatePropagation) this.originalEvent.stopImmediatePropagation() this.isImmediatePropagationStopped = function () { return true } } Event.prototype.isImmediatePropagationStopped = function () { return this.originalEvent.isImmediatePropagationStopped && this.originalEvent.isImmediatePropagationStopped() } Event.prototype.clone = function (currentTarget) { //TODO: this is ripe for optimisation, new events are *expensive* // improving this will speed up delegated events var ne = new Event(this, this.element, this.isNative) ne.currentTarget = currentTarget return ne } return Event }()) // if we're in old IE we can't do onpropertychange on doc or win so we use doc.documentElement for both , targetElement = function (element, isNative) { return !W3C_MODEL && !isNative && (element === doc || element === win) ? root : element } /** * Bean maintains an internal registry for event listeners. We don't touch elements, objects * or functions to identify them, instead we store everything in the registry. * Each event listener has a RegEntry object, we have one 'registry' for the whole instance. */ , RegEntry = (function () { // each handler is wrapped so we can handle delegation and custom events var wrappedHandler = function (element, fn, condition, args) { var call = function (event, eargs) { return fn.apply(element, args ? slice.call(eargs, event ? 0 : 1).concat(args) : eargs) } , findTarget = function (event, eventElement) { return fn.__beanDel ? fn.__beanDel.ft(event.target, element) : eventElement } , handler = condition ? function (event) { var target = findTarget(event, this) // deleated event if (condition.apply(target, arguments)) { if (event) event.currentTarget = target return call(event, arguments) } } : function (event) { if (fn.__beanDel) event = event.clone(findTarget(event)) // delegated event, fix the fix return call(event, arguments) } handler.__beanDel = fn.__beanDel return handler } , RegEntry = function (element, type, handler, original, namespaces, args, root) { var customType = customEvents[type] , isNative if (type == 'unload') { // self clean-up handler = once(removeListener, element, type, handler, original) } if (customType) { if (customType.condition) { handler = wrappedHandler(element, handler, customType.condition, args) } type = customType.base || type } this.isNative = isNative = nativeEvents[type] && !!element[eventSupport] this.customType = !W3C_MODEL && !isNative && type this.element = element this.type = type this.original = original this.namespaces = namespaces this.eventType = W3C_MODEL || isNative ? type : 'propertychange' this.target = targetElement(element, isNative) this[eventSupport] = !!this.target[eventSupport] this.root = root this.handler = wrappedHandler(element, handler, null, args) } // given a list of namespaces, is our entry in any of them? RegEntry.prototype.inNamespaces = function (checkNamespaces) { var i, j, c = 0 if (!checkNamespaces) return true if (!this.namespaces) return false for (i = checkNamespaces.length; i--;) { for (j = this.namespaces.length; j--;) { if (checkNamespaces[i] == this.namespaces[j]) c++ } } return checkNamespaces.length === c } // match by element, original fn (opt), handler fn (opt) RegEntry.prototype.matches = function (checkElement, checkOriginal, checkHandler) { return this.element === checkElement && (!checkOriginal || this.original === checkOriginal) && (!checkHandler || this.handler === checkHandler) } return RegEntry }()) , registry = (function () { // our map stores arrays by event type, just because it's better than storing // everything in a single array. // uses '$' as a prefix for the keys for safety and 'r' as a special prefix for // rootListeners so we can look them up fast var map = {} // generic functional search of our registry for matching listeners, // `fn` returns false to break out of the loop , forAll = function (element, type, original, handler, root, fn) { var pfx = root ? 'r' : '$' if (!type || type == '*') { // search the whole registry for (var t in map) { if (t.charAt(0) == pfx) { forAll(element, t.substr(1), original, handler, root, fn) } } } else { var i = 0, l, list = map[pfx + type], all = element == '*' if (!list) return for (l = list.length; i < l; i++) { if ((all || list[i].matches(element, original, handler)) && !fn(list[i], list, i, type)) return } } } , has = function (element, type, original, root) { // we're not using forAll here simply because it's a bit slower and this // needs to be fast var i, list = map[(root ? 'r' : '$') + type] if (list) { for (i = list.length; i--;) { if (!list[i].root && list[i].matches(element, original, null)) return true } } return false } , get = function (element, type, original, root) { var entries = [] forAll(element, type, original, null, root, function (entry) { return entries.push(entry) }) return entries } , put = function (entry) { var has = !entry.root && !this.has(entry.element, entry.type, null, false) , key = (entry.root ? 'r' : '$') + entry.type ;(map[key] || (map[key] = [])).push(entry) return has } , del = function (entry) { forAll(entry.element, entry.type, null, entry.handler, entry.root, function (entry, list, i) { list.splice(i, 1) entry.removed = true if (list.length === 0) delete map[(entry.root ? 'r' : '$') + entry.type] return false }) } // dump all entries, used for onunload , entries = function () { var t, entries = [] for (t in map) { if (t.charAt(0) == '$') entries = entries.concat(map[t]) } return entries } return { has: has, get: get, put: put, del: del, entries: entries } }()) // we need a selector engine for delegated events, use querySelectorAll if it exists // but for older browsers we need Qwery, Sizzle or similar , selectorEngine , setSelectorEngine = function (e) { if (!arguments.length) { selectorEngine = doc.querySelectorAll ? function (s, r) { return r.querySelectorAll(s) } : function () { throw new Error('Bean: No selector engine installed') // eeek } } else { selectorEngine = e } } // we attach this listener to each DOM event that we need to listen to, only once // per event type per DOM element , rootListener = function (event, type) { if (!W3C_MODEL && type && event && event.propertyName != '_on' + type) return var listeners = registry.get(this, type || event.type, null, false) , l = listeners.length , i = 0 event = new Event(event, this, true) if (type) event.type = type // iterate through all handlers registered for this type, calling them unless they have // been removed by a previous handler or stopImmediatePropagation() has been called for (; i < l && !event.isImmediatePropagationStopped(); i++) { if (!listeners[i].removed) listeners[i].handler.call(this, event) } } // add and remove listeners to DOM elements , listener = W3C_MODEL ? function (element, type, add) { // new browsers element[add ? addEvent : removeEvent](type, rootListener, false) } : function (element, type, add, custom) { // IE8 and below, use attachEvent/detachEvent and we have to piggy-back propertychange events // to simulate event bubbling etc. var entry if (add) { registry.put(entry = new RegEntry( element , custom || type , function (event) { // handler rootListener.call(element, event, custom) } , rootListener , null , null , true // is root )) if (custom && element['_on' + custom] == null) element['_on' + custom] = 0 entry.target.attachEvent('on' + entry.eventType, entry.handler) } else { entry = registry.get(element, custom || type, rootListener, true)[0] if (entry) { entry.target.detachEvent('on' + entry.eventType, entry.handler) registry.del(entry) } } } , once = function (rm, element, type, fn, originalFn) { // wrap the handler in a handler that does a remove as well return function () { fn.apply(this, arguments) rm(element, type, originalFn) } } , removeListener = function (element, orgType, handler, namespaces) { var type = orgType && orgType.replace(nameRegex, '') , handlers = registry.get(element, type, null, false) , removed = {} , i, l for (i = 0, l = handlers.length; i < l; i++) { if ((!handler || handlers[i].original === handler) && handlers[i].inNamespaces(namespaces)) { // TODO: this is problematic, we have a registry.get() and registry.del() that // both do registry searches so we waste cycles doing this. Needs to be rolled into // a single registry.forAll(fn) that removes while finding, but the catch is that // we'll be splicing the arrays that we're iterating over. Needs extra tests to // make sure we don't screw it up. @rvagg registry.del(handlers[i]) if (!removed[handlers[i].eventType] && handlers[i][eventSupport]) removed[handlers[i].eventType] = { t: handlers[i].eventType, c: handlers[i].type } } } // check each type/element for removed listeners and remove the rootListener where it's no longer needed for (i in removed) { if (!registry.has(element, removed[i].t, null, false)) { // last listener of this type, remove the rootListener listener(element, removed[i].t, false, removed[i].c) } } } // set up a delegate helper using the given selector, wrap the handler function , delegate = function (selector, fn) { //TODO: findTarget (therefore $) is called twice, once for match and once for // setting e.currentTarget, fix this so it's only needed once var findTarget = function (target, root) { var i, array = isString(selector) ? selectorEngine(selector, root) : selector for (; target && target !== root; target = target.parentNode) { for (i = array.length; i--;) { if (array[i] === target) return target } } } , handler = function (e) { var match = findTarget(e.target, this) if (match) fn.apply(match, arguments) } // __beanDel isn't pleasant but it's a private function, not exposed outside of Bean handler.__beanDel = { ft : findTarget // attach it here for customEvents to use too , selector : selector } return handler } , fireListener = W3C_MODEL ? function (isNative, type, element) { // modern browsers, do a proper dispatchEvent() var evt = doc.createEvent(isNative ? 'HTMLEvents' : 'UIEvents') evt[isNative ? 'initEvent' : 'initUIEvent'](type, true, true, win, 1) element.dispatchEvent(evt) } : function (isNative, type, element) { // old browser use onpropertychange, just increment a custom property to trigger the event element = targetElement(element, isNative) isNative ? element.fireEvent('on' + type, doc.createEventObject()) : element['_on' + type]++ } /** * Public API: off(), on(), add(), (remove()), one(), fire(), clone() */ /** * off(element[, eventType(s)[, handler ]]) */ , off = function (element, typeSpec, fn) { var isTypeStr = isString(typeSpec) , k, type, namespaces, i if (isTypeStr && typeSpec.indexOf(' ') > 0) { // off(el, 't1 t2 t3', fn) or off(el, 't1 t2 t3') typeSpec = str2arr(typeSpec) for (i = typeSpec.length; i--;) off(element, typeSpec[i], fn) return element } type = isTypeStr && typeSpec.replace(nameRegex, '') if (type && customEvents[type]) type = customEvents[type].base if (!typeSpec || isTypeStr) { // off(el) or off(el, t1.ns) or off(el, .ns) or off(el, .ns1.ns2.ns3) if (namespaces = isTypeStr && typeSpec.replace(namespaceRegex, '')) namespaces = str2arr(namespaces, '.') removeListener(element, type, fn, namespaces) } else if (isFunction(typeSpec)) { // off(el, fn) removeListener(element, null, typeSpec) } else { // off(el, { t1: fn1, t2, fn2 }) for (k in typeSpec) { if (typeSpec.hasOwnProperty(k)) off(element, k, typeSpec[k]) } } return element } /** * on(element, eventType(s)[, selector], handler[, args ]) */ , on = function(element, events, selector, fn) { var originalFn, type, types, i, args, entry, first //TODO: the undefined check means you can't pass an 'args' argument, fix this perhaps? if (selector === undefined && typeof events == 'object') { //TODO: this can't handle delegated events for (type in events) { if (events.hasOwnProperty(type)) { on.call(this, element, type, events[type]) } } return } if (!isFunction(selector)) { // delegated event originalFn = fn args = slice.call(arguments, 4) fn = delegate(selector, originalFn, selectorEngine) } else { args = slice.call(arguments, 3) fn = originalFn = selector } types = str2arr(events) // special case for one(), wrap in a self-removing handler if (this === ONE) { fn = once(off, element, events, fn, originalFn) } for (i = types.length; i--;) { // add new handler to the registry and check if it's the first for this element/type first = registry.put(entry = new RegEntry( element , types[i].replace(nameRegex, '') // event type , fn , originalFn , str2arr(types[i].replace(namespaceRegex, ''), '.') // namespaces , args , false // not root )) if (entry[eventSupport] && first) { // first event of this type on this element, add root listener listener(element, entry.eventType, true, entry.customType) } } return element } /** * add(element[, selector], eventType(s), handler[, args ]) * * Deprecated: kept (for now) for backward-compatibility */ , add = function (element, events, fn, delfn) { return on.apply( null , !isString(fn) ? slice.call(arguments) : [ element, fn, events, delfn ].concat(arguments.length > 3 ? slice.call(arguments, 5) : []) ) } /** * one(element, eventType(s)[, selector], handler[, args ]) */ , one = function () { return on.apply(ONE, arguments) } /** * fire(element, eventType(s)[, args ]) * * The optional 'args' argument must be an array, if no 'args' argument is provided * then we can use the browser's DOM event system, otherwise we trigger handlers manually */ , fire = function (element, type, args) { var types = str2arr(type) , i, j, l, names, handlers for (i = types.length; i--;) { type = types[i].replace(nameRegex, '') if (names = types[i].replace(namespaceRegex, '')) names = str2arr(names, '.') if (!names && !args && element[eventSupport]) { fireListener(nativeEvents[type], type, element) } else { // non-native event, either because of a namespace, arguments or a non DOM element // iterate over all listeners and manually 'fire' handlers = registry.get(element, type, null, false) args = [false].concat(args) for (j = 0, l = handlers.length; j < l; j++) { if (handlers[j].inNamespaces(names)) { handlers[j].handler.apply(element, args) } } } } return element } /** * clone(dstElement, srcElement[, eventType ]) * * TODO: perhaps for consistency we should allow the same flexibility in type specifiers? */ , clone = function (element, from, type) { var handlers = registry.get(from, type, null, false) , l = handlers.length , i = 0 , args, beanDel for (; i < l; i++) { if (handlers[i].original) { args = [ element, handlers[i].type ] if (beanDel = handlers[i].handler.__beanDel) args.push(beanDel.selector) args.push(handlers[i].original) on.apply(null, args) } } return element } , bean = { 'on' : on , 'add' : add , 'one' : one , 'off' : off , 'remove' : off , 'clone' : clone , 'fire' : fire , 'Event' : Event , 'setSelectorEngine' : setSelectorEngine , 'noConflict' : function () { context[name] = old return this } } // for IE, clean up on unload to avoid leaks if (win.attachEvent) { var cleanup = function () { var i, entries = registry.entries() for (i in entries) { if (entries[i].type && entries[i].type !== 'unload') off(entries[i].element, entries[i].type) } win.detachEvent('onunload', cleanup) win.CollectGarbage && win.CollectGarbage() } win.attachEvent('onunload', cleanup) } // initialize selector engine to internal default (qSA or throw Error) setSelectorEngine(Ink.ss) return bean }); /** * Keep this declaration here and off Bean as it extends the Event * object and some properties are readonly in strict mode */ 'use strict'; var InkEvent = { KEY_BACKSPACE: 8, KEY_TAB: 9, KEY_RETURN: 13, KEY_ESC: 27, KEY_SPACE: 32, KEY_LEFT: 37, KEY_UP: 38, KEY_RIGHT: 39, KEY_DOWN: 40, KEY_DELETE: 46, KEY_HOME: 36, KEY_END: 35, KEY_PAGEUP: 33, KEY_PAGEDOWN: 34, KEY_INSERT: 45, /** * Creates a debounced version of a function. * Returns a function which calls `func`, waiting at least `wait` milliseconds between calls. This is useful for events such as `scroll` or `resize`, which can be triggered too many times per second, slowing down the browser with needless function calls. * * *note:* This does not delay the first function call to the function. * * @method throttle * @param {Function} func Function to call. Arguments and context are both passed. * @param {Number} [wait]=0 Milliseconds to wait between calls. * @sample Ink_Dom_Event_1_throttle.html **/ throttle: function (func, wait) { wait = wait || 0; var lastCall = 0; // Warning: This breaks on Jan 1st 1970 0:00 var timeout; var throttled = function () { var now = +new Date(); var timeDiff = now - lastCall; if (timeDiff >= wait) { lastCall = now; return func.apply(this, [].slice.call(arguments)); } else { var that = this; var args = [].slice.call(arguments); if (!timeout) { timeout = setTimeout(function () { timeout = null; return throttled.apply(that, args); }, wait - timeDiff); } } }; return throttled; }, /** * Gets the event's target element. * * @method element * @param {Object} ev Event object * @return {DOMNode} The target * @sample Ink_Dom_Event_1_element.html */ element: function(ev) { var node = ev.delegationTarget || ev.target || // IE stuff (ev.type === 'mouseout' && ev.fromElement) || (ev.type === 'mouseleave' && ev.fromElement) || (ev.type === 'mouseover' && ev.toElement) || (ev.type === 'mouseenter' && ev.toElement) || ev.srcElement || null; return node && (node.nodeType === 3 || node.nodeType === 4) ? node.parentNode : node; }, /** * Gets the event's related target element. * * @method relatedTarget * @param {Object} ev event object * @return {DOMNode} The related target * @sample Ink_Dom_Event_1_relatedTarget.html */ relatedTarget: function(ev){ var node = ev.relatedTarget || // IE stuff (ev.type === 'mouseout' && ev.toElement) || (ev.type === 'mouseleave' && ev.toElement) || (ev.type === 'mouseover' && ev.fromElement) || (ev.type === 'mouseenter' && ev.fromElement) || null; return node && (node.nodeType === 3 || node.nodeType === 4) ? node.parentNode : node; }, /** * Find closest ancestor element by tag name related to the event target. * Navigate up the DOM tree, looking for a tag with the name `elmTagName`. * * If such tag is not found, `document` is returned. * * @method findElement * @param {Object} ev Event object * @param {String} elmTagName Tag name to find * @param {Boolean} [force]=false Flag to skip returning `document` and to return `false` instead. * @return {DOMElement} the first element which matches given tag name or the document element if the wanted tag is not found * @sample Ink_Dom_Event_1_findElement.html */ findElement: function(ev, elmTagName, force) { var node = this.element(ev); while(true) { if(node.nodeName.toLowerCase() === elmTagName.toLowerCase()) { return node; } else { node = node.parentNode; if(!node) { if(force) { return false; } return document; } if(!node.parentNode){ if(force){ return false; } return document; } } } }, /** * Attaches an event to element * * @method observe * @param {DOMElement|String} element Element id or element * @param {String} eventName Event name * @param {Function} callBack Receives the event object as a parameter. If you're manually firing custom events, check it's eventName property to make sure you're handling the right event. * @param {Boolean} [useCapture] Flag to change event listening from bubbling to capture. * @return {Function} The event handler used. Hang on to this if you want to `stopObserving` later. * @sample Ink_Dom_Event_1_observe.html */ observe: function(element, eventName, callBack, useCapture) { element = Ink.i(element); if(element) { if(element.addEventListener) { element.addEventListener(eventName, callBack, !!useCapture); } else { element.attachEvent('on' + eventName, (callBack = Ink.bind(callBack, element))); } return callBack; } }, /** * Like observe, but listen to the event only once. * * @method observeOnce * @param {DOMElement|String} element Element id or element * @param {String} eventName Event name * @param {Function} callBack Receives the event object as a parameter. If you're manually firing custom events, check it's eventName property to make sure you're handling the right event. * @param {Boolean} [useCapture] Flag to change event listening from bubbling to capture. * @return {Function} The event handler used. Hang on to this if you want to `stopObserving` later. * @sample Ink_Dom_Event_1_observeOnce.html */ observeOnce: function (element, eventName, callBack, useCapture) { var onceBack = function () { InkEvent.stopObserving(element, eventName, onceBack); return callBack(); }; return InkEvent.observe(element, eventName, onceBack, useCapture); }, /** * Attaches an event to a selector or array of elements. * * @method observeMulti * @param {Array|String} elements * @param {String} eventName Event name * @param {Function} callBack Receives the event object as a parameter. If you're manually firing custom events, check it's eventName property to make sure you're handling the right event. * @param {Boolean} [useCapture] Flag change event listening from bubbling to capture. * @return {Function} The used callback. * @sample Ink_Dom_Event_1_observeMulti.html */ observeMulti: function (elements, eventName, callBack, useCapture) { if (typeof elements === 'string') { elements = Ink.ss(elements); } else if ( /* is an element */ elements && elements.nodeType === 1) { elements = [elements]; } if (!elements[0]) { return false; } for (var i = 0, len = elements.length; i < len; i++) { this.observe(elements[i], eventName, callBack, useCapture); } return callBack; }, /** * Observes an event on an element and its descendants matching the selector. * * Requires Ink.Dom.Selector if you need to use a selector. * * @method observeDelegated * @param {DOMElement|String} element Element to observe. * @param {String} eventName Event name to observe. * @param {String} selector Child element selector. When null, finds any element. * @param {Function} callback Callback to be called when the event is fired * @return {Function} The used callback, for ceasing to listen to the event later. * @sample Ink_Dom_Event_1_observeDelegated.html **/ observeDelegated: function (element, eventName, selector, callback) { return InkEvent.observe(element, eventName, function (event) { var fromElement = InkEvent.element(event); if (!fromElement || fromElement === element) { return; } var cursor = fromElement; // Go up the document tree until we hit the element itself. while (cursor !== element && cursor !== document && cursor) { if (Ink.Dom.Selector_1.matchesSelector(cursor, selector)) { event.delegationTarget = cursor; return callback(event); } cursor = cursor.parentNode; } }); }, /** * Removes an event attached to an element. * * @method stopObserving * @param {DOMElement|String} element Element id or element * @param {String} eventName Event name * @param {Function} callBack Callback function * @param {Boolean} [useCapture] Set to true if the event was being observed with useCapture set to true as well. * @sample Ink_Dom_Event_1_stopObserving.html */ stopObserving: function(element, eventName, callBack, useCapture) { element = Ink.i(element); if(element) { if(element.removeEventListener) { element.removeEventListener(eventName, callBack, !!useCapture); } else { element.detachEvent('on' + eventName, callBack); } } }, /** * Stops event propagation and bubbling. * * @method stop * @param {Object} event Event handle * @sample Ink_Dom_Event_1_stop.html */ stop: function(event) { if(event.cancelBubble !== null) { event.cancelBubble = true; } if(event.stopPropagation) { event.stopPropagation(); } if(event.preventDefault) { event.preventDefault(); } if(window.attachEvent) { event.returnValue = false; } if(event.cancel !== null) { event.cancel = true; } }, /** * Stops event propagation. * * @method stopPropagation * @param {Object} event Event handle * @sample Ink_Dom_Event_1_stopPropagation.html */ stopPropagation: function(event) { if(event.cancelBubble !== null) { event.cancelBubble = true; } if(event.stopPropagation) { event.stopPropagation(); } }, /** * Stops event default behaviour. * * @method stopDefault * @param {Object} event Event handle * @sample Ink_Dom_Event_1_stopDefault.html */ stopDefault: function(event) { if(event.preventDefault) { event.preventDefault(); } if(window.attachEvent) { event.returnValue = false; } if(event.cancel !== null) { event.cancel = true; } }, /** * Gets the pointer's coordinates from the event object. * * @method pointer * @param {Object} ev Event object * @return {Object} An object with the mouse X and Y position * @sample Ink_Dom_Event_1_pointer.html */ pointer: function(ev) { return { x: this.pointerX(ev), y: this.pointerY(ev) }; }, /** * Gets the pointer's X coordinate. * * @method pointerX * @param {Object} ev Event object * @return {Number} Mouse X position */ pointerX: function(ev) { return (ev.touches && ev.touches[0] && ev.touches[0].clientX) || (ev.pageX) || (ev.clientX + (document.documentElement.scrollLeft || document.body.scrollLeft)); }, /** * Gets the pointer's Y coordinate. * * @method pointerY * @param {Object} ev Event object * @return {Number} Mouse Y position */ pointerY: function(ev) { return (ev.touches && ev.touches[0] && ev.touches[0].clientY) || (ev.pageY) || (ev.clientY + (document.documentElement.scrollTop || document.body.scrollTop)); }, /** * Checks if an event is a left click. * * @method isLeftClick * @param {Object} ev Event object * @return {Boolean} True if the event is a left click * @sample Ink_Dom_Event_1_isLeftClick.html */ isLeftClick: function(ev) { if (window.addEventListener) { if(ev.button === 0){ return true; } else if(ev.type === 'touchend' && ev.button === null){ // [todo] do the above check for pointerEvents too return true; } } else { if(ev.button === 1){ return true; } } return false; }, /** * Checks if an event is a right click. * * @method isRightClick * @param {Object} ev Event object * @return {Boolean} True if the event is a right click * @sample Ink_Dom_Event_1_isRightClick.html */ isRightClick: function(ev) { return (ev.button === 2); }, /** * Checks if an event is a middle click. * * @method isMiddleClick * @param {Object} ev Event object * @return {Boolean} True if the event is a middle click * @sample Ink_Dom_Event_1_isMiddleClick.html */ isMiddleClick: function(ev) { if (window.addEventListener) { return (ev.button === 1); } else { return (ev.button === 4); } return false; }, /** * Gets character from an event. * * @method getCharFromKeyboardEvent * @param {Object} event Keyboard event * @param {Boolean} [changeCasing] If true uppercases, if false lowercases, otherwise keeps casing * @return {String} Character representation of pressed key combination * @sample Ink_Dom_Event_1_getCharFromKeyboardEvent.html */ getCharFromKeyboardEvent: function(event, changeCasing) { var k = event.keyCode; var c = String.fromCharCode(k); var shiftOn = event.shiftKey; if (k >= 65 && k <= 90) { // A-Z if (typeof changeCasing === 'boolean') { shiftOn = changeCasing; } return (shiftOn) ? c : c.toLowerCase(); } else if (k >= 96 && k <= 105) { // numpad digits return String.fromCharCode( 48 + (k-96) ); } switch (k) { case 109: case 189: return '-'; case 107: case 187: return '+'; } return c; }, debug: function(){} }; /** * Lets you attach event listeners to both elements and objects. * http://github.com/fat/bean#on * * @method on * @param {DOMElement|Object} element An HTML DOM element or any JavaScript Object * @param {String} eventType An Event (or multiple events, space separated) to listen to * @param {String} [selector] A CSS DOM Element selector string to bind the listener to child elements matching the selector * @param {Function} [handler] The callback function * @param {Object} [args...] Additional arguments to pass to the callback function when triggered * * @return {DOMElement|Object} Returns the original DOM Element or Javascript Object * @sample Ink_Dom_Event_1_on.html */ /** * Alias for `on` but will only be executed once. * bean.one() is an alias for bean.on() except that the handler will only be executed once and then removed for the event type(s). * http://github.com/fat/bean#one * * @method one * @param {DOMElement|Object} element An HTML DOM element or any JavaScript Object * @param {String} eventType An Event (or multiple events, space separated) to listen to * @param {String} [selector] A CSS DOM Element selector string to bind the listener to child elements matching the selector * @param {Function} [handler] The callback function * @param [args...] Additional arguments to pass to the callback function when triggered * * @return {DOMElement|Object} Returns the original DOM Element or Javascript Object * @sample Ink_Dom_Event_1_one.html */ /** * Removes event handlers. * bean.off() is how you get rid of handlers once you no longer want them active. It's also a good idea to call off on elements before you remove them from your DOM; this gives Bean a chance to clean up some things and prevents memory leaks. * http://github.com/fat/bean#off * * @method off * @param {DOMElement|Object} element An HTML DOM element or any JavaScript Object * @param {String} eventType An Event (or multiple events, space separated) to remove * @param {Function} [handler] The specific callback function to remove * * @return {DOMElement|Object} Returns the original DOM Element or Javascript Object * @sample Ink_Dom_Event_1_off.html */ /** * Clones events from one object to another * bean.clone() is a method for cloning events from one DOM element or object to another. * http://github.com/fat/bean#clone * * @method clone * @param {DOMElement|Object} destElement An HTML DOM element or any JavaScript Object to copy events to * @param {String} srcElement An HTML DOM element or any JavaScript Object to copy events from * @param {String} [eventType] An Event (or multiple events, space separated) to clone * * @return {DOMElement|Object} Returns the original DOM Element or Javascript Object * @sample Ink_Dom_Event_1_clone.html */ /** * Triggers events. * http://github.com/fat/bean#fire * * @method fire * @param {DOMElement|Object} destElement An HTML DOM element or any JavaScript Object fire the event on * @param {String} eventType An Event (or multiple events, space separated) to fire * @param [args...] Additional arguments to pass to the callback function when triggered * * @return {DOMElement|Object} Returns the original DOM Element or Javascript Object * @sample Ink_Dom_Event_1_fire.html */ return Ink.extendObj(InkEvent, bean); }); /** * @module Ink.Dom.FormSerialize_1 * Two way serialization of form data and javascript objects. * Valid applications are ad hoc AJAX/syndicated submission of forms, restoring form values from server side state, etc. */ Ink.createModule('Ink.Dom.FormSerialize', 1, ['Ink.Util.Array_1', 'Ink.Dom.Element_1', 'Ink.Dom.Selector_1'], function (InkArray, InkElement, Selector) { 'use strict'; // Check whether something is not a string or a DOM element, but still has length. function isArrayIsh(obj) { return obj != null && (!InkElement.isDOMElement(obj)) && (InkArray.isArray(obj) || (typeof obj !== 'string' && typeof obj.length === 'number')); } function toArray(obj) { if (isArrayIsh(obj)) { return obj; } else { return [obj]; } } /** * @namespace Ink.Dom.FormSerialize * @static **/ var FormSerialize = { /** * Serializes a form element into a JS object * It turns field names into keys and field values into values. * * note: Multi-select and checkboxes with multiple values will result in arrays * * @method serialize * @param {DOMElement|String} form Form element to extract data * @return {Object} Map of fieldName -> String|String[]|Boolean * @sample Ink_Dom_FormSerialize_serialize.html */ serialize: function(form) { var out = {}; var emptyArrayToken = {}; // A hack so that empty select[multiple] elements appear although empty. var pairs = this.asPairs(form, { elements: true, emptyArray: emptyArrayToken }); if (pairs == null) { return pairs; } InkArray.forEach(pairs, function (pair) { var name = pair[0].replace(/\[\]$/, ''); var value = pair[1]; var el = pair[2]; if (value === emptyArrayToken) { out[name] = []; // It's an empty select[multiple] } else if (!(FormSerialize._resultsInArray(el) || /\[\]$/.test(pair[0]))) { out[name] = value; } else { out[name] = out[name] || []; out[name].push(value); } }); return out; }, /** * Like `serialize`, but returns an array of [fieldName, value] pairs. * * @method asPairs * @param {DOMElement|String} form Form element * @param {Object} [options] Options object, containing: * @param {Boolean} [options.elements] Instead of returning an array of [fieldName, value] pairs, return an array of [fieldName, value, fieldElement] triples. * @param {Boolean} [options.emptyArray] What to emit as the value of an empty select[multiple]. If you don't pass this option, nothing comes out. * * @return Array of [fieldName, value] pairs. **/ asPairs: function (form, options) { var out = []; options = options || {}; function emit(name, val, el) { if (options.elements) { out.push([name, val, el]); } else { out.push([name, val]); } } function serializeEl(el) { if (el.nodeName.toLowerCase() === 'select' && el.multiple) { var didEmit = false; InkArray.forEach(Selector.select('option:checked', el), function (thisOption) { emit(el.name, thisOption.value, el); didEmit = true; }); if (!didEmit && 'emptyArray' in options) { emit(el.name, options.emptyArray, el); } } else { emit(el.name, el.value, el); } } if ((form = Ink.i(form))) { var inputs = InkArray.filter(form.elements, FormSerialize._isSerialized); for (var i = 0, len = inputs.length; i < len; i++) { serializeEl(inputs[i]); } return out; } return null; }, /** * Sets form elements' values with values from an object * * Note: You can't set the values of an input with `type="file"` (browser prohibits it) * * @method fillIn * @param {DOMElement|String} form Form element to be populated * @param {Object|Array} map2 mapping of fields to values contained in fields. Can be a hash (keys as names, strings or arrays for values), or an array of [name, value] pairs. * @sample Ink_Dom_FormSerialize_fillIn.html */ fillIn: function(form, map2) { if (!(form = Ink.i(form))) { return null; } var pairs; if (typeof map2 === 'object' && !isArrayIsh(map2)) { pairs = FormSerialize._objToPairs(map2); } else if (isArrayIsh(map2)) { pairs = map2; } else { return null; } return FormSerialize._fillInPairs(form, pairs); }, _objToPairs: function (obj) { var pairs = []; var val; for (var name in obj) if (obj.hasOwnProperty(name)) { val = toArray(obj[name]); for (var i = 0, len = val.length; i < len; i++) { pairs.push([name, val[i]]); } if (len === 0) { pairs.push([name, []]); } } return pairs; }, _fillInPairs: function (form, pairs) { pairs = InkArray.groupBy(pairs, { key: function (pair) { return pair[0].replace(/\[\]$/, ''); } }); // For each chunk... pairs = InkArray.map(pairs, function (pair) { // Join the items in the chunk by concatenating the values together and leaving the names alone var values = InkArray.reduce(pair, function (left, right) { return [null, left[1].concat([right[1]])]; }, [null, []])[1]; return [pair[0][0], values]; }); var name; var inputs; var values; for (var i = 0, len = pairs.length; i < len; i++) { name = pairs[i][0]; if (name in form) { inputs = form[name]; } else if ((name + '[]') in form) { inputs = form[name + '[]']; name = name + '[]'; } else { continue; } inputs = toArray(inputs); values = pairs[i][1]; FormSerialize._fillInOne(name, inputs, values); } }, _fillInOne: function (name, inputs, values) { var firstOne = inputs[0]; var firstNodeName = firstOne.nodeName.toLowerCase(); var firstType = firstOne.getAttribute('type'); firstType = firstType && firstType.toLowerCase(); var isSelectMulti = firstNodeName === 'select' && InkElement.hasAttribute(firstOne, 'multiple'); if (firstType === 'checkbox' || firstType === 'radio') { FormSerialize._fillInBoolean(inputs, values, 'checked'); } else if (isSelectMulti) { FormSerialize._fillInBoolean(inputs[0].options, values, 'selected'); } else { if (inputs.length !== values.length) { Ink.warn('Form had ' + inputs.length + ' inputs named "' + name + '", but received ' + values.length + ' values.'); } for (var i = 0, len = Math.min(inputs.length, values.length); i < len; i += 1) { inputs[i].value = values[i]; } } }, _fillInBoolean: function (inputs, values, checkAttr /* 'selected' or 'checked' */) { InkArray.forEach(inputs, function (input) { var isChecked = InkArray.inArray(input.value, values); input[checkAttr] = isChecked; }); }, /** * Whether FormSerialize.serialize() should produce an array when looking at this element. * @method _resultsInArray * @private * @param element **/ _resultsInArray: function (element) { var type = element.getAttribute('type'); var nodeName = element.nodeName.toLowerCase(); return type === 'checkbox' || (nodeName === 'select' && InkElement.hasAttribute(element, 'multiple')); }, _isSerialized: function (element) { if (!InkElement.isDOMElement(element)) { return false; } if (!InkElement.hasAttribute(element, 'name')) { return false; } var nodeName = element.nodeName.toLowerCase(); if (!nodeName || nodeName === 'fieldset') { return false; } if (element.type === 'checkbox' || element.type === 'radio') { return !!element.checked; } return true; } }; return FormSerialize; }); /** * Execute code only when the DOM is loaded. * @module Ink.Dom.Loaded_1 * @version 1 */ Ink.createModule('Ink.Dom.Loaded', 1, [], function() { 'use strict'; /** * @namespace Ink.Dom.Loaded_1 **/ var Loaded = { /** * Callbacks and their contexts. Array of 2-arrays. * * [] * * @attribute _contexts Array * @private * */ _contexts: [], // Callbacks' queue /** * Specify a function to execute when the DOM is fully loaded. * * @method run * @param {Object} [win]=window Window object to attach/add the event * @param {Function} fn Callback function to be executed after the DOM is ready * @public * @sample Ink_Dom_Loaded_run.html */ run: function(win, fn) { if (!fn) { fn = win; win = window; } var context; for (var i = 0, len = this._contexts.length; i < len; i++) { if (this._contexts[i][0] === win) { context = this._contexts[i][1]; break; } } if (!context) { context = { cbQueue: [], win: win, doc: win.document, root: win.document.documentElement, done: false, top: true }; context.handlers = { checkState: Ink.bindEvent(this._checkState, this, context), poll: Ink.bind(this._poll, this, context) }; this._contexts.push( [win, context] // Javascript Objects cannot map different windows to // different values. ); } var ael = context.doc.addEventListener; context.add = ael ? 'addEventListener' : 'attachEvent'; context.rem = ael ? 'removeEventListener' : 'detachEvent'; context.pre = ael ? '' : 'on'; context.det = ael ? 'DOMContentLoaded' : 'onreadystatechange'; context.wet = context.pre + 'load'; var csf = context.handlers.checkState; var alreadyLoaded = ( /complete|interactive|loaded/.test(context.doc.readyState) && context.win.location.toString() !== 'about:blank'); // https://code.google.com/p/chromium/issues/detail?id=32357 if (alreadyLoaded){ setTimeout(Ink.bind(function () { fn.call(context.win, 'lazy'); }, this), 0); } else { context.cbQueue.push(fn); context.doc[context.add]( context.det , csf ); context.win[context.add]( context.wet , csf ); var frameElement = 1; try{ frameElement = context.win.frameElement; } catch(e) {} if ( !ael && context.root && context.root.doScroll ) { // IE HACK try { context.top = !frameElement; } catch(e) { } if (context.top) { this._poll(context); } } } }, /** * Function that will be running the callbacks after the page is loaded * * @method _checkState * @param {Event} event Triggered event * @private */ _checkState: function(event, context) { if ( !event || (event.type === 'readystatechange' && context.doc.readyState !== 'complete')) { return; } var where = (event.type === 'load') ? context.win : context.doc; where[context.rem](context.pre+event.type, context.handlers.checkState, false); this._ready(context); }, /** * Polls the load progress of the page to see if it has already loaded or not * * @method _poll * @private */ /** * * function _poll */ _poll: function(context) { try { context.root.doScroll('left'); } catch(e) { return setTimeout(context.handlers.poll, 50); } this._ready(context); }, /** * Function that runs the callbacks from the queue when the document is ready. * * @method _ready * @private */ _ready: function(context) { if (!context.done) { context.done = true; for (var i = 0; i < context.cbQueue.length; ++i) { context.cbQueue[i].call(context.win); } context.cbQueue = []; } } }; return Loaded; }); /** * CSS selector engine * @module Ink.Dom.Selector_1 * @version 1 */ Ink.createModule('Ink.Dom.Selector', 1, [], function() { /*jshint forin:false, eqnull:true, noempty:false, expr:true, boss:true, maxdepth:false*/ 'use strict'; /*! * Sizzle CSS Selector Engine * Copyright 2013 jQuery Foundation and other contributors * Released under the MIT license * http://sizzlejs.com/ */ var i, cachedruns, Expr, getText, isXML, compile, outermostContext, recompare, sortInput, // Local document vars setDocument, document, docElem, documentIsHTML, rbuggyQSA, rbuggyMatches, matches, contains, // Instance-specific data expando = "sizzle" + -(new Date()), preferredDoc = window.document, support = {}, dirruns = 0, done = 0, classCache = createCache(), tokenCache = createCache(), compilerCache = createCache(), hasDuplicate = false, sortOrder = function() { return 0; }, // General-purpose constants strundefined = typeof undefined, MAX_NEGATIVE = 1 << 31, // Array methods arr = [], pop = arr.pop, push_native = arr.push, push = arr.push, slice = arr.slice, // Use a stripped-down indexOf if we can't use a native one indexOf = arr.indexOf || function( elem ) { var i = 0, len = this.length; for ( ; i < len; i++ ) { if ( this[i] === elem ) { return i; } } return -1; }, // Regular expressions // Whitespace characters http://www.w3.org/TR/css3-selectors/#whitespace whitespace = "[\\x20\\t\\r\\n\\f]", // http://www.w3.org/TR/css3-syntax/#characters characterEncoding = "(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+", // Loosely modeled on CSS identifier characters // An unquoted value should be a CSS identifier http://www.w3.org/TR/css3-selectors/#attribute-selectors // Proper syntax: http://www.w3.org/TR/CSS21/syndata.html#value-def-identifier identifier = characterEncoding.replace( "w", "w#" ), // Acceptable operators http://www.w3.org/TR/selectors/#attribute-selectors operators = "([*^$|!~]?=)", attributes = "\\[" + whitespace + "*(" + characterEncoding + ")" + whitespace + "*(?:" + operators + whitespace + "*(?:(['\"])((?:\\\\.|[^\\\\])*?)\\3|(" + identifier + ")|)|)" + whitespace + "*\\]", // Prefer arguments quoted, // then not containing pseudos/brackets, // then attribute selectors/non-parenthetical expressions, // then anything else // These preferences are here to reduce the number of selectors // needing tokenize in the PSEUDO preFilter pseudos = ":(" + characterEncoding + ")(?:\\(((['\"])((?:\\\\.|[^\\\\])*?)\\3|((?:\\\\.|[^\\\\()[\\]]|" + attributes.replace( 3, 8 ) + ")*)|.*)\\)|)", // Leading and non-escaped trailing whitespace, capturing some non-whitespace characters preceding the latter rtrim = new RegExp( "^" + whitespace + "+|((?:^|[^\\\\])(?:\\\\.)*)" + whitespace + "+$", "g" ), rcomma = new RegExp( "^" + whitespace + "*," + whitespace + "*" ), rcombinators = new RegExp( "^" + whitespace + "*([\\x20\\t\\r\\n\\f>+~])" + whitespace + "*" ), rpseudo = new RegExp( pseudos ), ridentifier = new RegExp( "^" + identifier + "$" ), matchExpr = { "ID": new RegExp( "^#(" + characterEncoding + ")" ), "CLASS": new RegExp( "^\\.(" + characterEncoding + ")" ), "NAME": new RegExp( "^\\[name=['\"]?(" + characterEncoding + ")['\"]?\\]" ), "TAG": new RegExp( "^(" + characterEncoding.replace( "w", "w*" ) + ")" ), "ATTR": new RegExp( "^" + attributes ), "PSEUDO": new RegExp( "^" + pseudos ), "CHILD": new RegExp( "^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\(" + whitespace + "*(even|odd|(([+-]|)(\\d*)n|)" + whitespace + "*(?:([+-]|)" + whitespace + "*(\\d+)|))" + whitespace + "*\\)|)", "i" ), // For use in libraries implementing .is() // We use this for POS matching in `select` "needsContext": new RegExp( "^" + whitespace + "*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\(" + whitespace + "*((?:-\\d)?\\d*)" + whitespace + "*\\)|)(?=[^-]|$)", "i" ) }, rsibling = /[\x20\t\r\n\f]*[+~]/, rnative = /^[^{]+\{\s*\[native code/, // Easily-parseable/retrievable ID or TAG or CLASS selectors rquickExpr = /^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/, rinputs = /^(?:input|select|textarea|button)$/i, rheader = /^h\d$/i, rescape = /'|\\/g, rattributeQuotes = /\=[\x20\t\r\n\f]*([^'"\]]*)[\x20\t\r\n\f]*\]/g, // CSS escapes http://www.w3.org/TR/CSS21/syndata.html#escaped-characters runescape = /\\([\da-fA-F]{1,6}[\x20\t\r\n\f]?|.)/g, funescape = function( _, escaped ) { var high = "0x" + escaped - 0x10000; // NaN means non-codepoint return high !== high ? escaped : // BMP codepoint high < 0 ? String.fromCharCode( high + 0x10000 ) : // Supplemental Plane codepoint (surrogate pair) String.fromCharCode( high >> 10 | 0xD800, high & 0x3FF | 0xDC00 ); }; // Optimize for push.apply( _, NodeList ) try { push.apply( (arr = slice.call( preferredDoc.childNodes )), preferredDoc.childNodes ); // Support: Android<4.0 // Detect silently failing push.apply arr[ preferredDoc.childNodes.length ].nodeType; } catch ( e ) { push = { apply: arr.length ? // Leverage slice if possible function( target, els ) { push_native.apply( target, slice.call(els) ); } : // Support: IE<9 // Otherwise append directly function( target, els ) { var j = target.length, i = 0; // Can't trust NodeList.length while ( (target[j++] = els[i++]) ) {} target.length = j - 1; } }; } /* * For feature detection * @param {Function} fn The function to test for native support */ function isNative( fn ) { return rnative.test( fn + "" ); } /* * Create key-value caches of limited size * @returns {Function(string, Object)} Returns the Object data after storing it on itself with * property name the (space-suffixed) string and (if the cache is larger than Expr.cacheLength) * deleting the oldest entry */ function createCache() { var cache, keys = []; return (cache = function( key, value ) { // Use (key + " ") to avoid collision with native prototype properties (see Issue #157) if ( keys.push( key += " " ) > Expr.cacheLength ) { // Only keep the most recent entries delete cache[ keys.shift() ]; } return (cache[ key ] = value); }); } /* * Mark a function for special use by Sizzle * @param {Function} fn The function to mark */ function markFunction( fn ) { fn[ expando ] = true; return fn; } /* * Support testing using an element * @param {Function} fn Passed the created div and expects a boolean result */ function assert( fn ) { var div = document.createElement("div"); try { return !!fn( div ); } catch (e) { return false; } finally { // release memory in IE div = null; } } function Sizzle( selector, context, results, seed ) { var match, elem, m, nodeType, // QSA vars i, groups, old, nid, newContext, newSelector; if ( ( context ? context.ownerDocument || context : preferredDoc ) !== document ) { setDocument( context ); } context = context || document; results = results || []; if ( !selector || typeof selector !== "string" ) { return results; } if ( (nodeType = context.nodeType) !== 1 && nodeType !== 9 ) { return []; } if ( documentIsHTML && !seed ) { // Shortcuts if ( (match = rquickExpr.exec( selector )) ) { // Speed-up: Sizzle("#ID") if ( (m = match[1]) ) { if ( nodeType === 9 ) { elem = context.getElementById( m ); // Check parentNode to catch when Blackberry 4.6 returns // nodes that are no longer in the document #6963 if ( elem && elem.parentNode ) { // Handle the case where IE, Opera, and Webkit return items // by name instead of ID if ( elem.id === m ) { results.push( elem ); return results; } } else { return results; } } else { // Context is not a document if ( context.ownerDocument && (elem = context.ownerDocument.getElementById( m )) && contains( context, elem ) && elem.id === m ) { results.push( elem ); return results; } } // Speed-up: Sizzle("TAG") } else if ( match[2] ) { push.apply( results, context.getElementsByTagName( selector ) ); return results; // Speed-up: Sizzle(".CLASS") } else if ( (m = match[3]) && support.getElementsByClassName && context.getElementsByClassName ) { push.apply( results, context.getElementsByClassName( m ) ); return results; } } // QSA path if ( support.qsa && !rbuggyQSA.test(selector) ) { old = true; nid = expando; newContext = context; newSelector = nodeType === 9 && selector; // qSA works strangely on Element-rooted queries // We can work around this by specifying an extra ID on the root // and working up from there (Thanks to Andrew Dupont for the technique) // IE 8 doesn't work on object elements if ( nodeType === 1 && context.nodeName.toLowerCase() !== "object" ) { groups = tokenize( selector ); if ( (old = context.getAttribute("id")) ) { nid = old.replace( rescape, "\\$&" ); } else { context.setAttribute( "id", nid ); } nid = "[id='" + nid + "'] "; i = groups.length; while ( i-- ) { groups[i] = nid + toSelector( groups[i] ); } newContext = rsibling.test( selector ) && context.parentNode || context; newSelector = groups.join(","); } if ( newSelector ) { try { push.apply( results, newContext.querySelectorAll( newSelector ) ); return results; } catch(qsaError) { } finally { if ( !old ) { context.removeAttribute("id"); } } } } } // All others return select( selector.replace( rtrim, "$1" ), context, results, seed ); } /* * Detect xml * @param {Element|Object} elem An element or a document */ isXML = Sizzle.isXML = function( elem ) { // documentElement is verified for cases where it doesn't yet exist // (such as loading iframes in IE - #4833) var documentElement = elem && (elem.ownerDocument || elem).documentElement; return documentElement ? documentElement.nodeName !== "HTML" : false; }; /* * Sets document-related variables once based on the current document * @param {Element|Object} [doc] An element or document object to use to set the document * @returns {Object} Returns the current document */ setDocument = Sizzle.setDocument = function( node ) { var doc = node ? node.ownerDocument || node : preferredDoc; // If no document and documentElement is available, return if ( doc === document || doc.nodeType !== 9 || !doc.documentElement ) { return document; } // Set our document document = doc; docElem = doc.documentElement; // Support tests documentIsHTML = !isXML( doc ); // Check if getElementsByTagName("*") returns only elements support.getElementsByTagName = assert(function( div ) { div.appendChild( doc.createComment("") ); return !div.getElementsByTagName("*").length; }); // Check if attributes should be retrieved by attribute nodes support.attributes = assert(function( div ) { div.innerHTML = ""; var type = typeof div.lastChild.getAttribute("multiple"); // IE8 returns a string for some attributes even when not present return type !== "boolean" && type !== "string"; }); // Check if getElementsByClassName can be trusted support.getElementsByClassName = assert(function( div ) { // Opera can't find a second classname (in 9.6) div.innerHTML = ""; if ( !div.getElementsByClassName || !div.getElementsByClassName("e").length ) { return false; } // Safari 3.2 caches class attributes and doesn't catch changes div.lastChild.className = "e"; return div.getElementsByClassName("e").length === 2; }); // Check if getElementsByName privileges form controls or returns elements by ID // If so, assume (for broader support) that getElementById returns elements by name support.getByName = assert(function( div ) { // Inject content div.id = expando + 0; // Support: Windows 8 Native Apps // Assigning innerHTML with "name" attributes throws uncatchable exceptions // http://msdn.microsoft.com/en-us/library/ie/hh465388.aspx div.appendChild( document.createElement("a") ).setAttribute( "name", expando ); div.appendChild( document.createElement("i") ).setAttribute( "name", expando ); docElem.appendChild( div ); // Test var pass = doc.getElementsByName && // buggy browsers will return fewer than the correct 2 doc.getElementsByName( expando ).length === 2 + // buggy browsers will return more than the correct 0 doc.getElementsByName( expando + 0 ).length; // Cleanup docElem.removeChild( div ); return pass; }); // Support: Webkit<537.32 // Detached nodes confoundingly follow *each other* support.sortDetached = assert(function( div1 ) { return div1.compareDocumentPosition && // Should return 1, but Webkit returns 4 (following) (div1.compareDocumentPosition( document.createElement("div") ) & 1); }); // IE6/7 return modified attributes Expr.attrHandle = assert(function( div ) { div.innerHTML = ""; return div.firstChild && typeof div.firstChild.getAttribute !== strundefined && div.firstChild.getAttribute("href") === "#"; }) ? {} : { "href": function( elem ) { return elem.getAttribute( "href", 2 ); }, "type": function( elem ) { return elem.getAttribute("type"); } }; // ID find and filter if ( support.getByName ) { Expr.find["ID"] = function( id, context ) { if ( typeof context.getElementById !== strundefined && documentIsHTML ) { var m = context.getElementById( id ); // Check parentNode to catch when Blackberry 4.6 returns // nodes that are no longer in the document #6963 return m && m.parentNode ? [m] : []; } }; Expr.filter["ID"] = function( id ) { var attrId = id.replace( runescape, funescape ); return function( elem ) { return elem.getAttribute("id") === attrId; }; }; } else { Expr.find["ID"] = function( id, context ) { if ( typeof context.getElementById !== strundefined && documentIsHTML ) { var m = context.getElementById( id ); return m ? m.id === id || typeof m.getAttributeNode !== strundefined && m.getAttributeNode("id").value === id ? [m] : undefined : []; } }; Expr.filter["ID"] = function( id ) { var attrId = id.replace( runescape, funescape ); return function( elem ) { var node = typeof elem.getAttributeNode !== strundefined && elem.getAttributeNode("id"); return node && node.value === attrId; }; }; } // Tag Expr.find["TAG"] = support.getElementsByTagName ? function( tag, context ) { if ( typeof context.getElementsByTagName !== strundefined ) { return context.getElementsByTagName( tag ); } } : function( tag, context ) { var elem, tmp = [], i = 0, results = context.getElementsByTagName( tag ); // Filter out possible comments if ( tag === "*" ) { while ( (elem = results[i++]) ) { if ( elem.nodeType === 1 ) { tmp.push( elem ); } } return tmp; } return results; }; // Name Expr.find["NAME"] = support.getByName && function( tag, context ) { if ( typeof context.getElementsByName !== strundefined ) { return context.getElementsByName( name ); } }; // Class Expr.find["CLASS"] = support.getElementsByClassName && function( className, context ) { if ( typeof context.getElementsByClassName !== strundefined && documentIsHTML ) { return context.getElementsByClassName( className ); } }; // QSA and matchesSelector support // matchesSelector(:active) reports false when true (IE9/Opera 11.5) rbuggyMatches = []; // qSa(:focus) reports false when true (Chrome 21), // no need to also add to buggyMatches since matches checks buggyQSA // A support test would require too much code (would include document ready) rbuggyQSA = [ ":focus" ]; if ( (support.qsa = isNative(doc.querySelectorAll)) ) { // Build QSA regex // Regex strategy adopted from Diego Perini assert(function( div ) { // Select is set to empty string on purpose // This is to test IE's treatment of not explicitly // setting a boolean content attribute, // since its presence should be enough // http://bugs.jquery.com/ticket/12359 div.innerHTML = ""; // IE8 - Some boolean attributes are not treated correctly if ( !div.querySelectorAll("[selected]").length ) { rbuggyQSA.push( "\\[" + whitespace + "*(?:checked|disabled|ismap|multiple|readonly|selected|value)" ); } // Webkit/Opera - :checked should return selected option elements // http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked // IE8 throws error here and will not see later tests if ( !div.querySelectorAll(":checked").length ) { rbuggyQSA.push(":checked"); } }); assert(function( div ) { // Opera 10-12/IE8 - ^= $= *= and empty values // Should not select anything div.innerHTML = ""; if ( div.querySelectorAll("[i^='']").length ) { rbuggyQSA.push( "[*^$]=" + whitespace + "*(?:\"\"|'')" ); } // FF 3.5 - :enabled/:disabled and hidden elements (hidden elements are still enabled) // IE8 throws error here and will not see later tests if ( !div.querySelectorAll(":enabled").length ) { rbuggyQSA.push( ":enabled", ":disabled" ); } // Opera 10-11 does not throw on post-comma invalid pseudos div.querySelectorAll("*,:x"); rbuggyQSA.push(",.*:"); }); } if ( (support.matchesSelector = isNative( (matches = docElem.matchesSelector || docElem.mozMatchesSelector || docElem.webkitMatchesSelector || docElem.oMatchesSelector || docElem.msMatchesSelector) )) ) { assert(function( div ) { // Check to see if it's possible to do matchesSelector // on a disconnected node (IE 9) support.disconnectedMatch = matches.call( div, "div" ); // This should fail with an exception // Gecko does not error, returns false instead matches.call( div, "[s!='']:x" ); rbuggyMatches.push( "!=", pseudos ); }); } rbuggyQSA = new RegExp( rbuggyQSA.join("|") ); rbuggyMatches = rbuggyMatches.length && new RegExp( rbuggyMatches.join("|") ); // Element contains another // Purposefully does not implement inclusive descendent // As in, an element does not contain itself contains = isNative(docElem.contains) || docElem.compareDocumentPosition ? function( a, b ) { var adown = a.nodeType === 9 ? a.documentElement : a, bup = b && b.parentNode; return a === bup || !!( bup && bup.nodeType === 1 && ( adown.contains ? adown.contains( bup ) : a.compareDocumentPosition && a.compareDocumentPosition( bup ) & 16 )); } : function( a, b ) { if ( b ) { while ( (b = b.parentNode) ) { if ( b === a ) { return true; } } } return false; }; // Document order sorting sortOrder = docElem.compareDocumentPosition ? function( a, b ) { // Flag for duplicate removal if ( a === b ) { hasDuplicate = true; return 0; } var compare = b.compareDocumentPosition && a.compareDocumentPosition && a.compareDocumentPosition( b ); if ( compare ) { // Disconnected nodes if ( compare & 1 || (recompare && b.compareDocumentPosition( a ) === compare) ) { // Choose the first element that is related to our preferred document if ( a === doc || contains(preferredDoc, a) ) { return -1; } if ( b === doc || contains(preferredDoc, b) ) { return 1; } // Maintain original order return sortInput ? ( indexOf.call( sortInput, a ) - indexOf.call( sortInput, b ) ) : 0; } return compare & 4 ? -1 : 1; } // Not directly comparable, sort on existence of method return a.compareDocumentPosition ? -1 : 1; } : function( a, b ) { var cur, i = 0, aup = a.parentNode, bup = b.parentNode, ap = [ a ], bp = [ b ]; // Exit early if the nodes are identical if ( a === b ) { hasDuplicate = true; return 0; // Parentless nodes are either documents or disconnected } else if ( !aup || !bup ) { return a === doc ? -1 : b === doc ? 1 : aup ? -1 : bup ? 1 : 0; // If the nodes are siblings, we can do a quick check } else if ( aup === bup ) { return siblingCheck( a, b ); } // Otherwise we need full lists of their ancestors for comparison cur = a; while ( (cur = cur.parentNode) ) { ap.unshift( cur ); } cur = b; while ( (cur = cur.parentNode) ) { bp.unshift( cur ); } // Walk down the tree looking for a discrepancy while ( ap[i] === bp[i] ) { i++; } return i ? // Do a sibling check if the nodes have a common ancestor siblingCheck( ap[i], bp[i] ) : // Otherwise nodes in our document sort first ap[i] === preferredDoc ? -1 : bp[i] === preferredDoc ? 1 : 0; }; return document; }; Sizzle.matches = function( expr, elements ) { return Sizzle( expr, null, null, elements ); }; Sizzle.matchesSelector = function( elem, expr ) { // Set document vars if needed if ( ( elem.ownerDocument || elem ) !== document ) { setDocument( elem ); } // Make sure that attribute selectors are quoted expr = expr.replace( rattributeQuotes, "='$1']" ); // rbuggyQSA always contains :focus, so no need for an existence check if ( support.matchesSelector && documentIsHTML && (!rbuggyMatches || !rbuggyMatches.test(expr)) && !rbuggyQSA.test(expr) ) { try { var ret = matches.call( elem, expr ); // IE 9's matchesSelector returns false on disconnected nodes if ( ret || support.disconnectedMatch || // As well, disconnected nodes are said to be in a document // fragment in IE 9 elem.document && elem.document.nodeType !== 11 ) { return ret; } } catch(e) {} } return Sizzle( expr, document, null, [elem] ).length > 0; }; Sizzle.contains = function( context, elem ) { // Set document vars if needed if ( ( context.ownerDocument || context ) !== document ) { setDocument( context ); } return contains( context, elem ); }; Sizzle.attr = function( elem, name ) { var val; // Set document vars if needed if ( ( elem.ownerDocument || elem ) !== document ) { setDocument( elem ); } if ( documentIsHTML ) { name = name.toLowerCase(); } if ( (val = Expr.attrHandle[ name ]) ) { return val( elem ); } if ( !documentIsHTML || support.attributes ) { return elem.getAttribute( name ); } return ( (val = elem.getAttributeNode( name )) || elem.getAttribute( name ) ) && elem[ name ] === true ? name : val && val.specified ? val.value : null; }; Sizzle.error = function( msg ) { throw new Error( "Syntax error, unrecognized expression: " + msg ); }; // Document sorting and removing duplicates Sizzle.uniqueSort = function( results ) { var elem, duplicates = [], j = 0, i = 0; // Unless we *know* we can detect duplicates, assume their presence hasDuplicate = !support.detectDuplicates; // Compensate for sort limitations recompare = !support.sortDetached; sortInput = !support.sortStable && results.slice( 0 ); results.sort( sortOrder ); if ( hasDuplicate ) { while ( (elem = results[i++]) ) { if ( elem === results[ i ] ) { j = duplicates.push( i ); } } while ( j-- ) { results.splice( duplicates[ j ], 1 ); } } return results; }; /* * Checks document order of two siblings * @param {Element} a * @param {Element} b * @returns Returns -1 if a precedes b, 1 if a follows b */ function siblingCheck( a, b ) { var cur = b && a, diff = cur && ( ~b.sourceIndex || MAX_NEGATIVE ) - ( ~a.sourceIndex || MAX_NEGATIVE ); // Use IE sourceIndex if available on both nodes if ( diff ) { return diff; } // Check if b follows a if ( cur ) { while ( (cur = cur.nextSibling) ) { if ( cur === b ) { return -1; } } } return a ? 1 : -1; } // Returns a function to use in pseudos for input types function createInputPseudo( type ) { return function( elem ) { var name = elem.nodeName.toLowerCase(); return name === "input" && elem.type === type; }; } // Returns a function to use in pseudos for buttons function createButtonPseudo( type ) { return function( elem ) { var name = elem.nodeName.toLowerCase(); return (name === "input" || name === "button") && elem.type === type; }; } // Returns a function to use in pseudos for positionals function createPositionalPseudo( fn ) { return markFunction(function( argument ) { argument = +argument; return markFunction(function( seed, matches ) { var j, matchIndexes = fn( [], seed.length, argument ), i = matchIndexes.length; // Match elements found at the specified indexes while ( i-- ) { if ( seed[ (j = matchIndexes[i]) ] ) { seed[j] = !(matches[j] = seed[j]); } } }); }); } /* * Utility function for retrieving the text value of an array of DOM nodes * @param {Array|Element} elem */ getText = Sizzle.getText = function( elem ) { var node, ret = "", i = 0, nodeType = elem.nodeType; if ( !nodeType ) { // If no nodeType, this is expected to be an array for ( ; (node = elem[i]); i++ ) { // Do not traverse comment nodes ret += getText( node ); } } else if ( nodeType === 1 || nodeType === 9 || nodeType === 11 ) { // Use textContent for elements // innerText usage removed for consistency of new lines (see #11153) if ( typeof elem.textContent === "string" ) { return elem.textContent; } else { // Traverse its children for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) { ret += getText( elem ); } } } else if ( nodeType === 3 || nodeType === 4 ) { return elem.nodeValue; } // Do not include comment or processing instruction nodes return ret; }; Expr = Sizzle.selectors = { // Can be adjusted by the user cacheLength: 50, createPseudo: markFunction, match: matchExpr, find: {}, relative: { ">": { dir: "parentNode", first: true }, " ": { dir: "parentNode" }, "+": { dir: "previousSibling", first: true }, "~": { dir: "previousSibling" } }, preFilter: { "ATTR": function( match ) { match[1] = match[1].replace( runescape, funescape ); // Move the given value to match[3] whether quoted or unquoted match[3] = ( match[4] || match[5] || "" ).replace( runescape, funescape ); if ( match[2] === "~=" ) { match[3] = " " + match[3] + " "; } return match.slice( 0, 4 ); }, "CHILD": function( match ) { /* matches from matchExpr["CHILD"] 1 type (only|nth|...) 2 what (child|of-type) 3 argument (even|odd|\d*|\d*n([+-]\d+)?|...) 4 xn-component of xn+y argument ([+-]?\d*n|) 5 sign of xn-component 6 x of xn-component 7 sign of y-component 8 y of y-component */ match[1] = match[1].toLowerCase(); if ( match[1].slice( 0, 3 ) === "nth" ) { // nth-* requires argument if ( !match[3] ) { Sizzle.error( match[0] ); } // numeric x and y parameters for Expr.filter.CHILD // remember that false/true cast respectively to 0/1 match[4] = +( match[4] ? match[5] + (match[6] || 1) : 2 * ( match[3] === "even" || match[3] === "odd" ) ); match[5] = +( ( match[7] + match[8] ) || match[3] === "odd" ); // other types prohibit arguments } else if ( match[3] ) { Sizzle.error( match[0] ); } return match; }, "PSEUDO": function( match ) { var excess, unquoted = !match[5] && match[2]; if ( matchExpr["CHILD"].test( match[0] ) ) { return null; } // Accept quoted arguments as-is if ( match[4] ) { match[2] = match[4]; // Strip excess characters from unquoted arguments } else if ( unquoted && rpseudo.test( unquoted ) && // Get excess from tokenize (recursively) (excess = tokenize( unquoted, true )) && // advance to the next closing parenthesis (excess = unquoted.indexOf( ")", unquoted.length - excess ) - unquoted.length) ) { // excess is a negative index match[0] = match[0].slice( 0, excess ); match[2] = unquoted.slice( 0, excess ); } // Return only captures needed by the pseudo filter method (type and argument) return match.slice( 0, 3 ); } }, filter: { "TAG": function( nodeName ) { if ( nodeName === "*" ) { return function() { return true; }; } nodeName = nodeName.replace( runescape, funescape ).toLowerCase(); return function( elem ) { return elem.nodeName && elem.nodeName.toLowerCase() === nodeName; }; }, "CLASS": function( className ) { var pattern = classCache[ className + " " ]; return pattern || (pattern = new RegExp( "(^|" + whitespace + ")" + className + "(" + whitespace + "|$)" )) && classCache( className, function( elem ) { return pattern.test( elem.className || (typeof elem.getAttribute !== strundefined && elem.getAttribute("class")) || "" ); }); }, "ATTR": function( name, operator, check ) { return function( elem ) { var result = Sizzle.attr( elem, name ); if ( result == null ) { return operator === "!="; } if ( !operator ) { return true; } result += ""; return operator === "=" ? result === check : operator === "!=" ? result !== check : operator === "^=" ? check && result.indexOf( check ) === 0 : operator === "*=" ? check && result.indexOf( check ) > -1 : operator === "$=" ? check && result.slice( -check.length ) === check : operator === "~=" ? ( " " + result + " " ).indexOf( check ) > -1 : operator === "|=" ? result === check || result.slice( 0, check.length + 1 ) === check + "-" : false; }; }, "CHILD": function( type, what, argument, first, last ) { var simple = type.slice( 0, 3 ) !== "nth", forward = type.slice( -4 ) !== "last", ofType = what === "of-type"; return first === 1 && last === 0 ? // Shortcut for :nth-*(n) function( elem ) { return !!elem.parentNode; } : function( elem, context, xml ) { var cache, outerCache, node, diff, nodeIndex, start, dir = simple !== forward ? "nextSibling" : "previousSibling", parent = elem.parentNode, name = ofType && elem.nodeName.toLowerCase(), useCache = !xml && !ofType; if ( parent ) { // :(first|last|only)-(child|of-type) if ( simple ) { while ( dir ) { node = elem; while ( (node = node[ dir ]) ) { if ( ofType ? node.nodeName.toLowerCase() === name : node.nodeType === 1 ) { return false; } } // Reverse direction for :only-* (if we haven't yet done so) start = dir = type === "only" && !start && "nextSibling"; } return true; } start = [ forward ? parent.firstChild : parent.lastChild ]; // non-xml :nth-child(...) stores cache data on `parent` if ( forward && useCache ) { // Seek `elem` from a previously-cached index outerCache = parent[ expando ] || (parent[ expando ] = {}); cache = outerCache[ type ] || []; nodeIndex = cache[0] === dirruns && cache[1]; diff = cache[0] === dirruns && cache[2]; node = nodeIndex && parent.childNodes[ nodeIndex ]; while ( (node = ++nodeIndex && node && node[ dir ] || // Fallback to seeking `elem` from the start (diff = nodeIndex = 0) || start.pop()) ) { // When found, cache indexes on `parent` and break if ( node.nodeType === 1 && ++diff && node === elem ) { outerCache[ type ] = [ dirruns, nodeIndex, diff ]; break; } } // Use previously-cached element index if available } else if ( useCache && (cache = (elem[ expando ] || (elem[ expando ] = {}))[ type ]) && cache[0] === dirruns ) { diff = cache[1]; // xml :nth-child(...) or :nth-last-child(...) or :nth(-last)?-of-type(...) } else { // Use the same loop as above to seek `elem` from the start while ( (node = ++nodeIndex && node && node[ dir ] || (diff = nodeIndex = 0) || start.pop()) ) { if ( ( ofType ? node.nodeName.toLowerCase() === name : node.nodeType === 1 ) && ++diff ) { // Cache the index of each encountered element if ( useCache ) { (node[ expando ] || (node[ expando ] = {}))[ type ] = [ dirruns, diff ]; } if ( node === elem ) { break; } } } } // Incorporate the offset, then check against cycle size diff -= last; return diff === first || ( diff % first === 0 && diff / first >= 0 ); } }; }, "PSEUDO": function( pseudo, argument ) { // pseudo-class names are case-insensitive // http://www.w3.org/TR/selectors/#pseudo-classes // Prioritize by case sensitivity in case custom pseudos are added with uppercase letters // Remember that setFilters inherits from pseudos var args, fn = Expr.pseudos[ pseudo ] || Expr.setFilters[ pseudo.toLowerCase() ] || Sizzle.error( "unsupported pseudo: " + pseudo ); // The user may use createPseudo to indicate that // arguments are needed to create the filter function // just as Sizzle does if ( fn[ expando ] ) { return fn( argument ); } // But maintain support for old signatures if ( fn.length > 1 ) { args = [ pseudo, pseudo, "", argument ]; return Expr.setFilters.hasOwnProperty( pseudo.toLowerCase() ) ? markFunction(function( seed, matches ) { var idx, matched = fn( seed, argument ), i = matched.length; while ( i-- ) { idx = indexOf.call( seed, matched[i] ); seed[ idx ] = !( matches[ idx ] = matched[i] ); } }) : function( elem ) { return fn( elem, 0, args ); }; } return fn; } }, pseudos: { // Potentially complex pseudos "not": markFunction(function( selector ) { // Trim the selector passed to compile // to avoid treating leading and trailing // spaces as combinators var input = [], results = [], matcher = compile( selector.replace( rtrim, "$1" ) ); return matcher[ expando ] ? markFunction(function( seed, matches, context, xml ) { var elem, unmatched = matcher( seed, null, xml, [] ), i = seed.length; // Match elements unmatched by `matcher` while ( i-- ) { if ( (elem = unmatched[i]) ) { seed[i] = !(matches[i] = elem); } } }) : function( elem, context, xml ) { input[0] = elem; matcher( input, null, xml, results ); return !results.pop(); }; }), "has": markFunction(function( selector ) { return function( elem ) { return Sizzle( selector, elem ).length > 0; }; }), "contains": markFunction(function( text ) { return function( elem ) { return ( elem.textContent || elem.innerText || getText( elem ) ).indexOf( text ) > -1; }; }), // "Whether an element is represented by a :lang() selector // is based solely on the element's language value // being equal to the identifier C, // or beginning with the identifier C immediately followed by "-". // The matching of C against the element's language value is performed case-insensitively. // The identifier C does not have to be a valid language name." // http://www.w3.org/TR/selectors/#lang-pseudo "lang": markFunction( function( lang ) { // lang value must be a valid identifier if ( !ridentifier.test(lang || "") ) { Sizzle.error( "unsupported lang: " + lang ); } lang = lang.replace( runescape, funescape ).toLowerCase(); return function( elem ) { var elemLang; do { if ( (elemLang = documentIsHTML ? elem.lang : elem.getAttribute("xml:lang") || elem.getAttribute("lang")) ) { elemLang = elemLang.toLowerCase(); return elemLang === lang || elemLang.indexOf( lang + "-" ) === 0; } } while ( (elem = elem.parentNode) && elem.nodeType === 1 ); return false; }; }), // Miscellaneous "target": function( elem ) { var hash = window.location && window.location.hash; return hash && hash.slice( 1 ) === elem.id; }, "root": function( elem ) { return elem === docElem; }, "focus": function( elem ) { return elem === document.activeElement && (!document.hasFocus || document.hasFocus()) && !!(elem.type || elem.href || ~elem.tabIndex); }, // Boolean properties "enabled": function( elem ) { return elem.disabled === false; }, "disabled": function( elem ) { return elem.disabled === true; }, "checked": function( elem ) { // In CSS3, :checked should return both checked and selected elements // http://www.w3.org/TR/2011/REC-css3-selectors-20110929/#checked var nodeName = elem.nodeName.toLowerCase(); return (nodeName === "input" && !!elem.checked) || (nodeName === "option" && !!elem.selected); }, "selected": function( elem ) { // Accessing this property makes selected-by-default // options in Safari work properly if ( elem.parentNode ) { elem.parentNode.selectedIndex; } return elem.selected === true; }, // Contents "empty": function( elem ) { // http://www.w3.org/TR/selectors/#empty-pseudo // :empty is only affected by element nodes and content nodes(including text(3), cdata(4)), // not comment, processing instructions, or others // Thanks to Diego Perini for the nodeName shortcut // Greater than "@" means alpha characters (specifically not starting with "#" or "?") for ( elem = elem.firstChild; elem; elem = elem.nextSibling ) { if ( elem.nodeName > "@" || elem.nodeType === 3 || elem.nodeType === 4 ) { return false; } } return true; }, "parent": function( elem ) { return !Expr.pseudos["empty"]( elem ); }, // Element/input types "header": function( elem ) { return rheader.test( elem.nodeName ); }, "input": function( elem ) { return rinputs.test( elem.nodeName ); }, "button": function( elem ) { var name = elem.nodeName.toLowerCase(); return name === "input" && elem.type === "button" || name === "button"; }, "text": function( elem ) { var attr; // IE6 and 7 will map elem.type to 'text' for new HTML5 types (search, etc) // use getAttribute instead to test this case return elem.nodeName.toLowerCase() === "input" && elem.type === "text" && ( (attr = elem.getAttribute("type")) == null || attr.toLowerCase() === elem.type ); }, // Position-in-collection "first": createPositionalPseudo(function() { return [ 0 ]; }), "last": createPositionalPseudo(function( matchIndexes, length ) { return [ length - 1 ]; }), "eq": createPositionalPseudo(function( matchIndexes, length, argument ) { return [ argument < 0 ? argument + length : argument ]; }), "even": createPositionalPseudo(function( matchIndexes, length ) { var i = 0; for ( ; i < length; i += 2 ) { matchIndexes.push( i ); } return matchIndexes; }), "odd": createPositionalPseudo(function( matchIndexes, length ) { var i = 1; for ( ; i < length; i += 2 ) { matchIndexes.push( i ); } return matchIndexes; }), "lt": createPositionalPseudo(function( matchIndexes, length, argument ) { var i = argument < 0 ? argument + length : argument; for ( ; --i >= 0; ) { matchIndexes.push( i ); } return matchIndexes; }), "gt": createPositionalPseudo(function( matchIndexes, length, argument ) { var i = argument < 0 ? argument + length : argument; for ( ; ++i < length; ) { matchIndexes.push( i ); } return matchIndexes; }) } }; // Add button/input type pseudos for ( i in { radio: true, checkbox: true, file: true, password: true, image: true } ) { Expr.pseudos[ i ] = createInputPseudo( i ); } for ( i in { submit: true, reset: true } ) { Expr.pseudos[ i ] = createButtonPseudo( i ); } function tokenize( selector, parseOnly ) { var matched, match, tokens, type, soFar, groups, preFilters, cached = tokenCache[ selector + " " ]; if ( cached ) { return parseOnly ? 0 : cached.slice( 0 ); } soFar = selector; groups = []; preFilters = Expr.preFilter; while ( soFar ) { // Comma and first run if ( !matched || (match = rcomma.exec( soFar )) ) { if ( match ) { // Don't consume trailing commas as valid soFar = soFar.slice( match[0].length ) || soFar; } groups.push( tokens = [] ); } matched = false; // Combinators if ( (match = rcombinators.exec( soFar )) ) { matched = match.shift(); tokens.push( { value: matched, // Cast descendant combinators to space type: match[0].replace( rtrim, " " ) } ); soFar = soFar.slice( matched.length ); } // Filters for ( type in Expr.filter ) { if ( (match = matchExpr[ type ].exec( soFar )) && (!preFilters[ type ] || (match = preFilters[ type ]( match ))) ) { matched = match.shift(); tokens.push( { value: matched, type: type, matches: match } ); soFar = soFar.slice( matched.length ); } } if ( !matched ) { break; } } // Return the length of the invalid excess // if we're just parsing // Otherwise, throw an error or return tokens return parseOnly ? soFar.length : soFar ? Sizzle.error( selector ) : // Cache the tokens tokenCache( selector, groups ).slice( 0 ); } function toSelector( tokens ) { var i = 0, len = tokens.length, selector = ""; for ( ; i < len; i++ ) { selector += tokens[i].value; } return selector; } function addCombinator( matcher, combinator, base ) { var dir = combinator.dir, checkNonElements = base && dir === "parentNode", doneName = done++; return combinator.first ? // Check against closest ancestor/preceding element function( elem, context, xml ) { while ( (elem = elem[ dir ]) ) { if ( elem.nodeType === 1 || checkNonElements ) { return matcher( elem, context, xml ); } } } : // Check against all ancestor/preceding elements function( elem, context, xml ) { var data, cache, outerCache, dirkey = dirruns + " " + doneName; // We can't set arbitrary data on XML nodes, so they don't benefit from dir caching if ( xml ) { while ( (elem = elem[ dir ]) ) { if ( elem.nodeType === 1 || checkNonElements ) { if ( matcher( elem, context, xml ) ) { return true; } } } } else { while ( (elem = elem[ dir ]) ) { if ( elem.nodeType === 1 || checkNonElements ) { outerCache = elem[ expando ] || (elem[ expando ] = {}); if ( (cache = outerCache[ dir ]) && cache[0] === dirkey ) { if ( (data = cache[1]) === true || data === cachedruns ) { return data === true; } } else { cache = outerCache[ dir ] = [ dirkey ]; cache[1] = matcher( elem, context, xml ) || cachedruns; if ( cache[1] === true ) { return true; } } } } } }; } function elementMatcher( matchers ) { return matchers.length > 1 ? function( elem, context, xml ) { var i = matchers.length; while ( i-- ) { if ( !matchers[i]( elem, context, xml ) ) { return false; } } return true; } : matchers[0]; } function condense( unmatched, map, filter, context, xml ) { var elem, newUnmatched = [], i = 0, len = unmatched.length, mapped = map != null; for ( ; i < len; i++ ) { if ( (elem = unmatched[i]) ) { if ( !filter || filter( elem, context, xml ) ) { newUnmatched.push( elem ); if ( mapped ) { map.push( i ); } } } } return newUnmatched; } function setMatcher( preFilter, selector, matcher, postFilter, postFinder, postSelector ) { if ( postFilter && !postFilter[ expando ] ) { postFilter = setMatcher( postFilter ); } if ( postFinder && !postFinder[ expando ] ) { postFinder = setMatcher( postFinder, postSelector ); } return markFunction(function( seed, results, context, xml ) { var temp, i, elem, preMap = [], postMap = [], preexisting = results.length, // Get initial elements from seed or context elems = seed || multipleContexts( selector || "*", context.nodeType ? [ context ] : context, [] ), // Prefilter to get matcher input, preserving a map for seed-results synchronization matcherIn = preFilter && ( seed || !selector ) ? condense( elems, preMap, preFilter, context, xml ) : elems, matcherOut = matcher ? // If we have a postFinder, or filtered seed, or non-seed postFilter or preexisting results, postFinder || ( seed ? preFilter : preexisting || postFilter ) ? // ...intermediate processing is necessary [] : // ...otherwise use results directly results : matcherIn; // Find primary matches if ( matcher ) { matcher( matcherIn, matcherOut, context, xml ); } // Apply postFilter if ( postFilter ) { temp = condense( matcherOut, postMap ); postFilter( temp, [], context, xml ); // Un-match failing elements by moving them back to matcherIn i = temp.length; while ( i-- ) { if ( (elem = temp[i]) ) { matcherOut[ postMap[i] ] = !(matcherIn[ postMap[i] ] = elem); } } } if ( seed ) { if ( postFinder || preFilter ) { if ( postFinder ) { // Get the final matcherOut by condensing this intermediate into postFinder contexts temp = []; i = matcherOut.length; while ( i-- ) { if ( (elem = matcherOut[i]) ) { // Restore matcherIn since elem is not yet a final match temp.push( (matcherIn[i] = elem) ); } } postFinder( null, (matcherOut = []), temp, xml ); } // Move matched elements from seed to results to keep them synchronized i = matcherOut.length; while ( i-- ) { if ( (elem = matcherOut[i]) && (temp = postFinder ? indexOf.call( seed, elem ) : preMap[i]) > -1 ) { seed[temp] = !(results[temp] = elem); } } } // Add elements to results, through postFinder if defined } else { matcherOut = condense( matcherOut === results ? matcherOut.splice( preexisting, matcherOut.length ) : matcherOut ); if ( postFinder ) { postFinder( null, results, matcherOut, xml ); } else { push.apply( results, matcherOut ); } } }); } function matcherFromTokens( tokens ) { var checkContext, matcher, j, len = tokens.length, leadingRelative = Expr.relative[ tokens[0].type ], implicitRelative = leadingRelative || Expr.relative[" "], i = leadingRelative ? 1 : 0, // The foundational matcher ensures that elements are reachable from top-level context(s) matchContext = addCombinator( function( elem ) { return elem === checkContext; }, implicitRelative, true ), matchAnyContext = addCombinator( function( elem ) { return indexOf.call( checkContext, elem ) > -1; }, implicitRelative, true ), matchers = [ function( elem, context, xml ) { return ( !leadingRelative && ( xml || context !== outermostContext ) ) || ( (checkContext = context).nodeType ? matchContext( elem, context, xml ) : matchAnyContext( elem, context, xml ) ); } ]; for ( ; i < len; i++ ) { if ( (matcher = Expr.relative[ tokens[i].type ]) ) { matchers = [ addCombinator(elementMatcher( matchers ), matcher) ]; } else { matcher = Expr.filter[ tokens[i].type ].apply( null, tokens[i].matches ); // Return special upon seeing a positional matcher if ( matcher[ expando ] ) { // Find the next relative operator (if any) for proper handling j = ++i; for ( ; j < len; j++ ) { if ( Expr.relative[ tokens[j].type ] ) { break; } } return setMatcher( i > 1 && elementMatcher( matchers ), i > 1 && toSelector( tokens.slice( 0, i - 1 ) ).replace( rtrim, "$1" ), matcher, i < j && matcherFromTokens( tokens.slice( i, j ) ), j < len && matcherFromTokens( (tokens = tokens.slice( j )) ), j < len && toSelector( tokens ) ); } matchers.push( matcher ); } } return elementMatcher( matchers ); } function matcherFromGroupMatchers( elementMatchers, setMatchers ) { // A counter to specify which element is currently being matched var matcherCachedRuns = 0, bySet = setMatchers.length > 0, byElement = elementMatchers.length > 0, superMatcher = function( seed, context, xml, results, expandContext ) { var elem, j, matcher, setMatched = [], matchedCount = 0, i = "0", unmatched = seed && [], outermost = expandContext != null, contextBackup = outermostContext, // We must always have either seed elements or context elems = seed || byElement && Expr.find["TAG"]( "*", expandContext && context.parentNode || context ), // Use integer dirruns iff this is the outermost matcher dirrunsUnique = (dirruns += contextBackup == null ? 1 : Math.random() || 0.1); if ( outermost ) { outermostContext = context !== document && context; cachedruns = matcherCachedRuns; } // Add elements passing elementMatchers directly to results // Keep `i` a string if there are no elements so `matchedCount` will be "00" below for ( ; (elem = elems[i]) != null; i++ ) { if ( byElement && elem ) { j = 0; while ( (matcher = elementMatchers[j++]) ) { if ( matcher( elem, context, xml ) ) { results.push( elem ); break; } } if ( outermost ) { dirruns = dirrunsUnique; cachedruns = ++matcherCachedRuns; } } // Track unmatched elements for set filters if ( bySet ) { // They will have gone through all possible matchers if ( (elem = !matcher && elem) ) { matchedCount--; } // Lengthen the array for every element, matched or not if ( seed ) { unmatched.push( elem ); } } } // Apply set filters to unmatched elements matchedCount += i; if ( bySet && i !== matchedCount ) { j = 0; while ( (matcher = setMatchers[j++]) ) { matcher( unmatched, setMatched, context, xml ); } if ( seed ) { // Reintegrate element matches to eliminate the need for sorting if ( matchedCount > 0 ) { while ( i-- ) { if ( !(unmatched[i] || setMatched[i]) ) { setMatched[i] = pop.call( results ); } } } // Discard index placeholder values to get only actual matches setMatched = condense( setMatched ); } // Add matches to results push.apply( results, setMatched ); // Seedless set matches succeeding multiple successful matchers stipulate sorting if ( outermost && !seed && setMatched.length > 0 && ( matchedCount + setMatchers.length ) > 1 ) { Sizzle.uniqueSort( results ); } } // Override manipulation of globals by nested matchers if ( outermost ) { dirruns = dirrunsUnique; outermostContext = contextBackup; } return unmatched; }; return bySet ? markFunction( superMatcher ) : superMatcher; } compile = Sizzle.compile = function( selector, group /* Internal Use Only */ ) { var i, setMatchers = [], elementMatchers = [], cached = compilerCache[ selector + " " ]; if ( !cached ) { // Generate a function of recursive functions that can be used to check each element if ( !group ) { group = tokenize( selector ); } i = group.length; while ( i-- ) { cached = matcherFromTokens( group[i] ); if ( cached[ expando ] ) { setMatchers.push( cached ); } else { elementMatchers.push( cached ); } } // Cache the compiled function cached = compilerCache( selector, matcherFromGroupMatchers( elementMatchers, setMatchers ) ); } return cached; }; function multipleContexts( selector, contexts, results ) { var i = 0, len = contexts.length; for ( ; i < len; i++ ) { Sizzle( selector, contexts[i], results ); } return results; } function select( selector, context, results, seed ) { var i, tokens, token, type, find, match = tokenize( selector ); if ( !seed ) { // Try to minimize operations if there is only one group if ( match.length === 1 ) { // Take a shortcut and set the context if the root selector is an ID tokens = match[0] = match[0].slice( 0 ); if ( tokens.length > 2 && (token = tokens[0]).type === "ID" && context.nodeType === 9 && documentIsHTML && Expr.relative[ tokens[1].type ] ) { context = ( Expr.find["ID"]( token.matches[0].replace(runescape, funescape), context ) || [] )[0]; if ( !context ) { return results; } selector = selector.slice( tokens.shift().value.length ); } // Fetch a seed set for right-to-left matching i = matchExpr["needsContext"].test( selector ) ? 0 : tokens.length; while ( i-- ) { token = tokens[i]; // Abort if we hit a combinator if ( Expr.relative[ (type = token.type) ] ) { break; } if ( (find = Expr.find[ type ]) ) { // Search, expanding context for leading sibling combinators if ( (seed = find( token.matches[0].replace( runescape, funescape ), rsibling.test( tokens[0].type ) && context.parentNode || context )) ) { // If seed is empty or no tokens remain, we can return early tokens.splice( i, 1 ); selector = seed.length && toSelector( tokens ); if ( !selector ) { push.apply( results, seed ); return results; } break; } } } } } // Compile and execute a filtering function // Provide `match` to avoid retokenization if we modified the selector above compile( selector, match )( seed, context, !documentIsHTML, results, rsibling.test( selector ) ); return results; } // Deprecated Expr.pseudos["nth"] = Expr.pseudos["eq"]; // Easy API for creating new setFilters function setFilters() {} setFilters.prototype = Expr.filters = Expr.pseudos; Expr.setFilters = new setFilters(); // Check sort stability support.sortStable = expando.split("").sort( sortOrder ).join("") === expando; // Initialize with the default document setDocument(); // Always assume the presence of duplicates if sort doesn't // pass them to our comparison function (as in Google Chrome). [0, 0].sort( sortOrder ); support.detectDuplicates = hasDuplicate; // EXPOSE /*if ( typeof define === "function" && define.amd ) { define(function() { return Sizzle; }); } else { window.Sizzle = Sizzle; }*/ // EXPOSE /** * @namespace Ink.Dom.Selector * @static */ /** * Alias for the Sizzle selector engine * * @method select * @param {String} selector CSS selector to search for elements * @param {DOMElement} [context] By default the search is done in the document element. However, you can specify an element as search context * @param {Array} [results] By default this is considered an empty array. But if you want to merge it with other searches you did, pass their result array through here. * @return {Array} Array of resulting DOM Elements * @sample Ink_Dom_Selector_select.html */ /** * Filters elements that match a CSS selector. * * @method matches * @param {String} selector CSS selector to search for elements * @param {Array} matches Elements to be 'matched' with * @return {Array} Elements that matched * @sample Ink_Dom_Selector_matches.html */ /** * Checks if an element matches a given selector * * @method matchesSelector * @param {DOMElement} element Element to test * @param {String} selector CSS selector to test the element with * @return {Boolean} True if element matches the CSS selector * @sample Ink_Dom_Selector_matchesSelector.html */ return { select: Sizzle, matches: Sizzle.matches, matchesSelector: Sizzle.matchesSelector }; }); //( window ); /** * Array Utilities * @module Ink.Util.Array_1 * @version 1 */ Ink.createModule('Ink.Util.Array', '1', [], function() { 'use strict'; var arrayProto = Array.prototype; /** * @namespace Ink.Util.Array_1 */ var InkArray = { /** * Checks if a value is an array * * @method isArray * @param testedObject {Mixed} The object we want to check **/ isArray: Array.isArray || function (testedObject) { return {}.toString.call(testedObject) === '[object Array]'; }, /** * Loops through an array, grouping similar items together. * @method groupBy * @param arr {Array} The input array. * @param [options] {Object} options object, containing: * @param [options.key] {Function} A function which computes the group key by which the items are grouped. * @param [options.pairs] {Boolean} Set to `true` if you want to output an array of `[key, [group...]]` pairs instead of an array of groups. * @return {Array} An array of arrays of chunks. * * @example * * InkArray.groupBy([1, 1, 2, 2, 3, 1]) // -> [ [1, 1], [2, 2], [3], [1] ] * InkArray.groupBy([1.1, 1.2, 2.1], { key: Math.floor }) // -> [ [1.1, 1.2], [2.1] ] * InkArray.groupBy([1.1, 1.2, 2.1], { key: Math.floor, pairs: true }) // -> [ [1, [1.1, 1.2]], [2, [2.1]] ] * **/ groupBy: function (arr, options) { options = options || {}; var ret = []; var latestGroup; function eq(a, b) { return outKey(a) === outKey(b); } function outKey(item) { if (typeof options.key === 'function') { return options.key(item); } else { return item; } } for (var i = 0, len = arr.length; i < len; i++) { latestGroup = [arr[i]]; // Chunkin' while ((i + 1 < len) && eq(arr[i], arr[i + 1])) { latestGroup.push(arr[i + 1]); i++; } if (options.pairs) { ret.push([outKey(arr[i]), latestGroup]); } else { ret.push(latestGroup); } } return ret; }, /** * Replacement for Array.prototype.reduce. * * Produces a single result from a list of values by calling an "aggregator" function. * * Falls back to Array.prototype.reduce if available. * * @method reduce * @param array {Array} Input array to be reduced. * @param callback {Function} `function (previousValue, currentValue, index, all) { return {Mixed} }` to execute for each value. * @param initial {Mixed} Object used as the first argument to the first call of `callback` * * @example * var sum = InkArray.reduce([1, 2, 3], function (a, b) { return a + b; }); // -> 6 */ reduce: function (array, callback, initial) { if (arrayProto.reduce) { return arrayProto.reduce.apply(array, [].slice.call(arguments, 1)); } // From https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/Reduce#Polyfill var t = Object( array ), len = t.length >>> 0, k = 0, value; if ( arguments.length >= 3 ) { value = initial; } else { while ( k < len && !(k in t) ) k++; if ( k >= len ) throw new TypeError('Reduce of empty array with no initial value'); value = t[ k++ ]; } for ( ; k < len ; k++ ) { if ( k in t ) { value = callback( value, t[k], k, t ); } } return value; }, /** * Checks if a value exists in array * * @method inArray * @public * @static * @param {Mixed} value Value to check * @param {Array} arr Array to search in * @return {Boolean} True if value exists in the array * @sample Ink_Util_Array_inArray.html */ inArray: function(value, arr) { if (typeof arr === 'object') { for (var i = 0, f = arr.length; i < f; ++i) { if (arr[i] === value) { return true; } } } return false; }, /** * Sorts an array of objects by an object property * * @method sortMulti * @param {Array} arr Array of objects to sort * @param {String} key Property to sort by * @return {Array|Boolean} False if it's not an array, returns a sorted array if it's an array. * @public * @static * @sample Ink_Util_Array_sortMulti.html */ sortMulti: function(arr, key) { if (typeof arr === 'undefined' || arr.constructor !== Array) { return false; } if (typeof key !== 'string') { return arr.sort(); } if (arr.length > 0) { if (typeof(arr[0][key]) === 'undefined') { return false; } arr.sort(function(a, b){ var x = a[key]; var y = b[key]; return ((x < y) ? -1 : ((x > y) ? 1 : 0)); }); } return arr; }, /** * Gets the indexes of a value in an array * * @method keyValue * @param {String} value Value to search for. * @param {Array} arr Array to run the search in. * @param {Boolean} [first] Flag to stop the search at the first match. It also returns an index number instead of an array of indexes. * @return {Boolean|Number|Array} False for no matches. Array of matches or first match index. * @public * @static * @sample Ink_Util_Array_keyValue.html */ keyValue: function(value, arr, first) { if (typeof value !== 'undefined' && typeof arr === 'object' && this.inArray(value, arr)) { var aKeys = []; for (var i = 0, f = arr.length; i < f; ++i) { if (arr[i] === value) { if (typeof first !== 'undefined' && first === true) { return i; } else { aKeys.push(i); } } } return aKeys; } return false; }, /** * Shuffles an array. * * @method shuffle * @param {Array} arr Array to shuffle * @return {Array|Boolean} Shuffled Array or false if not an array. * @public * @static * @sample Ink_Util_Array_shuffle.html */ shuffle: function(arr) { if (typeof(arr) !== 'undefined' && arr.constructor !== Array) { return false; } var total = arr.length, tmp1 = false, rnd = false; while (total--) { rnd = Math.floor(Math.random() * (total + 1)); tmp1 = arr[total]; arr[total] = arr[rnd]; arr[rnd] = tmp1; } return arr; }, /** * Runs a function through each of the elements of an array * * @method forEach * @param {Array} arr The array to be cycled/iterated * @param {Function} cb The function receives as arguments the value, index and array. * @return {Array} Iterated array. * @public * @static * @sample Ink_Util_Array_forEach.html */ forEach: function(array, callback, context) { if (arrayProto.forEach) { return arrayProto.forEach.call(array, callback, context); } for (var i = 0, len = array.length >>> 0; i < len; i++) { callback.call(context, array[i], i, array); } }, /** * Alias for backwards compatibility. See forEach * * @method each */ each: function () { InkArray.forEach.apply(InkArray, [].slice.call(arguments)); }, /** * Runs a function for each item in the array. * That function will receive each item as an argument and its return value will change the corresponding array item. * @method map * @param {Array} array The array to map over * @param {Function} map The map function. Will take `(item, index, array)` as arguments and `this` will be the `context` argument. * @param {Object} [context] Object to be `this` in the map function. * * @sample Ink_Util_Array_map.html */ map: function (array, callback, context) { if (arrayProto.map) { return arrayProto.map.call(array, callback, context); } var mapped = new Array(len); for (var i = 0, len = array.length >>> 0; i < len; i++) { mapped[i] = callback.call(context, array[i], i, array); } return mapped; }, /** * Filters an array based on a truth test. * This method runs a test function on all the array values and returns a new array with all the values that pass the test. * @method filter * @param {Array} array The array to filter * @param {Function} test A test function taking `(item, index, array)` * @param {Object} [context] Object to be `this` in the test function. * @return {Array} Returns the filtered array * * @sample Ink_Util_Array_filter.html */ filter: function (array, test, context) { if (arrayProto.filter) { return arrayProto.filter.call(array, test, context); } var filtered = [], val = null; for (var i = 0, len = array.length; i < len; i++) { val = array[i]; // it might be mutated if (test.call(context, val, i, array)) { filtered.push(val); } } return filtered; }, /** * Checks if some element in the array passes a truth test * * @method some * @param {Array} arr The array to iterate through * @param {Function} cb The callback to be called on the array's elements. It receives the value, the index and the array as arguments. * @param {Object} context Object of the callback function * @return {Boolean} True if the callback returns true at any point, false otherwise * @public * @static * @sample Ink_Util_Array_some.html */ some: function(arr, cb, context){ if (arr === null){ throw new TypeError('First argument is invalid.'); } var t = Object(arr); var len = t.length >>> 0; if (typeof cb !== "function"){ throw new TypeError('Second argument must be a function.'); } for (var i = 0; i < len; i++) { if (i in t && cb.call(context, t[i], i, t)){ return true; } } return false; }, /** * Compares the values of two arrays and return the matches * * @method intersect * @param {Array} arr1 First array * @param {Array} arr2 Second array * @return {Array} Empty array if one of the arrays is false (or do not intersect) | Array with the intersected values * @public * @static * @sample Ink_Util_Array_intersect.html */ intersect: function(arr1, arr2) { if (!arr1 || !arr2 || arr1 instanceof Array === false || arr2 instanceof Array === false) { return []; } var shared = []; for (var i = 0, I = arr1.length; i 0) { for (x = a; x < b; x += step) { r.push(x); } } else { for (x = a; x > b; x += step) { r.push(x); } } return r; }, /** * Inserts a value on a specified index * * @method insert * @param {Array} arr Array where the value will be inserted * @param {Number} idx Index of the array where the value should be inserted * @param {Mixed} value Value to be inserted * @public * @static * @sample Ink_Util_Array_insert.html */ insert: function(arr, idx, value) { arr.splice(idx, 0, value); }, /** * Removes a range of values from the array * * @method remove * @param {Array} arr Array where the value will be removed * @param {Number} from Index of the array where the removal will start removing. * @param {Number} rLen Number of items to be removed from the index onwards. * @return {Array} An array with the remaining values * @public * @static * @sample Ink_Util_Array_remove.html */ remove: function(arr, from, rLen){ var output = []; for(var i = 0, iLen = arr.length; i < iLen; i++){ if(i >= from && i < from + rLen){ continue; } output.push(arr[i]); } return output; } }; return InkArray; }); /** * Binary Packing algorithm implementation * @module Ink.Util.BinPack_1 * @version 1 */ Ink.createModule('Ink.Util.BinPack', '1', [], function() { 'use strict'; /*jshint boss:true */ // https://github.com/jakesgordon/bin-packing/ /* Copyright (c) 2011, 2012, 2013 Jake Gordon and contributors Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ var Packer = function(w, h) { this.init(w, h); }; Packer.prototype = { init: function(w, h) { this.root = { x: 0, y: 0, w: w, h: h }; }, fit: function(blocks) { var n, node, block; for (n = 0; n < blocks.length; ++n) { block = blocks[n]; if (node = this.findNode(this.root, block.w, block.h)) { block.fit = this.splitNode(node, block.w, block.h); } } }, findNode: function(root, w, h) { if (root.used) { return this.findNode(root.right, w, h) || this.findNode(root.down, w, h); } else if ((w <= root.w) && (h <= root.h)) { return root; } else { return null; } }, splitNode: function(node, w, h) { node.used = true; node.down = { x: node.x, y: node.y + h, w: node.w, h: node.h - h }; node.right = { x: node.x + w, y: node.y, w: node.w - w, h: h }; return node; } }; var GrowingPacker = function() {}; GrowingPacker.prototype = { fit: function(blocks) { var n, node, block, len = blocks.length; var w = len > 0 ? blocks[0].w : 0; var h = len > 0 ? blocks[0].h : 0; this.root = { x: 0, y: 0, w: w, h: h }; for (n = 0; n < len ; n++) { block = blocks[n]; if (node = this.findNode(this.root, block.w, block.h)) { block.fit = this.splitNode(node, block.w, block.h); } else { block.fit = this.growNode(block.w, block.h); } } }, findNode: function(root, w, h) { if (root.used) { return this.findNode(root.right, w, h) || this.findNode(root.down, w, h); } else if ((w <= root.w) && (h <= root.h)) { return root; } else { return null; } }, splitNode: function(node, w, h) { node.used = true; node.down = { x: node.x, y: node.y + h, w: node.w, h: node.h - h }; node.right = { x: node.x + w, y: node.y, w: node.w - w, h: h }; return node; }, growNode: function(w, h) { var canGrowDown = (w <= this.root.w); var canGrowRight = (h <= this.root.h); var shouldGrowRight = canGrowRight && (this.root.h >= (this.root.w + w)); // attempt to keep square-ish by growing right when height is much greater than width var shouldGrowDown = canGrowDown && (this.root.w >= (this.root.h + h)); // attempt to keep square-ish by growing down when width is much greater than height if (shouldGrowRight) { return this.growRight(w, h); } else if (shouldGrowDown) { return this.growDown(w, h); } else if (canGrowRight) { return this.growRight(w, h); } else if (canGrowDown) { return this.growDown(w, h); } else { return null; // need to ensure sensible root starting size to avoid this happening } }, growRight: function(w, h) { this.root = { used: true, x: 0, y: 0, w: this.root.w + w, h: this.root.h, down: this.root, right: { x: this.root.w, y: 0, w: w, h: this.root.h } }; var node; if (node = this.findNode(this.root, w, h)) { return this.splitNode(node, w, h); } else { return null; } }, growDown: function(w, h) { this.root = { used: true, x: 0, y: 0, w: this.root.w, h: this.root.h + h, down: { x: 0, y: this.root.h, w: this.root.w, h: h }, right: this.root }; var node; if (node = this.findNode(this.root, w, h)) { return this.splitNode(node, w, h); } else { return null; } } }; var sorts = { random: function() { return Math.random() - 0.5; }, w: function(a, b) { return b.w - a.w; }, h: function(a, b) { return b.h - a.h; }, a: function(a, b) { return b.area - a.area; }, max: function(a, b) { return Math.max(b.w, b.h) - Math.max(a.w, a.h); }, min: function(a, b) { return Math.min(b.w, b.h) - Math.min(a.w, a.h); }, height: function(a, b) { return sorts.msort(a, b, ['h', 'w']); }, width: function(a, b) { return sorts.msort(a, b, ['w', 'h']); }, area: function(a, b) { return sorts.msort(a, b, ['a', 'h', 'w']); }, maxside: function(a, b) { return sorts.msort(a, b, ['max', 'min', 'h', 'w']); }, msort: function(a, b, criteria) { /* sort by multiple criteria */ var diff, n; for (n = 0; n < criteria.length; ++n) { diff = sorts[ criteria[n] ](a, b); if (diff !== 0) { return diff; } } return 0; } }; // end of Jake's code // aux, used to display blocks in unfitted property var toString = function() { return [this.w, ' x ', this.h].join(''); }; /** * Binary Packing algorithm implementation * * Based on the work of Jake Gordon * * see https://github.com/jakesgordon/bin-packing/ * * @namespace Ink.Util.BinPack * @version 1 * @static */ var BinPack = { /** * @method binPack * @param {Object} o Options * @param {Array} o.blocks Array of items with width and height integer attributes. * @param {Array} [o.dimensions] Flag to fix container dimensions * @param {String} [o.sorter] Sorting function. One of: random, height, width, area, maxside * @return {Object} Returns an object containing container dimensions, filled ratio, fitted blocks, unfitted blocks and all blocks * @static */ binPack: function(o) { var i, f, bl; // calculate area if not there already for (i = 0, f = o.blocks.length; i < f; ++i) { bl = o.blocks[i]; if (! ('area' in bl) ) { bl.area = bl.w * bl.h; } } // apply algorithm var packer = o.dimensions ? new Packer(o.dimensions[0], o.dimensions[1]) : new GrowingPacker(); if (!o.sorter) { o.sorter = 'maxside'; } o.blocks.sort( sorts[ o.sorter ] ); packer.fit(o.blocks); var dims2 = [packer.root.w, packer.root.h]; // layout is done here, generating report data... var fitted = []; var unfitted = []; for (i = 0, f = o.blocks.length; i < f; ++i) { bl = o.blocks[i]; if (bl.fit) { fitted.push(bl); } else { bl.toString = toString; // TO AID SERIALIZATION unfitted.push(bl); } } var area = dims2[0] * dims2[1]; var fit = 0; for (i = 0, f = fitted.length; i < f; ++i) { bl = fitted[i]; fit += bl.area; } return { dimensions: dims2, filled: fit / area, blocks: o.blocks, fitted: fitted, unfitted: unfitted }; } }; return BinPack; }); /** * Cookie Utilities * @module Ink.Util.Cookie_1 * @version 1 */ Ink.createModule('Ink.Util.Cookie', '1', [], function() { 'use strict'; /** * @namespace Ink.Util.Cookie_1 */ var Cookie = { /** * Gets an object with the current page cookies. * * @method get * @param {String} name The cookie name. * @return {String|Object} If the name is specified, it returns the value of that key. Otherwise it returns the full cookie object * @public * @static * @sample Ink_Util_Cookie_get.html */ get: function(name) { var cookie = document.cookie || false; var _Cookie = {}; if(cookie) { cookie = cookie.replace(new RegExp("; ", "g"), ';'); var aCookie = cookie.split(';'); var aItem = []; if(aCookie.length > 0) { for(var i=0; i < aCookie.length; i++) { aItem = aCookie[i].split('='); if(aItem.length === 2) { _Cookie[aItem[0]] = decodeURIComponent(aItem[1]); } aItem = []; } } } if(name) { if(typeof(_Cookie[name]) !== 'undefined') { return _Cookie[name]; } else { return null; } } return _Cookie; }, /** * Sets a cookie. * * @method set * @param {String} name Cookie name. * @param {String} value Cookie value. * @param {Number} [expires] Number of seconds the cookie will be valid for. * @param {String} [path] Path for the cookie. Defaults to '/'. * @param {String} [domain] Domain for the cookie. Defaults to current hostname. * @param {Boolean} [secure] Flag for secure. Default 'false'. * @public * @static * @sample Ink_Util_Cookie_set.html */ set: function(name, value, expires, path, domain, secure) { var sName; if(!name || value===false || typeof(name) === 'undefined' || typeof(value) === 'undefined') { return false; } else { sName = name+'='+encodeURIComponent(value); } var sExpires = false; var sPath = false; var sDomain = false; var sSecure = false; if(expires && typeof(expires) !== 'undefined' && !isNaN(expires)) { var oDate = new Date(); var sDate = (parseInt(Number(oDate.valueOf()), 10) + (Number(parseInt(expires, 10)) * 1000)); var nDate = new Date(sDate); var expiresString = nDate.toGMTString(); var re = new RegExp("([^\\s]+)(\\s\\d\\d)\\s(\\w\\w\\w)\\s(.*)"); expiresString = expiresString.replace(re, "$1$2-$3-$4"); sExpires = 'expires='+expiresString; } else { if(typeof(expires) !== 'undefined' && !isNaN(expires) && Number(parseInt(expires, 10))===0) { sExpires = ''; } else { sExpires = 'expires=Thu, 01-Jan-2037 00:00:01 GMT'; } } if(path && typeof(path) !== 'undefined') { sPath = 'path='+path; } else { sPath = 'path=/'; } if(domain && typeof(domain) !== 'undefined') { sDomain = 'domain='+domain; } else { var portClean = new RegExp(":(.*)"); sDomain = 'domain='+window.location.host; sDomain = sDomain.replace(portClean,""); } if(secure && typeof(secure) !== 'undefined') { sSecure = secure; } else { sSecure = false; } document.cookie = sName+'; '+sExpires+'; '+sPath+'; '+sDomain+'; '+sSecure; }, /** * Deletes a cookie. * * @method remove * @param {String} cookieName Cookie name. * @param {String} [path] Path of the cookie. Defaults to '/'. * @param {String} [domain] Domain of the cookie. Defaults to current hostname. * @public * @static * @sample Ink_Util_Cookie_remove.html */ remove: function(cookieName, path, domain) { //var expiresDate = 'Thu, 01-Jan-1970 00:00:01 GMT'; var sPath = false; var sDomain = false; var expiresDate = -999999999; if(path && typeof(path) !== 'undefined') { sPath = path; } else { sPath = '/'; } if(domain && typeof(domain) !== 'undefined') { sDomain = domain; } else { sDomain = window.location.host; } this.set(cookieName, 'deleted', expiresDate, sPath, sDomain); } }; return Cookie; }); /** * Date utility functions * @module Ink.Util.Date_1 * @version 1 */ Ink.createModule('Ink.Util.Date', '1', [], function() { 'use strict'; /** * @namespace Ink.Util.Date_1 */ var InkDate = { /** * Function that returns the string representation of the month [PT only] * * @method _months * @param {Number} index Month javascript (0 to 11) * @return {String} The month's name * @private * @static * @example * console.log( InkDate._months(0) ); // Result: Janeiro */ _months: function(index){ var _m = ['Janeiro', 'Fevereiro', 'Março', 'Abril', 'Maio', 'Junho', 'Julho', 'Agosto', 'Setembro', 'Outubro', 'Novembro', 'Dezembro']; return _m[index]; }, /** * Function that returns the month [PT only] ( 0 to 11 ) * * @method _iMonth * @param {String} month Month javascript (0 to 11) * @return {Number} The month's number * @private * @static * @example * console.log( InkDate._iMonth('maio') ); // Result: 4 */ _iMonth : function( month ) { if ( Number( month ) ) { return +month - 1; } return { 'janeiro' : 0 , 'jan' : 0 , 'fevereiro' : 1 , 'fev' : 1 , 'março' : 2 , 'mar' : 2 , 'abril' : 3 , 'abr' : 3 , 'maio' : 4 , 'mai' : 4 , 'junho' : 5 , 'jun' : 5 , 'julho' : 6 , 'jul' : 6 , 'agosto' : 7 , 'ago' : 7 , 'setembro' : 8 , 'set' : 8 , 'outubro' : 9 , 'out' : 9 , 'novembro' : 10 , 'nov' : 10 , 'dezembro' : 11 , 'dez' : 11 }[ month.toLowerCase( ) ]; } , /** * Function that returns the representation the day of the week [PT Only] * * @method _wDays * @param {Number} index Week's day index * @return {String} The week's day name * @private * @static * @example * console.log( InkDate._wDays(0) ); // Result: Domingo */ _wDays: function(index){ var _d = ['Domingo', 'Segunda', 'Terça', 'Quarta', 'Quinta', 'Sexta', 'Sábado']; return _d[index]; }, /** * Function that returns day of the week in javascript 1 to 7 * * @method _iWeek * @param {String} week Week's day name * @return {Number} The week's day index * @private * @static * @example * console.log( InkDate._iWeek('quarta') ); // Result: 3 */ _iWeek: function( week ) { if ( Number( week ) ) { return +week || 7; } return { 'segunda' : 1 , 'seg' : 1 , 'terça' : 2 , 'ter' : 2 , 'quarta' : 3 , 'qua' : 3 , 'quinta' : 4 , 'qui' : 4 , 'sexta' : 5 , 'sex' : 5 , 'sábado' : 6 , 'sáb' : 6 , 'domingo' : 7 , 'dom' : 7 }[ week.toLowerCase( ) ]; }, /** * Function that returns the number of days of a given month (m) on a given year (y) * * @method _daysInMonth * @param {Number} _m Month * @param {Number} _y Year * @return {Number} Number of days of a give month on a given year * @private * @static * @example * console.log( InkDate._daysInMonth(2,2013) ); // Result: 28 */ _daysInMonth: function(_m,_y){ var nDays; if(_m===1 || _m===3 || _m===5 || _m===7 || _m===8 || _m===10 || _m===12) { nDays= 31; } else if ( _m===4 || _m===6 || _m===9 || _m===11) { nDays = 30; } else { if((_y%400===0) || (_y%4===0 && _y%100!==0)) { nDays = 29; } else { nDays = 28; } } return nDays; }, /** * Formats a date object. * This works exactly as php date() function. http://php.net/manual/en/function.date.php * * @method get * @param {String} format The format in which the date it will be formatted. * @param {Date} [_date] The date to format. Can receive unix timestamp or a date object. Defaults to current time. * @return {String} Formatted date * @public * @static * @sample Ink_Util_Date_get.html */ get: function(format, _date){ /*jshint maxcomplexity:65 */ if(typeof(format) === 'undefined' || format === ''){ format = "Y-m-d"; } var iFormat = format.split(""); var result = new Array(iFormat.length); var escapeChar = "\\"; var jsDate; if (typeof(_date) === 'undefined'){ jsDate = new Date(); } else if (typeof(_date)==='number'){ jsDate = new Date(_date*1000); } else { jsDate = new Date(_date); } var jsFirstDay, jsThisDay, jsHour; /* This switch is presented in the same order as in php date function (PHP 5.2.2) */ for (var i = 0; i < iFormat.length; i++) { switch(iFormat[i]) { case escapeChar: result[i] = iFormat[i+1]; i++; break; /* DAY */ case "d": /* Day of the month, 2 digits with leading zeros; ex: 01 to 31 */ var jsDay = jsDate.getDate(); result[i] = (String(jsDay).length > 1) ? jsDay : "0" + jsDay; break; case "D": /* A textual representation of a day, three letters; Seg to Dom */ result[i] = this._wDays(jsDate.getDay()).substring(0, 3); break; case "j": /* Day of the month without leading zeros; ex: 1 to 31 */ result[i] = jsDate.getDate(); break; case "l": /* A full textual representation of the day of the week; Domingo to Sabado */ result[i] = this._wDays(jsDate.getDay()); break; case "N": /* ISO-8601 numeric representation of the day of the week; 1 (Segunda) to 7 (Domingo) */ result[i] = jsDate.getDay() || 7; break; case "S": /* English ordinal suffix for the day of the month, 2 characters; st, nd, rd or th. Works well with j */ var temp = jsDate.getDate(); var suffixes = ["st", "nd", "rd"]; var suffix = ""; if (temp >= 11 && temp <= 13) { result[i] = "th"; } else { result[i] = (suffix = suffixes[String(temp).substr(-1) - 1]) ? (suffix) : ("th"); } break; case "w": /* Numeric representation of the day of the week; 0 (for Sunday) through 6 (for Saturday) */ result[i] = jsDate.getDay(); break; case "z": /* The day of the year (starting from 0); 0 to 365 */ jsFirstDay = Date.UTC(jsDate.getFullYear(), 0, 0); jsThisDay = Date.UTC(jsDate.getFullYear(), jsDate.getMonth(), jsDate.getDate()); result[i] = Math.floor((jsThisDay - jsFirstDay) / (1000 * 60 * 60 * 24)); break; /* WEEK */ case "W": /* ISO-8601 week number of year, weeks starting on Monday; ex: 42 (the 42nd week in the year) */ var jsYearStart = new Date( jsDate.getFullYear( ) , 0 , 1 ); jsFirstDay = jsYearStart.getDay() || 7; var days = Math.floor( ( jsDate - jsYearStart ) / ( 24 * 60 * 60 * 1000 ) + 1 ); result[ i ] = Math.ceil( ( days - ( 8 - jsFirstDay ) ) / 7 ) + 1; break; /* MONTH */ case "F": /* A full textual representation of a month, such as Janeiro or Marco; Janeiro a Dezembro */ result[i] = this._months(jsDate.getMonth()); break; case "m": /* Numeric representation of a month, with leading zeros; 01 to 12 */ var jsMonth = String(jsDate.getMonth() + 1); result[i] = (jsMonth.length > 1) ? jsMonth : "0" + jsMonth; break; case "M": /* A short textual representation of a month, three letters; Jan a Dez */ result[i] = this._months(jsDate.getMonth()).substring(0,3); break; case "n": /* Numeric representation of a month, without leading zeros; 1 a 12 */ result[i] = jsDate.getMonth() + 1; break; case "t": /* Number of days in the given month; ex: 28 */ result[i] = this._daysInMonth(jsDate.getMonth()+1,jsDate.getYear()); break; /* YEAR */ case "L": /* Whether it's a leap year; 1 if it is a leap year, 0 otherwise. */ var jsYear = jsDate.getFullYear(); result[i] = (jsYear % 4) ? false : ( (jsYear % 100) ? true : ( (jsYear % 400) ? false : true ) ); break; case "o": /* ISO-8601 year number. This has the same value as Y, except that if the ISO week number (W) belongs to the previous or next year, that year is used instead. */ throw '"o" not implemented!'; case "Y": /* A full numeric representation of a year, 4 digits; 1999 */ result[i] = jsDate.getFullYear(); break; case "y": /* A two digit representation of a year; 99 */ result[i] = String(jsDate.getFullYear()).substring(2); break; /* TIME */ case "a": /* Lowercase Ante meridiem and Post meridiem; am or pm */ result[i] = (jsDate.getHours() < 12) ? "am" : "pm"; break; case "A": /* Uppercase Ante meridiem and Post meridiem; AM or PM */ result[i] = (jsDate.getHours < 12) ? "AM" : "PM"; break; case "B": /* Swatch Internet time; 000 through 999 */ throw '"B" not implemented!'; case "g": /* 12-hour format of an hour without leading zeros; 1 to 12 */ jsHour = jsDate.getHours(); result[i] = (jsHour <= 12) ? jsHour : (jsHour - 12); break; case "G": /* 24-hour format of an hour without leading zeros; 1 to 23 */ result[i] = String(jsDate.getHours()); break; case "h": /* 12-hour format of an hour with leading zeros; 01 to 12 */ jsHour = String(jsDate.getHours()); jsHour = (jsHour <= 12) ? jsHour : (jsHour - 12); result[i] = (jsHour.length > 1) ? jsHour : "0" + jsHour; break; case "H": /* 24-hour format of an hour with leading zeros; 01 to 24 */ jsHour = String(jsDate.getHours()); result[i] = (jsHour.length > 1) ? jsHour : "0" + jsHour; break; case "i": /* Minutes with leading zeros; 00 to 59 */ var jsMinute = String(jsDate.getMinutes()); result[i] = (jsMinute.length > 1) ? jsMinute : "0" + jsMinute; break; case "s": /* Seconds with leading zeros; 00 to 59; */ var jsSecond = String(jsDate.getSeconds()); result[i] = (jsSecond.length > 1) ? jsSecond : "0" + jsSecond; break; case "u": /* Microseconds */ throw '"u" not implemented!'; /* TIMEZONE */ case "e": /* Timezone identifier */ throw '"e" not implemented!'; case "I": /* "1" if Daylight Savings Time, "0" otherwise. Works only on the northern hemisphere */ jsFirstDay = new Date(jsDate.getFullYear(), 0, 1); result[i] = (jsDate.getTimezoneOffset() !== jsFirstDay.getTimezoneOffset()) ? (1) : (0); break; case "O": /* Difference to Greenwich time (GMT) in hours */ var jsMinZone = jsDate.getTimezoneOffset(); var jsMinutes = jsMinZone % 60; jsHour = String(((jsMinZone - jsMinutes) / 60) * -1); if (jsHour.charAt(0) !== "-") { jsHour = "+" + jsHour; } jsHour = (jsHour.length === 3) ? (jsHour) : (jsHour.replace(/([+\-])(\d)/, "$1" + 0 + "$2")); result[i] = jsHour + jsMinutes + "0"; break; case "P": /* Difference to Greenwich time (GMT) with colon between hours and minutes */ throw '"P" not implemented!'; case "T": /* Timezone abbreviation */ throw '"T" not implemented!'; case "Z": /* Timezone offset in seconds. The offset for timezones west of UTC is always negative, and for those east of UTC is always positive. */ result[i] = jsDate.getTimezoneOffset() * 60; break; /* FULL DATE/TIME */ case "c": /* ISO 8601 date */ throw '"c" not implemented!'; case "r": /* RFC 2822 formatted date */ var jsDayName = this._wDays(jsDate.getDay()).substr(0, 3); var jsMonthName = this._months(jsDate.getMonth()).substr(0, 3); result[i] = jsDayName + ", " + jsDate.getDate() + " " + jsMonthName + this.get(" Y H:i:s O",jsDate); break; case "U": /* Seconds since the Unix Epoch (January 1 1970 00:00:00 GMT) */ result[i] = Math.floor(jsDate.getTime() / 1000); break; default: result[i] = iFormat[i]; } } return result.join(''); }, /** * Creates a date object based on a format string. * This works exactly as php date() function. http://php.net/manual/en/function.date.php * * @method set * @param {String} [format] The format in which the date will be formatted. Defaults to 'Y-m-d' * @param {String} str_date The date formatted. * @return {Date} Date object based on the formatted date and format * @public * @static * @sample Ink_Util_Date_set.html */ set : function( format , str_date ) { if ( typeof str_date === 'undefined' ) { return ; } if ( typeof format === 'undefined' || format === '' ) { format = "Y-m-d"; } var iFormat = format.split(""); var result = new Array( iFormat.length ); var escapeChar = "\\"; var mList; var objIndex = { year : undefined , month : undefined , day : undefined , dayY : undefined , dayW : undefined , week : undefined , hour : undefined , hourD : undefined , min : undefined , sec : undefined , msec : undefined , ampm : undefined , diffM : undefined , diffH : undefined , date : undefined }; var matches = 0; /* This switch is presented in the same order as in php date function (PHP 5.2.2) */ for ( var i = 0; i < iFormat.length; i++) { switch( iFormat[ i ] ) { case escapeChar: result[i] = iFormat[ i + 1 ]; i++; break; /* DAY */ case "d": /* Day of the month, 2 digits with leading zeros; ex: 01 to 31 */ result[ i ] = '(\\d{2})'; objIndex.day = { original : i , match : matches++ }; break; case "j": /* Day of the month without leading zeros; ex: 1 to 31 */ result[ i ] = '(\\d{1,2})'; objIndex.day = { original : i , match : matches++ }; break; case "D": /* A textual representation of a day, three letters; Seg to Dom */ result[ i ] = '([\\wá]{3})'; objIndex.dayW = { original : i , match : matches++ }; break; case "l": /* A full textual representation of the day of the week; Domingo to Sabado */ result[i] = '([\\wá]{5,7})'; objIndex.dayW = { original : i , match : matches++ }; break; case "N": /* ISO-8601 numeric representation of the day of the week; 1 (Segunda) to 7 (Domingo) */ result[ i ] = '(\\d)'; objIndex.dayW = { original : i , match : matches++ }; break; case "w": /* Numeric representation of the day of the week; 0 (for Sunday) through 6 (for Saturday) */ result[ i ] = '(\\d)'; objIndex.dayW = { original : i , match : matches++ }; break; case "S": /* English ordinal suffix for the day of the month, 2 characters; st, nd, rd or th. Works well with j */ result[ i ] = '\\w{2}'; break; case "z": /* The day of the year (starting from 0); 0 to 365 */ result[ i ] = '(\\d{1,3})'; objIndex.dayY = { original : i , match : matches++ }; break; /* WEEK */ case "W": /* ISO-8601 week number of year, weeks starting on Monday; ex: 42 (the 42nd week in the year) */ result[ i ] = '(\\d{1,2})'; objIndex.week = { original : i , match : matches++ }; break; /* MONTH */ case "F": /* A full textual representation of a month, such as Janeiro or Marco; Janeiro a Dezembro */ result[ i ] = '([\\wç]{4,9})'; objIndex.month = { original : i , match : matches++ }; break; case "M": /* A short textual representation of a month, three letters; Jan a Dez */ result[ i ] = '(\\w{3})'; objIndex.month = { original : i , match : matches++ }; break; case "m": /* Numeric representation of a month, with leading zeros; 01 to 12 */ result[ i ] = '(\\d{2})'; objIndex.month = { original : i , match : matches++ }; break; case "n": /* Numeric representation of a month, without leading zeros; 1 a 12 */ result[ i ] = '(\\d{1,2})'; objIndex.month = { original : i , match : matches++ }; break; case "t": /* Number of days in the given month; ex: 28 */ result[ i ] = '\\d{2}'; break; /* YEAR */ case "L": /* Whether it's a leap year; 1 if it is a leap year, 0 otherwise. */ result[ i ] = '\\w{4,5}'; break; case "o": /* ISO-8601 year number. This has the same value as Y, except that if the ISO week number (W) belongs to the previous or next year, that year is used instead. */ throw '"o" not implemented!'; case "Y": /* A full numeric representation of a year, 4 digits; 1999 */ result[ i ] = '(\\d{4})'; objIndex.year = { original : i , match : matches++ }; break; case "y": /* A two digit representation of a year; 99 */ result[ i ] = '(\\d{2})'; if ( typeof objIndex.year === 'undefined' || iFormat[ objIndex.year.original ] !== 'Y' ) { objIndex.year = { original : i , match : matches++ }; } break; /* TIME */ case "a": /* Lowercase Ante meridiem and Post meridiem; am or pm */ result[ i ] = '(am|pm)'; objIndex.ampm = { original : i , match : matches++ }; break; case "A": /* Uppercase Ante meridiem and Post meridiem; AM or PM */ result[ i ] = '(AM|PM)'; objIndex.ampm = { original : i , match : matches++ }; break; case "B": /* Swatch Internet time; 000 through 999 */ throw '"B" not implemented!'; case "g": /* 12-hour format of an hour without leading zeros; 1 to 12 */ result[ i ] = '(\\d{1,2})'; objIndex.hourD = { original : i , match : matches++ }; break; case "G": /* 24-hour format of an hour without leading zeros; 1 to 23 */ result[ i ] = '(\\d{1,2})'; objIndex.hour = { original : i , match : matches++ }; break; case "h": /* 12-hour format of an hour with leading zeros; 01 to 12 */ result[ i ] = '(\\d{2})'; objIndex.hourD = { original : i , match : matches++ }; break; case "H": /* 24-hour format of an hour with leading zeros; 01 to 24 */ result[ i ] = '(\\d{2})'; objIndex.hour = { original : i , match : matches++ }; break; case "i": /* Minutes with leading zeros; 00 to 59 */ result[ i ] = '(\\d{2})'; objIndex.min = { original : i , match : matches++ }; break; case "s": /* Seconds with leading zeros; 00 to 59; */ result[ i ] = '(\\d{2})'; objIndex.sec = { original : i , match : matches++ }; break; case "u": /* Microseconds */ throw '"u" not implemented!'; /* TIMEZONE */ case "e": /* Timezone identifier */ throw '"e" not implemented!'; case "I": /* "1" if Daylight Savings Time, "0" otherwise. Works only on the northern hemisphere */ result[i] = '\\d'; break; case "O": /* Difference to Greenwich time (GMT) in hours */ result[ i ] = '([-+]\\d{4})'; objIndex.diffH = { original : i , match : matches++ }; break; case "P": /* Difference to Greenwich time (GMT) with colon between hours and minutes */ throw '"P" not implemented!'; case "T": /* Timezone abbreviation */ throw '"T" not implemented!'; case "Z": /* Timezone offset in seconds. The offset for timezones west of UTC is always negative, and for those east of UTC is always positive. */ result[ i ] = '(\\-?\\d{1,5})'; objIndex.diffM = { original : i , match : matches++ }; break; /* FULL DATE/TIME */ case "c": /* ISO 8601 date */ throw '"c" not implemented!'; case "r": /* RFC 2822 formatted date */ result[ i ] = '([\\wá]{3}, \\d{1,2} \\w{3} \\d{4} \\d{2}:\\d{2}:\\d{2} [+\\-]\\d{4})'; objIndex.date = { original : i , match : matches++ }; break; case "U": /* Seconds since the Unix Epoch (January 1 1970 00:00:00 GMT) */ result[ i ] = '(\\d{1,13})'; objIndex.date = { original : i , match : matches++ }; break; default: result[ i ] = iFormat[ i ]; } } var pattr = new RegExp( result.join('') ); try { mList = str_date.match( pattr ); if ( !mList ) { return; } } catch ( e ) { return ; } var _haveDatetime = typeof objIndex.date !== 'undefined'; var _haveYear = typeof objIndex.year !== 'undefined'; var _haveYDay = typeof objIndex.dayY !== 'undefined'; var _haveDay = typeof objIndex.day !== 'undefined'; var _haveMonth = typeof objIndex.month !== 'undefined'; var _haveMonthDay = _haveMonth && _haveDay; var _haveOnlyDay = !_haveMonth && _haveDay; var _haveWDay = typeof objIndex.dayW !== 'undefined'; var _haveWeek = typeof objIndex.week !== 'undefined'; var _haveWeekWDay = _haveWeek && _haveWDay; var _haveOnlyWDay = !_haveWeek && _haveWDay; var _validDate = _haveYDay || _haveMonthDay || !_haveYear && _haveOnlyDay || _haveWeekWDay || !_haveYear && _haveOnlyWDay; var _noDate = !_haveYear && !_haveYDay && !_haveDay && !_haveMonth && !_haveWDay && !_haveWeek; var _haveHour12 = typeof objIndex.hourD !== 'undefined' && typeof objIndex.ampm !== 'undefined'; var _haveHour24 = typeof objIndex.hour !== 'undefined'; var _haveHour = _haveHour12 || _haveHour24; var _haveMin = typeof objIndex.min !== 'undefined'; var _haveSec = typeof objIndex.sec !== 'undefined'; var _haveMSec = typeof objIndex.msec !== 'undefined'; var _haveMoreM = !_noDate || _haveHour; var _haveMoreS = _haveMoreM || _haveMin; var _haveDiffM = typeof objIndex.diffM !== 'undefined'; var _haveDiffH = typeof objIndex.diffH !== 'undefined'; //var _haveGMT = _haveDiffM || _haveDiffH; var hour; var min; if ( _haveDatetime ) { if ( iFormat[ objIndex.date.original ] === 'U' ) { return new Date( +mList[ objIndex.date.match + 1 ] * 1000 ); } var dList = mList[ objIndex.date.match + 1 ].match( /\w{3}, (\d{1,2}) (\w{3}) (\d{4}) (\d{2}):(\d{2}):(\d{2}) ([+\-]\d{4})/ ); hour = +dList[ 4 ] + ( +dList[ 7 ].slice( 0 , 3 ) ); min = +dList[ 5 ] + ( dList[ 7 ].slice( 0 , 1 ) + dList[ 7 ].slice( 3 ) ) / 100 * 60; return new Date( dList[ 3 ] , this._iMonth( dList[ 2 ] ) , dList[ 1 ] , hour , min , dList[ 6 ] ); } var _d = new Date( ); var year; var month; var day; var sec; var msec; var gmt; if ( !_validDate && !_noDate ) { return ; } if ( _validDate ) { if ( _haveYear ) { var _y = _d.getFullYear( ) - 50 + ''; year = mList[ objIndex.year.match + 1 ]; if ( iFormat[ objIndex.year.original ] === 'y' ) { year = +_y.slice( 0 , 2 ) + ( year >= ( _y ).slice( 2 ) ? 0 : 1 ) + year; } } else { year = _d.getFullYear(); } if ( _haveYDay ) { month = 0; day = mList[ objIndex.dayY.match + 1 ]; } else if ( _haveDay ) { if ( _haveMonth ) { month = this._iMonth( mList[ objIndex.month.match + 1 ] ); } else { month = _d.getMonth( ); } day = mList[ objIndex.day.match + 1 ]; } else { month = 0; var week; if ( _haveWeek ) { week = mList[ objIndex.week.match + 1 ]; } else { week = this.get( 'W' , _d ); } day = ( week - 2 ) * 7 + ( 8 - ( ( new Date( year , 0 , 1 ) ).getDay( ) || 7 ) ) + this._iWeek( mList[ objIndex.week.match + 1 ] ); } if ( month === 0 && day > 31 ) { var aux = new Date( year , month , day ); month = aux.getMonth( ); day = aux.getDate( ); } } else { year = _d.getFullYear( ); month = _d.getMonth( ); day = _d.getDate( ); } if ( _haveHour12 ) { hour = +mList[ objIndex.hourD.match + 1 ] + ( mList[ objIndex.ampm.match + 1 ] === 'pm' ? 12 : 0 ); } else if ( _haveHour24 ) { hour = mList[ objIndex.hour.match + 1 ]; } else if ( _noDate ) { hour = _d.getHours( ); } else { hour = '00'; } if ( _haveMin ) { min = mList[ objIndex.min.match + 1 ]; } else if ( !_haveMoreM ) { min = _d.getMinutes( ); } else { min = '00'; } if ( _haveSec ) { sec = mList[ objIndex.sec.match + 1 ]; } else if ( !_haveMoreS ) { sec = _d.getSeconds( ); } else { sec = '00'; } if ( _haveMSec ) { msec = mList[ objIndex.msec.match + 1 ]; } else { msec = '000'; } if ( _haveDiffH ) { gmt = mList[ objIndex.diffH.match + 1 ]; } else if ( _haveDiffM ) { gmt = String( -1 * mList[ objIndex.diffM.match + 1 ] / 60 * 100 ).replace( /^(\d)/ , '+$1' ).replace( /(^[\-+])(\d{3}$)/ , '$10$2' ); } else { gmt = '+0000'; } return new Date( year, month, day, hour, min, sec ); } }; return InkDate; }); /** * Dump/Profiling Utilities * @module Ink.Util.Dumper_1 * @version 1 */ Ink.createModule('Ink.Util.Dumper', '1', [], function() { 'use strict'; /** * @namespace Ink.Util.Dumper_1 */ var Dumper = { /** * Hex code for the 'tab' * * @property _tab * @type {String} * @private * @readOnly * @static * */ _tab: '\xA0\xA0\xA0\xA0', /** * Function that returns the argument passed formatted * * @method _formatParam * @param {Mixed} param * @return {String} The argument passed formatted * @private * @static */ _formatParam: function(param) { var formated = ''; switch(typeof(param)) { case 'string': formated = '(string) '+param; break; case 'number': formated = '(number) '+param; break; case 'boolean': formated = '(boolean) '+param; break; case 'object': if(param !== null) { if(param.constructor === Array) { formated = 'Array \n{\n' + this._outputFormat(param, 0) + '\n}'; } else { formated = 'Object \n{\n' + this._outputFormat(param, 0) + '\n}'; } } else { formated = 'null'; } break; default: formated = false; } return formated; }, /** * Function that returns the tabs concatenated * * @method _getTabs * @param {Number} numberOfTabs Number of Tabs * @return {String} Tabs concatenated * @private * @static */ _getTabs: function(numberOfTabs) { var tabs = ''; for(var _i = 0; _i < numberOfTabs; _i++) { tabs += this._tab; } return tabs; }, /** * Function that formats the parameter to display. * * @method _outputFormat * @param {Any} param * @param {Number} dim * @return {String} The parameter passed formatted to displat * @private * @static */ _outputFormat: function(param, dim) { var formated = ''; //var _strVal = false; var _typeof = false; for(var key in param) { if(param[key] !== null) { if(typeof(param[key]) === 'object' && (param[key].constructor === Array || param[key].constructor === Object)) { if(param[key].constructor === Array) { _typeof = 'Array'; } else if(param[key].constructor === Object) { _typeof = 'Object'; } formated += this._tab + this._getTabs(dim) + '[' + key + '] => '+_typeof+'\n'; formated += this._tab + this._getTabs(dim) + '{\n'; formated += this._outputFormat(param[key], dim + 1) + this._tab + this._getTabs(dim) + '}\n'; } else if(param[key].constructor === Function) { continue; } else { formated = formated + this._tab + this._getTabs(dim) + '[' + key + '] => ' + param[key] + '\n'; } } else { formated = formated + this._tab + this._getTabs(dim) + '[' + key + '] => null \n'; } } return formated; }, /** * Prints variable structure. * * @method printDump * @param {Any} param Variable to be dumped. * @param {DOMElement|String} [target] Element to print the dump on. * @public * @static * @sample Ink_Util_Dumper_printDump.html */ printDump: function(param, target) { /*jshint evil:true */ if(!target || typeof(target) === 'undefined') { document.write('
'+this._formatParam(param)+'
'); } else { if(typeof(target) === 'string') { document.getElementById(target).innerHTML = '
' + this._formatParam(param) + '
'; } else if(typeof(target) === 'object') { target.innerHTML = '
'+this._formatParam(param)+'
'; } else { throw "TARGET must be an element or an element ID"; } } }, /** * Get a variable's structure. * * @method returnDump * @param {Any} param Variable to get the structure. * @return {String} The variable's structure. * @public * @static * @sample Ink_Util_Dumper_returnDump.html */ returnDump: function(param) { return this._formatParam(param); }, /** * Alert a variable's structure. * * @method alertDump * @param {Any} param Variable to be dumped. * @public * @static * @sample Ink_Util_Dumper_alertDump.html */ alertDump: function(param) { window.alert(this._formatParam(param).replace(/()(Array|Object)(<\/b>)/g, "$2")); }, /** * Prints the variable structure to a new window. * * @method windowDump * @param {Any} param Variable to be dumped. * @public * @static * @sample Ink_Util_Dumper_windowDump.html */ windowDump: function(param) { var dumperwindow = 'dumperwindow_'+(Math.random() * 10000); var win = window.open('', dumperwindow, 'width=400,height=300,left=50,top=50,status,menubar,scrollbars,resizable' ); win.document.open(); win.document.write('
'+this._formatParam(param)+'
'); win.document.close(); win.focus(); } }; return Dumper; }); /** * Internationalization Utilities * @module Ink.Util.I18n_1 * @version 1 */ Ink.createModule('Ink.Util.I18n', '1', [], function () { 'use strict'; var pattrText = /\{(?:(\{.*?})|(?:%s:)?(\d+)|(?:%s)?|([\w-]+))}/g; var funcOrVal = function( ret , args ) { if ( typeof ret === 'function' ) { return ret.apply(this, args); } else if (typeof ret !== undefined) { return ret; } else { return ''; } }; /** * You can use this module to internationalize your applications. It roughly emulates GNU gettext's API. * * @class Ink.Util.I18n * @constructor * * @param {Object} dict Object mapping language codes (in the form of `pt_PT`, `pt_BR`, `fr`, `en_US`, etc.) to their `dictionaries` * @param {String} [lang='pt_PT'] language code of the target language * * @sample Ink_Util_I18n_1.html */ var I18n = function( dict , lang , testMode ) { if ( !( this instanceof I18n ) ) { return new I18n( dict , lang , testMode ); } this.reset( ) .lang( lang ) .testMode( testMode ) .append( dict || { } , lang ); }; I18n.prototype = { reset: function( ) { this._dicts = [ ]; this._dict = { }; this._testMode = false; this._lang = this._gLang; return this; }, /** * Adds translation strings for the helper to use. * * @method append * @param {Object} dict Object containing language objects identified by their language code * * @sample Ink_Util_I18n_1_append.html */ append: function( dict ) { this._dicts.push( dict ); this._dict = Ink.extendObj(this._dict , dict[ this._lang ] ); return this; }, /** * Gets or sets the language. * If there are more dictionaries available in cache, they will be loaded. * * @method lang * @param {String} lang Language code to set this instance to. */ lang: function( lang ) { if ( !arguments.length ) { return this._lang; } if ( lang && this._lang !== lang ) { this._lang = lang; this._dict = { }; for ( var i = 0, l = this._dicts.length; i < l; i++ ) { this._dict = Ink.extendObj( this._dict , this._dicts[ i ][ lang ] || { } ); } } return this; }, /** * Sets or unsets test mode. * In test mode, unknown strings are wrapped in `[ ... ]`. This is useful for debugging your application and to make sure all your translation keys are in place. * * @method testMode * @param {Boolean} bool Flag to set the test mode state */ testMode: function( bool ) { if ( !arguments.length ) { return !!this._testMode; } if ( bool !== undefined ) { this._testMode = !!bool; } return this; }, /** * Gest a key from the current dictionary * * @method getKey * @param {String} key * @return {Mixed} The object which happened to be in the current language dictionary on the given key. * * @sample Ink_Util_I18n_1_getKey.html */ getKey: function( key ) { var ret; var gLang = this._gLang; var lang = this._lang; if ( key in this._dict ) { ret = this._dict[ key ]; } else { I18n.langGlobal( lang ); ret = this._gDict[ key ]; I18n.langGlobal( gLang ); } return ret; }, /** * Translates a string. * Given a translation key, return a translated string, with replaced parameters. * When a translated string is not available, the original string is returned unchanged. * * @method text * @param {String} str Key to look for in i18n dictionary (which is returned verbatim if unknown) * @param {Object} [namedParms] Named replacements. Replaces {named} with values in this object. * @param {String} [args] Replacement #1 (replaces first {} and all {1}) * @param {String} [arg2] Replacement #2 (replaces second {} and all {2}) * @param {String} [argn*] Replacement #n (replaces nth {} and all {n}) * * @sample Ink_Util_I18n_1_text.html */ text: function( str /*, replacements...*/ ) { if ( typeof str !== 'string' ) { return; } // Backwards-compat var pars = Array.prototype.slice.call( arguments , 1 ); var idx = 0; var isObj = typeof pars[ 0 ] === 'object'; var original = this.getKey( str ); if ( original === undefined ) { original = this._testMode ? '[' + str + ']' : str; } if ( typeof original === 'number' ) { original += ''; } if (typeof original === 'string') { original = original.replace( pattrText , function( m , $1 , $2 , $3 ) { var ret = $1 ? $1 : $2 ? pars[ $2 - ( isObj ? 0 : 1 ) ] : $3 ? pars[ 0 ][ $3 ] || '' : pars[ (idx++) + ( isObj ? 1 : 0 ) ]; return funcOrVal( ret , [idx].concat(pars) ); }); return original; } return ( typeof original === 'function' ? original.apply( this , pars ) : original instanceof Array ? funcOrVal( original[ pars[ 0 ] ] , pars ) : typeof original === 'object' ? funcOrVal( original[ pars[ 0 ] ] , pars ) : ''); }, /** * Translates and pluralizes text. * Given a singular string, a plural string and a number, translates either the singular or plural string. * * @method ntext * @return {String} * * @param {String} strSin Word to use when count is 1 * @param {String} strPlur Word to use otherwise * @param {Number} count Number which defines which word to use * @param [args*] Extra arguments, to be passed to `text()` * * @sample Ink_Util_I18n_1_ntext.html */ ntext: function( strSin , strPlur , count ) { var pars = Array.prototype.slice.apply( arguments ); var original; if ( pars.length === 2 && typeof strPlur === 'number' ) { original = this.getKey( strSin ); if ( !( original instanceof Array ) ) { return ''; } pars.splice( 0 , 1 ); original = original[ strPlur === 1 ? 0 : 1 ]; } else { pars.splice( 0 , 2 ); original = count === 1 ? strSin : strPlur; } return this.text.apply( this , [ original ].concat( pars ) ); }, /** * Gets the ordinal suffix of a number. * * This works by using transforms (in the form of Objects or Functions) passed into the function or found in the special key `_ordinals` in the active language dictionary. * * @method ordinal * * @param {Number} num Input number * @param {Object|Function} [options]={} Dictionaries for translating. Each of these options' fallback is found in the current language's dictionary. The lookup order is the following: `exceptions`, `byLastDigit`, `default`. Each of these may be either an `Object` or a `Function`. If it's a function, it is called (with `number` and `digit` for any function except for byLastDigit, which is called with the `lastDigit` of the number in question), and if the function returns a string, that is used. If it's an object, the property is looked up using `obj[prop]`. If what is found is a string, it is used directly. * @param {Object|Function} [options.byLastDigit]={} If the language requires the last digit to be considered, mappings of last digits to ordinal suffixes can be created here. * @param {Object|Function} [options.exceptions]={} Map unique, special cases to their ordinal suffixes. * * @returns {String} Ordinal suffix for `num`. * * @sample Ink_Util_I18n_1_ordinal.html **/ ordinal: function( num ) { if ( num === undefined ) { return ''; } var lastDig = +num.toString( ).slice( -1 ); var ordDict = this.getKey( '_ordinals' ); if ( ordDict === undefined ) { return ''; } if ( typeof ordDict === 'string' ) { return ordDict; } var ret; if ( typeof ordDict === 'function' ) { ret = ordDict( num , lastDig ); if ( typeof ret === 'string' ) { return ret; } } if ( 'exceptions' in ordDict ) { ret = typeof ordDict.exceptions === 'function' ? ordDict.exceptions( num , lastDig ) : num in ordDict.exceptions ? funcOrVal( ordDict.exceptions[ num ] , [num , lastDig] ) : undefined; if ( typeof ret === 'string' ) { return ret; } } if ( 'byLastDigit' in ordDict ) { ret = typeof ordDict.byLastDigit === 'function' ? ordDict.byLastDigit( lastDig , num ) : lastDig in ordDict.byLastDigit ? funcOrVal( ordDict.byLastDigit[ lastDig ] , [lastDig , num] ) : undefined; if ( typeof ret === 'string' ) { return ret; } } if ( 'default' in ordDict ) { ret = funcOrVal( ordDict['default'] , [ num , lastDig ] ); if ( typeof ret === 'string' ) { return ret; } } return ''; }, /** * Create an alias. * * Returns an alias to this I18n instance. It contains the I18n methods documented here, but is also a function. If you call it, it just calls `text()`. This is commonly assigned to "_". * * @method alias * @returns {Function} an alias to `text()` on this instance. You can also access the rest of the translation API through this alias. * * @sample Ink_Util_I18n_1_alias.html */ alias: function( ) { var ret = Ink.bind( I18n.prototype.text , this ); ret.ntext = Ink.bind( I18n.prototype.ntext , this ); ret.append = Ink.bind( I18n.prototype.append , this ); ret.ordinal = Ink.bind( I18n.prototype.ordinal , this ); ret.testMode = Ink.bind( I18n.prototype.testMode , this ); return ret; } }; /** * Resets I18n global state (global dictionaries, and default language for instances) * * @method reset * @static * **/ I18n.reset = function( ) { I18n.prototype._gDicts = [ ]; I18n.prototype._gDict = { }; I18n.prototype._gLang = 'pt_PT'; }; I18n.reset( ); /** * Adds a dictionary to be used in all I18n instances for the corresponding language. * * @method appendGlobal * @static * * @param dict {Object} Dictionary to be added * @param lang {String} Language fo the dictionary being added * */ I18n.appendGlobal = function( dict , lang ) { if ( lang ) { if ( !( lang in dict ) ) { var obj = { }; obj[ lang ] = dict; dict = obj; } if ( lang !== I18n.prototype._gLang ) { I18n.langGlobal( lang ); } } I18n.prototype._gDicts.push( dict ); Ink.extendObj( I18n.prototype._gDict , dict[ I18n.prototype._gLang ] ); }; I18n.append = function () { // [3.1.0] remove this alias Ink.warn('Ink.Util.I18n.append() was renamed to appendGlobal().'); return I18n.appendGlobal.apply(I18n, [].slice.call(arguments)); }; /** * Gets or sets the current default language of I18n instances. * * @method langGlobal * @param lang the new language for all I18n instances * * @static * * @return {String} language code */ I18n.langGlobal = function( lang ) { if ( !arguments.length ) { return I18n.prototype._gLang; } if ( lang && I18n.prototype._gLang !== lang ) { I18n.prototype._gLang = lang; I18n.prototype._gDict = { }; for ( var i = 0, l = I18n.prototype._gDicts.length; i < l; i++ ) { Ink.extendObj( I18n.prototype._gDict , I18n.prototype._gDicts[ i ][ lang ] || { } ); } } }; I18n.lang = function () { // [3.1.0] remove this alias Ink.warn('Ink.Util.I18n.lang() was renamed to langGlobal().'); return I18n.langGlobal.apply(I18n, [].slice.call(arguments)); }; return I18n; }); /** * JSON Utilities * @module Ink.Util.Json_1 * @version 1 */ Ink.createModule('Ink.Util.Json', '1', [], function() { 'use strict'; var function_call = Function.prototype.call; var cx = /[\u0000\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g; function twoDigits(n) { var r = '' + n; if (r.length === 1) { return '0' + r; } else { return r; } } var dateToISOString = Date.prototype.toISOString ? Ink.bind(function_call, Date.prototype.toISOString) : function(date) { // Adapted from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Date/toISOString return date.getUTCFullYear() + '-' + twoDigits( date.getUTCMonth() + 1 ) + '-' + twoDigits( date.getUTCDate() ) + 'T' + twoDigits( date.getUTCHours() ) + ':' + twoDigits( date.getUTCMinutes() ) + ':' + twoDigits( date.getUTCSeconds() ) + '.' + String( (date.getUTCMilliseconds()/1000).toFixed(3) ).slice( 2, 5 ) + 'Z'; }; /** * Use this class to convert JSON strings to JavaScript objects * `.parse()` and also to do the opposite operation `.stringify()`. * Internally, the standard JSON implementation is used if available * Otherwise, the functions mimic the standard implementation. * * Here's how to produce JSON from an existing object: * * Ink.requireModules(['Ink.Util.Json_1'], function (Json) { * var obj = { * key1: 'value1', * key2: 'value2', * keyArray: ['arrayValue1', 'arrayValue2', 'arrayValue3'] * }; * Json.stringify(obj); // The above object as a JSON string * }); * * And here is how to parse JSON: * * Ink.requireModules(['Ink.Util.Json_1'], function (Json) { * var source = '{"key": "value", "array": [true, null, false]}'; * Json.parse(source); // The above JSON string as an object * }); * * @namespace Ink.Util.Json_1 * @static * */ var InkJson = { _nativeJSON: window.JSON || null, _convertToUnicode: false, // Escape characters so as to embed them in JSON strings _escape: function (theString) { var _m = { '\b': '\\b', '\t': '\\t', '\n': '\\n', '\f': '\\f', '\r': '\\r', '"': '\\"', '\\': '\\\\' }; if (/["\\\x00-\x1f]/.test(theString)) { theString = theString.replace(/([\x00-\x1f\\"])/g, function(a, b) { var c = _m[b]; if (c) { return c; } c = b.charCodeAt(); return '\\u00' + Math.floor(c / 16).toString(16) + (c % 16).toString(16); }); } return theString; }, // A character conversion map _toUnicode: function (theString) { if(!this._convertToUnicode) { return this._escape(theString); } else { var unicodeString = ''; var inInt = false; var theUnicode = false; var i = 0; var total = theString.length; while(i < total) { inInt = theString.charCodeAt(i); if( (inInt >= 32 && inInt <= 126) || //(inInt >= 48 && inInt <= 57) || //(inInt >= 65 && inInt <= 90) || //(inInt >= 97 && inInt <= 122) || inInt === 8 || inInt === 9 || inInt === 10 || inInt === 12 || inInt === 13 || inInt === 32 || inInt === 34 || inInt === 47 || inInt === 58 || inInt === 92) { if(inInt === 34 || inInt === 92 || inInt === 47) { theUnicode = '\\'+theString.charAt(i); } else if(inInt === 8) { theUnicode = '\\b'; } else if(inInt === 9) { theUnicode = '\\t'; } else if(inInt === 10) { theUnicode = '\\n'; } else if(inInt === 12) { theUnicode = '\\f'; } else if(inInt === 13) { theUnicode = '\\r'; } else { theUnicode = theString.charAt(i); } } else { if(this._convertToUnicode) { theUnicode = theString.charCodeAt(i).toString(16)+''.toUpperCase(); while (theUnicode.length < 4) { theUnicode = '0' + theUnicode; } theUnicode = '\\u' + theUnicode; } else { theUnicode = theString.charAt(i); } } unicodeString += theUnicode; i++; } return unicodeString; } }, _stringifyValue: function(param) { if (typeof param === 'string') { return '"' + this._toUnicode(param) + '"'; } else if (typeof param === 'number' && (isNaN(param) || !isFinite(param))) { // Unusable numbers go null return 'null'; } else if (typeof param === 'undefined' || param === null) { // And so does undefined return 'null'; } else if (typeof param.toJSON === 'function') { var t = param.toJSON(); if (typeof t === 'string') { return '"' + this._escape(t) + '"'; } else { return this._escape(t.toString()); } } else if (typeof param === 'number' || typeof param === 'boolean') { // These ones' toString methods return valid JSON. return '' + param; } else if (typeof param === 'function') { return 'null'; // match JSON.stringify } else if (param.constructor === Date) { return '"' + this._escape(dateToISOString(param)) + '"'; } else if (param.constructor === Array) { var arrayString = ''; for (var i = 0, len = param.length; i < len; i++) { if (i > 0) { arrayString += ','; } arrayString += this._stringifyValue(param[i]); } return '[' + arrayString + ']'; } else { // Object var objectString = ''; for (var k in param) { if ({}.hasOwnProperty.call(param, k)) { if (objectString !== '') { objectString += ','; } objectString += '"' + this._escape(k) + '": ' + this._stringifyValue(param[k]); } } return '{' + objectString + '}'; } }, /** * Serializes a JSON object into a string. * * @method stringify * @param {Object} input Data to be serialized into JSON * @param {Boolean} convertToUnicode When `true`, converts string contents to unicode \uXXXX * @return {String} Serialized string * * @sample Ink_Util_Json_stringify.html */ stringify: function(input, convertToUnicode) { this._convertToUnicode = !!convertToUnicode; if(!this._convertToUnicode && this._nativeJSON) { return this._nativeJSON.stringify(input); } return this._stringifyValue(input); // And recurse. }, /** * Parses a JSON text through a function * * @method parse * @param text {String} Input string * @param reviver {Function} Function receiving `(key, value)`, and `this`=(containing object), used to walk objects. * * @return {Object} JSON object * * @sample Ink_Util_Json_parse.html */ /* From https://github.com/douglascrockford/JSON-js/blob/master/json.js */ parse: function (text, reviver) { /*jshint evil:true*/ // The parse method takes a text and an optional reviver function, and returns // a JavaScript value if the text is a valid JSON text. var j; function walk(holder, key) { // The walk method is used to recursively walk the resulting structure so // that modifications can be made. var k, v, value = holder[key]; if (value && typeof value === 'object') { for (k in value) { if (Object.prototype.hasOwnProperty.call(value, k)) { v = walk(value, k); if (v !== undefined) { value[k] = v; } else { delete value[k]; } } } } return reviver.call(holder, key, value); } // Parsing happens in four stages. In the first stage, we replace certain // Unicode characters with escape sequences. JavaScript handles many characters // incorrectly, either silently deleting them, or treating them as line endings. text = String(text); cx.lastIndex = 0; if (cx.test(text)) { text = text.replace(cx, function (a) { return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4); }); } // In the second stage, we run the text against regular expressions that look // for non-JSON patterns. We are especially concerned with '()' and 'new' // because they can cause invocation, and '=' because it can cause mutation. // But just to be safe, we want to reject all unexpected forms. // We split the second stage into 4 regexp operations in order to work around // crippling inefficiencies in IE's and Safari's regexp engines. First we // replace the JSON backslash pairs with '@' (a non-JSON character). Second, we // replace all simple value tokens with ']' characters. Third, we delete all // open brackets that follow a colon or comma or that begin the text. Finally, // we look to see that the remaining characters are only whitespace or ']' or // ',' or ':' or '{' or '}'. If that is so, then the text is safe for eval. if (/^[\],:{}\s]*$/ .test(text.replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, '@') .replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']') .replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { // In the third stage we use the eval function to compile the text into a // JavaScript structure. The '{' operator is subject to a syntactic ambiguity // in JavaScript: it can begin a block or an object literal. We wrap the text // in parens to eliminate the ambiguity. j = eval('(' + text + ')'); // In the optional fourth stage, we recursively walk the new structure, passing // each name/value pair to a reviver function for possible transformation. return typeof reviver === 'function' ? walk({'': j}, '') : j; } // If the text is not JSON parseable, then a SyntaxError is thrown. throw new SyntaxError('JSON.parse'); } }; return InkJson; }); /** * String Utilities * @module Ink.Util.String_1 * @version 1 */ Ink.createModule('Ink.Util.String', '1', [], function() { 'use strict'; /** * @namespace Ink.Util.String_1 */ var InkUtilString = { /** * List of special chars * * @property _chars * @type {Array} * @private * @readOnly * @static */ _chars: ['&','à','á','â','ã','ä','å','æ','ç','è','é', 'ê','ë','ì','í','î','ï','ð','ñ','ò','ó','ô', 'õ','ö','ø','ù','ú','û','ü','ý','þ','ÿ','À', 'Á','Â','Ã','Ä','Å','Æ','Ç','È','É','Ê','Ë', 'Ì','Í','Î','Ï','Ð','Ñ','Ò','Ó','Ô','Õ','Ö', 'Ø','Ù','Ú','Û','Ü','Ý','Þ','€','\"','ß','<', '>','¢','£','¤','¥','¦','§','¨','©','ª','«', '¬','\xad','®','¯','°','±','²','³','´','µ','¶', '·','¸','¹','º','»','¼','½','¾'], /** * List of the special characters' html entities * * @property _entities * @type {Array} * @private * @readOnly * @static */ _entities: ['amp','agrave','aacute','acirc','atilde','auml','aring', 'aelig','ccedil','egrave','eacute','ecirc','euml','igrave', 'iacute','icirc','iuml','eth','ntilde','ograve','oacute', 'ocirc','otilde','ouml','oslash','ugrave','uacute','ucirc', 'uuml','yacute','thorn','yuml','Agrave','Aacute','Acirc', 'Atilde','Auml','Aring','AElig','Ccedil','Egrave','Eacute', 'Ecirc','Euml','Igrave','Iacute','Icirc','Iuml','ETH','Ntilde', 'Ograve','Oacute','Ocirc','Otilde','Ouml','Oslash','Ugrave', 'Uacute','Ucirc','Uuml','Yacute','THORN','euro','quot','szlig', 'lt','gt','cent','pound','curren','yen','brvbar','sect','uml', 'copy','ordf','laquo','not','shy','reg','macr','deg','plusmn', 'sup2','sup3','acute','micro','para','middot','cedil','sup1', 'ordm','raquo','frac14','frac12','frac34'], /** * List of accented chars * * @property _accentedChars * @type {Array} * @private * @readOnly * @static */ _accentedChars:['à','á','â','ã','ä','å', 'è','é','ê','ë', 'ì','í','î','ï', 'ò','ó','ô','õ','ö', 'ù','ú','û','ü', 'ç','ñ', 'À','Á','Â','Ã','Ä','Å', 'È','É','Ê','Ë', 'Ì','Í','Î','Ï', 'Ò','Ó','Ô','Õ','Ö', 'Ù','Ú','Û','Ü', 'Ç','Ñ'], /** * List of the accented chars (above), but without the accents * * @property _accentedRemovedChars * @type {Array} * @private * @readOnly * @static */ _accentedRemovedChars:['a','a','a','a','a','a', 'e','e','e','e', 'i','i','i','i', 'o','o','o','o','o', 'u','u','u','u', 'c','n', 'A','A','A','A','A','A', 'E','E','E','E', 'I','I','I','I', 'O','O','O','O','O', 'U','U','U','U', 'C','N'], /** * Object that contains the basic HTML unsafe chars, as keys, and their HTML entities as values * * @property _htmlUnsafeChars * @type {Object} * @private * @readOnly * @static */ _htmlUnsafeChars:{'<':'<','>':'>','&':'&','"':'"',"'":'''}, /** * Capitalizes a word. * If param as more than one word, it converts first letter of all words that have more than 2 letters * * @method ucFirst * @param {String} string String to capitalize. * @param {Boolean} [firstWordOnly]=false Flag to capitalize only the first word. * @return {String} Camel cased string. * @public * @static * @sample Ink_Util_String_ucFirst.html */ ucFirst: function(string, firstWordOnly) { var replacer = firstWordOnly ? /(^|\s)(\w)(\S{2,})/ : /(^|\s)(\w)(\S{2,})/g; return string ? String(string).replace(replacer, function(_, $1, $2, $3){ return $1 + $2.toUpperCase() + $3.toLowerCase(); }) : string; }, /** * Trims whitespace from strings * * @method trim * @param {String} string String to be trimmed * @return {String} Trimmed string * @public * @static * @sample Ink_Util_String_trim.html */ trim: function(string) { if (typeof string === 'string') { return string.replace(/^\s+|\s+$|\n+$/g, ''); } return string; }, /** * Strips HTML tags from strings * * @method stripTags * @param {String} string String to strip tags from. * @param {String} allowed Comma separated list of allowed tags. * @return {String} Stripped string * @public * @static * @sample Ink_Util_String_stripTags.html */ stripTags: function(string, allowed) { if (allowed && typeof allowed === 'string') { var aAllowed = InkUtilString.trim(allowed).split(','); var aNewAllowed = []; var cleanedTag = false; for(var i=0; i < aAllowed.length; i++) { if(InkUtilString.trim(aAllowed[i]) !== '') { cleanedTag = InkUtilString.trim(aAllowed[i].replace(/(<|\>)/g, '').replace(/\s/, '')); aNewAllowed.push('(<'+cleanedTag+'\\s[^>]+>|<(\\s|\\/)?(\\s|\\/)?'+cleanedTag+'>)'); } } var strAllowed = aNewAllowed.join('|'); var reAllowed = new RegExp(strAllowed, "i"); var aFoundTags = string.match(new RegExp("<[^>]*>", "g")); for(var j=0; j < aFoundTags.length; j++) { if(!aFoundTags[j].match(reAllowed)) { string = string.replace((new RegExp(aFoundTags[j], "gm")), ''); } } return string; } else { return string.replace(/<[^\>]+\>/g, ''); } }, /** * Encodes string into HTML entities. * * @method htmlEntitiesEncode * @param {String} string * @return {String} string encoded * @public * @static * @sample Ink_Util_String_htmlEntitiesEncode.html */ htmlEntitiesEncode: function(string) { if (string && string.replace) { var re = false; for (var i = 0; i < InkUtilString._chars.length; i++) { re = new RegExp(InkUtilString._chars[i], "gm"); string = string.replace(re, '&' + InkUtilString._entities[i] + ';'); } } return string; }, /** * Decodes string from HTML entities. * * @method htmlEntitiesDecode * @param {String} string String to be decoded * @return {String} Decoded string * @public * @static * @sample Ink_Util_String_htmlEntitiesDecode.html */ htmlEntitiesDecode: function(string) { if (string && string.replace) { var re = false; for (var i = 0; i < InkUtilString._entities.length; i++) { re = new RegExp("&"+InkUtilString._entities[i]+";", "gm"); string = string.replace(re, InkUtilString._chars[i]); } string = string.replace(/&#[^;]+;?/g, function($0){ if ($0.charAt(2) === 'x') { return String.fromCharCode(parseInt($0.substring(3), 16)); } else { return String.fromCharCode(parseInt($0.substring(2), 10)); } }); } return string; }, /** * Encode a string to UTF-8. * * @method utf8Encode * @param {String} string String to be encoded * @return {String} string UTF-8 encoded string * @public * @static */ utf8Encode: function(string) { /*jshint bitwise:false*/ string = string.replace(/\r\n/g,"\n"); var utfstring = ""; for (var n = 0; n < string.length; n++) { var c = string.charCodeAt(n); if (c < 128) { utfstring += String.fromCharCode(c); } else if((c > 127) && (c < 2048)) { utfstring += String.fromCharCode((c >> 6) | 192); utfstring += String.fromCharCode((c & 63) | 128); } else { utfstring += String.fromCharCode((c >> 12) | 224); utfstring += String.fromCharCode(((c >> 6) & 63) | 128); utfstring += String.fromCharCode((c & 63) | 128); } } return utfstring; }, /** * Truncates a string without breaking words. * * @method shortString * @param {String} str String to truncate * @param {Number} n Number of chars of the short string * @return {String} * @public * @static * @sample Ink_Util_String_shortString.html */ shortString: function(str,n) { var words = str.split(' '); var resultstr = ''; for(var i = 0; i < words.length; i++ ){ if((resultstr + words[i] + ' ').length>=n){ resultstr += '…'; break; } resultstr += words[i] + ' '; } return resultstr; }, /** * Truncates a string, breaking words and adding ... at the end. * * @method truncateString * @param {String} str String to truncate * @param {Number} length Limit for the returned string, ellipsis included. * @return {String} Truncated String * @public * @static * @sample Ink_Util_String_truncateString.html */ truncateString: function(str, length) { if(str.length - 1 > length) { return str.substr(0, length - 1) + "\u2026"; } else { return str; } }, /** * Decodes a string from UTF-8. * * @method utf8Decode * @param {String} string String to be decoded * @return {String} Decoded string * @public * @static */ utf8Decode: function(utfstring) { /*jshint bitwise:false*/ var string = ""; var i = 0, c = 0, c2 = 0, c3 = 0; while ( i < utfstring.length ) { c = utfstring.charCodeAt(i); if (c < 128) { string += String.fromCharCode(c); i++; } else if((c > 191) && (c < 224)) { c2 = utfstring.charCodeAt(i+1); string += String.fromCharCode(((c & 31) << 6) | (c2 & 63)); i += 2; } else { c2 = utfstring.charCodeAt(i+1); c3 = utfstring.charCodeAt(i+2); string += String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)); i += 3; } } return string; }, /** * Removes all accented characters from a string. * * @method removeAccentedChars * @param {String} string String to remove accents from * @return {String} String without accented chars * @public * @static * @sample Ink_Util_String_removeAccentedChars.html */ removeAccentedChars: function(string) { var newString = string; var re = false; for (var i = 0; i < InkUtilString._accentedChars.length; i++) { re = new RegExp(InkUtilString._accentedChars[i], "gm"); newString = newString.replace(re, '' + InkUtilString._accentedRemovedChars[i] + ''); } return newString; }, /** * Count the number of occurrences of a specific needle in a haystack * * @method substrCount * @param {String} haystack String to search in * @param {String} needle String to search for * @return {Number} Number of occurrences * @public * @static * @sample Ink_Util_String_substrCount.html */ substrCount: function(haystack,needle) { return haystack ? haystack.split(needle).length - 1 : 0; }, /** * Eval a JSON - We recommend you Ink.Util.Json * * @method evalJSON * @param {String} strJSON JSON string to eval * @param {Boolean} sanitize Flag to sanitize input * @return {Object} JS Object * @public * @static */ evalJSON: function(strJSON, sanitize) { /* jshint evil:true */ if( (typeof sanitize === 'undefined' || sanitize === null) || InkUtilString.isJSON(strJSON)) { try { if(typeof(JSON) !== "undefined" && typeof(JSON.parse) !== 'undefined'){ return JSON.parse(strJSON); } return eval('('+strJSON+')'); } catch(e) { throw new Error('ERROR: Bad JSON string...'); } } }, /** * Checks if a string is a valid JSON object (string encoded) * * @method isJSON * @param {String} str String to check * @return {Boolean} * @public * @static */ isJSON: function(str) { str = str.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, ''); return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str); }, /** * Escapes unsafe html chars as HTML entities * * @method htmlEscapeUnsafe * @param {String} str String to escape * @return {String} Escaped string * @public * @static * @sample Ink_Util_String_htmlEscapeUnsafe.html */ htmlEscapeUnsafe: function(str){ var chars = InkUtilString._htmlUnsafeChars; return str !== null ? String(str).replace(/[<>&'"]/g,function(c){return chars[c];}) : str; }, /** * Normalizes whitespace in string. * String is trimmed and sequences of whitespaces are collapsed. * * @method normalizeWhitespace * @param {String} str String to normalize * @return {String} Normalized string * @public * @static * @sample Ink_Util_String_normalizeWhitespace.html */ normalizeWhitespace: function(str){ return str !== null ? InkUtilString.trim(String(str).replace(/\s+/g,' ')) : str; }, /** * Converts string to unicode. * * @method toUnicode * @param {String} str String to convert * @return {String} Unicoded String * @public * @static * @sample Ink_Util_String_toUnicode.html */ toUnicode: function(str) { if (typeof str === 'string') { var unicodeString = ''; var inInt = false; var theUnicode = false; var total = str.length; var i=0; while(i < total) { inInt = str.charCodeAt(i); if( (inInt >= 32 && inInt <= 126) || inInt === 8 || inInt === 9 || inInt === 10 || inInt === 12 || inInt === 13 || inInt === 32 || inInt === 34 || inInt === 47 || inInt === 58 || inInt === 92) { /* if(inInt == 34 || inInt == 92 || inInt == 47) { theUnicode = '\\'+str.charAt(i); } else { } */ if(inInt === 8) { theUnicode = '\\b'; } else if(inInt === 9) { theUnicode = '\\t'; } else if(inInt === 10) { theUnicode = '\\n'; } else if(inInt === 12) { theUnicode = '\\f'; } else if(inInt === 13) { theUnicode = '\\r'; } else { theUnicode = str.charAt(i); } } else { theUnicode = str.charCodeAt(i).toString(16)+''.toUpperCase(); while (theUnicode.length < 4) { theUnicode = '0' + theUnicode; } theUnicode = '\\u' + theUnicode; } unicodeString += theUnicode; i++; } return unicodeString; } }, /** * Escapes a unicode character. * * @method escape * @param {String} c Character to escape * @return {String} Escaped character. Returns \xXX if hex smaller than 0x100, otherwise \uXXXX * @public * @static * @sample Ink_Util_String_escape.html */ escape: function(c) { var hex = (c).charCodeAt(0).toString(16).split(''); if (hex.length < 3) { while (hex.length < 2) { hex.unshift('0'); } hex.unshift('x'); } else { while (hex.length < 4) { hex.unshift('0'); } hex.unshift('u'); } hex.unshift('\\'); return hex.join(''); }, /** * Unescapes a unicode character escape sequence * * @method unescape * @param {String} es Escape sequence * @return {String} String un-unicoded * @public * @static * @sample Ink_Util_String_unescape.html */ unescape: function(es) { var idx = es.lastIndexOf('0'); idx = idx === -1 ? 2 : Math.min(idx, 2); //console.log(idx); var hexNum = es.substring(idx); //console.log(hexNum); var num = parseInt(hexNum, 16); return String.fromCharCode(num); }, /** * Escapes a string to unicode characters * * @method escapeText * @param {String} txt * @param {Array} [whiteList] Whitelist of characters * @return {String} String escaped to Unicode * @public * @static * @sample Ink_Util_String_escapeText.html */ escapeText: function(txt, whiteList) { if (whiteList === undefined) { whiteList = ['[', ']', '\'', ',']; } var txt2 = []; var c, C; for (var i = 0, f = txt.length; i < f; ++i) { c = txt[i]; C = c.charCodeAt(0); if (C < 32 || C > 126 && whiteList.indexOf(c) === -1) { c = InkUtilString.escape(c); } txt2.push(c); } return txt2.join(''); }, /** * Regex to check escaped strings * * @property escapedCharRegex * @type {Regex} * @public * @readOnly * @static */ escapedCharRegex: /(\\x[0-9a-fA-F]{2})|(\\u[0-9a-fA-F]{4})/g, /** * Unescapes a string * * @method unescapeText * @param {String} txt * @return {String} Unescaped string * @public * @static * @sample Ink_Util_String_unescapeText.html */ unescapeText: function(txt) { /*jshint boss:true */ var m; while (m = InkUtilString.escapedCharRegex.exec(txt)) { m = m[0]; txt = txt.replace(m, InkUtilString.unescape(m)); InkUtilString.escapedCharRegex.lastIndex = 0; } return txt; }, /** * Compares two strings. * * @method strcmp * @param {String} str1 First String * @param {String} str2 Second String * @return {Number} * @public * @static * @sample Ink_Util_String_strcmp.html */ strcmp: function(str1, str2) { return ((str1 === str2) ? 0 : ((str1 > str2) ? 1 : -1)); }, /** * Splits a string into smaller chunks * * @method packetize * @param {String} str String to divide * @param {Number} maxLen Maximum chunk size (in characters) * @return {Array} Chunks of the original string * @public * @static * @sample Ink_Util_String_packetize.html */ packetize: function(str, maxLen) { var len = str.length; var parts = new Array( Math.ceil(len / maxLen) ); var chars = str.split(''); var sz, i = 0; while (len) { sz = Math.min(maxLen, len); parts[i++] = chars.splice(0, sz).join(''); len -= sz; } return parts; } }; return InkUtilString; }); /** * URL Utilities * @module Ink.Util.Url_1 * @version 1 */ Ink.createModule('Ink.Util.Url', '1', [], function() { 'use strict'; /** * @namespace Ink.Util.Url_1 */ var Url = { /** * Auxiliary string for encoding * * @property _keyStr * @type {String} * @readOnly * @private */ _keyStr : 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=', /** * Gets URL of current page * * @method getUrl * @return Current URL * @public * @static * @sample Ink_Util_Url_getUrl.html */ getUrl: function() { return window.location.href; }, /** * Generates an URL string. * * @method genQueryString * @param {String} uri Base URL * @param {Object} params Object to transform to query string * @return {String} URI with query string set * @public * @static * @sample Ink_Util_Url_genQueryString.html */ genQueryString: function(uri, params) { var hasQuestionMark = uri.indexOf('?') !== -1; var sep, pKey, pValue, parts = [uri]; for (pKey in params) { if (params.hasOwnProperty(pKey)) { if (!hasQuestionMark) { sep = '?'; hasQuestionMark = true; } else { sep = '&'; } pValue = params[pKey]; if (typeof pValue !== 'number' && !pValue) { pValue = ''; } parts = parts.concat([sep, encodeURIComponent(pKey), '=', encodeURIComponent(pValue)]); } } return parts.join(''); }, /** * Gets an object from an URL encoded string. * * @method getQueryString * @param {String} [str] URL String. When not specified it uses the current URL. * @return {Object} Key-Value pair object * @public * @static * @sample Ink_Util_Url_getQueryString.html */ getQueryString: function(str) { var url; if(str && typeof(str) !== 'undefined') { url = str; } else { url = this.getUrl(); } var aParams = {}; if(url.match(/\?(.+)/i)) { var queryStr = url.replace(/^(.*)\?([^\#]+)(\#(.*))?/g, "$2"); if(queryStr.length > 0) { var aQueryStr = queryStr.split(/[;&]/); for(var i=0; i < aQueryStr.length; i++) { var pairVar = aQueryStr[i].split('='); aParams[decodeURIComponent(pairVar[0])] = (typeof(pairVar[1]) !== 'undefined' && pairVar[1]) ? decodeURIComponent(pairVar[1]) : false; } } } return aParams; }, /** * Gets the URL hash value * * @method getAnchor * @param {String} [str] URL String. Defaults to current page URL. * @return {String|Boolean} Hash in the URL. If there's no hash, returns false. * @public * @static * @sample Ink_Util_Url_getAnchor.html */ getAnchor: function(str) { var url; if(str && typeof(str) !== 'undefined') { url = str; } else { url = this.getUrl(); } var anchor = false; if(url.match(/#(.+)/)) { anchor = url.replace(/([^#]+)#(.*)/, "$2"); } return anchor; }, /** * Gets the anchor string of an URL * * @method getAnchorString * @param {String} [string] URL to parse. Defaults to current URL. * @return {Object} Key-value pair object of the URL's hashtag 'variables' * @public * @static * @sample Ink_Util_Url_getAnchorString.html */ getAnchorString: function(string) { var url; if(string && typeof(string) !== 'undefined') { url = string; } else { url = this.getUrl(); } var aParams = {}; if(url.match(/#(.+)/i)) { var anchorStr = url.replace(/^([^#]+)#(.*)?/g, "$2"); if(anchorStr.length > 0) { var aAnchorStr = anchorStr.split(/[;&]/); for(var i=0; i < aAnchorStr.length; i++) { var pairVar = aAnchorStr[i].split('='); aParams[decodeURIComponent(pairVar[0])] = (typeof(pairVar[1]) !== 'undefined' && pairVar[1]) ? decodeURIComponent(pairVar[1]) : false; } } } return aParams; }, /** * Parses URL string into URL parts * * @method parseUrl * @param {String} url URL to be parsed * @return {Object} Parsed URL as a key-value object. * @public * @static * @sample Ink_Util_Url_parseUrl.html */ parseUrl: function(url) { var aURL = {}; if(url && typeof url === 'string') { if(url.match(/^([^:]+):\/\//i)) { var re = /^([^:]+):\/\/([^\/]*)\/?([^\?#]*)\??([^#]*)#?(.*)/i; if(url.match(re)) { aURL.scheme = url.replace(re, "$1"); aURL.host = url.replace(re, "$2"); aURL.path = '/'+url.replace(re, "$3"); aURL.query = url.replace(re, "$4") || false; aURL.fragment = url.replace(re, "$5") || false; } } else { var re1 = new RegExp("^([^\\?]+)\\?([^#]+)#(.*)", "i"); var re2 = new RegExp("^([^\\?]+)\\?([^#]+)#?", "i"); var re3 = new RegExp("^([^\\?]+)\\??", "i"); if(url.match(re1)) { aURL.scheme = false; aURL.host = false; aURL.path = url.replace(re1, "$1"); aURL.query = url.replace(re1, "$2"); aURL.fragment = url.replace(re1, "$3"); } else if(url.match(re2)) { aURL.scheme = false; aURL.host = false; aURL.path = url.replace(re2, "$1"); aURL.query = url.replace(re2, "$2"); aURL.fragment = false; } else if(url.match(re3)) { aURL.scheme = false; aURL.host = false; aURL.path = url.replace(re3, "$1"); aURL.query = false; aURL.fragment = false; } } if(aURL.host) { var regPort = /^(.*?)\\:(\\d+)$/i; // check for port if(aURL.host.match(regPort)) { var tmpHost1 = aURL.host; aURL.host = tmpHost1.replace(regPort, "$1"); aURL.port = tmpHost1.replace(regPort, "$2"); } else { aURL.port = false; } // check for user and pass if(aURL.host.match(/@/i)) { var tmpHost2 = aURL.host; aURL.host = tmpHost2.split('@')[1]; var tmpUserPass = tmpHost2.split('@')[0]; if(tmpUserPass.match(/\:/)) { aURL.user = tmpUserPass.split(':')[0]; aURL.pass = tmpUserPass.split(':')[1]; } else { aURL.user = tmpUserPass; aURL.pass = false; } } } } return aURL; }, /** * Formats an URL object into an URL string. * * @method format * @param urlObj Window.location, a.href, or parseUrl object to format * @return {String} Full URL. */ format: function (urlObj) { var protocol = ''; var host = ''; var path = ''; var frag = ''; var query = ''; if (typeof urlObj.protocol === 'string') { protocol = urlObj.protocol + '//'; // here it comes with the colon } else if (typeof urlObj.scheme === 'string') { protocol = urlObj.scheme + '://'; } host = urlObj.host || urlObj.hostname || ''; path = urlObj.path || ''; if (typeof urlObj.query === 'string') { query = urlObj.query; } else if (typeof urlObj.search === 'string') { query = urlObj.search.replace(/^\?/, ''); } if (typeof urlObj.fragment === 'string') { frag = urlObj.fragment; } else if (typeof urlObj.hash === 'string') { frag = urlObj.hash.replace(/#$/, ''); } return [ protocol, host, path, query && '?' + query, frag && '#' + frag ].join(''); }, /** * Gets the last loaded script element * * @method currentScriptElement * @param {String} [match] String to match against the script src attribute * @return {DOMElement|Boolean} Returns the `script` DOM Element or false if unable to find it. * @public * @static * @sample Ink_Util_Url_currentScriptElement.html */ currentScriptElement: function(match) { var aScripts = document.getElementsByTagName('script'); if(typeof(match) === 'undefined') { if(aScripts.length > 0) { return aScripts[(aScripts.length - 1)]; } else { return false; } } else { var curScript = false; var re = new RegExp(""+match+"", "i"); for(var i=0, total = aScripts.length; i < total; i++) { curScript = aScripts[i]; if(re.test(curScript.src)) { return curScript; } } return false; } }, /* base64Encode: function(string) { /** * --function {String} ? * --Convert a string to BASE 64 * @param {String} string - string to convert * @return base64 encoded string * * if(!SAPO.Utility.String || typeof(SAPO.Utility.String) === 'undefined') { throw "SAPO.Utility.Url.base64Encode depends of SAPO.Utility.String, which has not been referred."; } var output = ""; var chr1, chr2, chr3, enc1, enc2, enc3, enc4; var i = 0; var input = SAPO.Utility.String.utf8Encode(string); while (i < input.length) { chr1 = input.charCodeAt(i++); chr2 = input.charCodeAt(i++); chr3 = input.charCodeAt(i++); enc1 = chr1 >> 2; enc2 = ((chr1 & 3) << 4) | (chr2 >> 4); enc3 = ((chr2 & 15) << 2) | (chr3 >> 6); enc4 = chr3 & 63; if (isNaN(chr2)) { enc3 = enc4 = 64; } else if (isNaN(chr3)) { enc4 = 64; } output = output + this._keyStr.charAt(enc1) + this._keyStr.charAt(enc2) + this._keyStr.charAt(enc3) + this._keyStr.charAt(enc4); } return output; }, base64Decode: function(string) { * --function {String} ? * Decode a BASE 64 encoded string * --param {String} string base64 encoded string * --return string decoded if(!SAPO.Utility.String || typeof(SAPO.Utility.String) === 'undefined') { throw "SAPO.Utility.Url.base64Decode depends of SAPO.Utility.String, which has not been referred."; } var output = ""; var chr1, chr2, chr3; var enc1, enc2, enc3, enc4; var i = 0; var input = string.replace(/[^A-Za-z0-9\+\/\=]/g, ""); while (i < input.length) { enc1 = this._keyStr.indexOf(input.charAt(i++)); enc2 = this._keyStr.indexOf(input.charAt(i++)); enc3 = this._keyStr.indexOf(input.charAt(i++)); enc4 = this._keyStr.indexOf(input.charAt(i++)); chr1 = (enc1 << 2) | (enc2 >> 4); chr2 = ((enc2 & 15) << 4) | (enc3 >> 2); chr3 = ((enc3 & 3) << 6) | enc4; output = output + String.fromCharCode(chr1); if (enc3 !== 64) { output = output + String.fromCharCode(chr2); } if (enc4 !== 64) { output = output + String.fromCharCode(chr3); } } output = SAPO.Utility.String.utf8Decode(output); return output; }, */ /** * Debug function ? * * @method _debug * @private * @static */ _debug: function() {} }; return Url; }); /** * Validation Utilities * @module Ink.Util.Validator_1 * @version 1 */ Ink.createModule('Ink.Util.Validator', '1', [], function() { 'use strict'; /** * @namespace Ink.Util.Validator_1 */ var Validator = { /** * List of country codes avaible for the isPhone method * * @property _countryCodes * @type {Array} * @private * @static * @readOnly */ _countryCodes : [ 'AO', 'CV', 'MZ', 'PT' ], /** * International number for portugal * * @property _internacionalPT * @type {Number} * @private * @static * @readOnly * */ _internacionalPT: 351, /** * List of all portuguese number prefixes * * @property _indicativosPT * @type {Object} * @private * @static * @readOnly * */ _indicativosPT: { 21: 'lisboa', 22: 'porto', 231: 'mealhada', 232: 'viseu', 233: 'figueira da foz', 234: 'aveiro', 235: 'arganil', 236: 'pombal', 238: 'seia', 239: 'coimbra', 241: 'abrantes', 242: 'ponte de sôr', 243: 'santarém', 244: 'leiria', 245: 'portalegre', 249: 'torres novas', 251: 'valença', 252: 'vila nova de famalicão', 253: 'braga', 254: 'peso da régua', 255: 'penafiel', 256: 'são joão da madeira', 258: 'viana do castelo', 259: 'vila real', 261: 'torres vedras', 262: 'caldas da raínha', 263: 'vila franca de xira', 265: 'setúbal', 266: 'évora', 268: 'estremoz', 269: 'santiago do cacém', 271: 'guarda', 272: 'castelo branco', 273: 'bragança', 274: 'proença-a-nova', 275: 'covilhã', 276: 'chaves', 277: 'idanha-a-nova', 278: 'mirandela', 279: 'moncorvo', 281: 'tavira', 282: 'portimão', 283: 'odemira', 284: 'beja', 285: 'moura', 286: 'castro verde', 289: 'faro', 291: 'funchal, porto santo', 292: 'corvo, faial, flores, horta, pico', 295: 'angra do heroísmo, graciosa, são jorge, terceira', 296: 'ponta delgada, são miguel, santa maria', 91 : 'rede móvel 91 (Vodafone / Yorn)', 93 : 'rede móvel 93 (Optimus)', 96 : 'rede móvel 96 (TMN)', 92 : 'rede móvel 92 (TODOS)', //925 : 'rede móvel 925 (TMN 925)', //926 : 'rede móvel 926 (TMN 926)', //927 : 'rede móvel 927 (TMN 927)', //922 : 'rede móvel 922 (Phone-ix)', 707: 'número único', 760: 'número único', 800: 'número grátis', 808: 'chamada local', 30: 'voip' }, /** * International number for Cabo Verde * * @property _internacionalCV * @type {Number} * @private * @static * @readOnly */ _internacionalCV: 238, /** * List of all Cabo Verde number prefixes * * @property _indicativosCV * @type {Object} * @private * @static * @readOnly */ _indicativosCV: { 2: 'fixo', 91: 'móvel 91', 95: 'móvel 95', 97: 'móvel 97', 98: 'móvel 98', 99: 'móvel 99' }, /** * International number for Angola * * @property _internacionalAO * @type {Number} * @private * @static * @readOnly */ _internacionalAO: 244, /** * List of all Angola number prefixes * * @property _indicativosAO * @type {Object} * @private * @static * @readOnly */ _indicativosAO: { 2: 'fixo', 91: 'móvel 91', 92: 'móvel 92' }, /** * International number for Mozambique * * @property _internacionalMZ * @type {Number} * @private * @static * @readOnly */ _internacionalMZ: 258, /** * List of all Mozambique number prefixes * * @property _indicativosMZ * @type {Object} * @private * @static * @readOnly */ _indicativosMZ: { 2: 'fixo', 82: 'móvel 82', 84: 'móvel 84' }, /** * International number for Timor * * @property _internacionalTL * @type {Number} * @private * @static * @readOnly */ _internacionalTL: 670, /** * List of all Timor number prefixes * * @property _indicativosTL * @type {Object} * @private * @static * @readOnly */ _indicativosTL: { 3: 'fixo', 7: 'móvel 7' }, /** * Regular expression groups for several groups of characters * * http://en.wikipedia.org/wiki/C0_Controls_and_Basic_Latin * http://en.wikipedia.org/wiki/Plane_%28Unicode%29#Basic_Multilingual_Plane * http://en.wikipedia.org/wiki/ISO_8859-1 * * @property _characterGroups * @type {Object} * @private * @static * @readOnly */ _characterGroups: { numbers: ['0-9'], asciiAlpha: ['a-zA-Z'], latin1Alpha: ['a-zA-Z', '\u00C0-\u00FF'], unicodeAlpha: ['a-zA-Z', '\u00C0-\u00FF', '\u0100-\u1FFF', '\u2C00-\uD7FF'], /* whitespace characters */ space: [' '], dash: ['-'], underscore: ['_'], nicknamePunctuation: ['_.-'], singleLineWhitespace: ['\t '], newline: ['\n'], whitespace: ['\t\n\u000B\f\r\u00A0 '], asciiPunctuation: ['\u0021-\u002F', '\u003A-\u0040', '\u005B-\u0060', '\u007B-\u007E'], latin1Punctuation: ['\u0021-\u002F', '\u003A-\u0040', '\u005B-\u0060', '\u007B-\u007E', '\u00A1-\u00BF', '\u00D7', '\u00F7'], unicodePunctuation: ['\u0021-\u002F', '\u003A-\u0040', '\u005B-\u0060', '\u007B-\u007E', '\u00A1-\u00BF', '\u00D7', '\u00F7', '\u2000-\u206F', '\u2E00-\u2E7F', '\u3000-\u303F'] }, /** * Creates a regular expression for several character groups. * * @method createRegExp * * @param Groups* {Object} * Groups to build regular expressions for. Possible keys are: * * - **numbers**: 0-9 * - **asciiAlpha**: a-z, A-Z * - **latin1Alpha**: asciiAlpha, plus printable characters in latin-1 * - **unicodeAlpha**: unicode alphanumeric characters. * - **space**: ' ', the space character. * - **dash**: dash character. * - **underscore**: underscore character. * - **nicknamePunctuation**: dash, dot, underscore * - **singleLineWhitespace**: space and tab (whitespace which only spans one line). * - **newline**: newline character ('\n') * - **whitespace**: whitespace characters in the ASCII character set. * - **asciiPunctuation**: punctuation characters in the ASCII character set. * - **latin1Punctuation**: punctuation characters in latin-1. * - **unicodePunctuation**: punctuation characters in unicode. * */ createRegExp: function (groups) { var re = '^['; for (var key in groups) if (groups.hasOwnProperty(key)) { if (!(key in Validator._characterGroups)) { throw new Error('group ' + key + ' is not a valid character group'); } else if (groups[key]) { re += Validator._characterGroups[key].join(''); } } if (re === '^[') { // No changes return new RegExp('$^'); // match nothing } return new RegExp(re + ']*?$'); }, /** * Checks if a field has the required groups. * * @method checkCharacterGroups * @param {String} s The validation string * @param {Object} [groups]={} What groups are included. See createRegexp * @sample Ink_Util_Validator_checkCharacterGroups.html */ checkCharacterGroups: function (s, groups) { return Validator.createRegExp(groups).test(s); }, /** * Checks if a field contains unicode printable characters. * * @method unicode * @param {String} s The validation string * @param {Object} [options]={} Optional configuration object. See createRegexp */ unicode: function (s, options) { return Validator.checkCharacterGroups(s, Ink.extendObj({ unicodeAlpha: true}, options)); }, /** * Checks if a field only contains latin-1 alphanumeric characters. * Takes options for allowing singleline whitespace, cross-line whitespace and punctuation. * * @method latin1 * * @param {String} s The validation string * @param {Object} [options]={} Optional configuration object. See createRegexp * @sample Ink_Util_Validator_latin1.html */ latin1: function (s, options) { return Validator.checkCharacterGroups(s, Ink.extendObj({ latin1Alpha: true}, options)); }, /** * Checks if a field only contains only ASCII alphanumeric characters. * Takes options for allowing singleline whitespace, cross-line whitespace and punctuation. * * @method ascii * * @param {String} s The validation string * @param {Object} [options]={} Optional configuration object. See createRegexp * @sample Ink_Util_Validator_ascii.html */ ascii: function (s, options) { return Validator.checkCharacterGroups(s, Ink.extendObj({ asciiAlpha: true}, options)); }, /** * Checks if a number is a valid * * @method number * @param {String} numb The number * @param {Object} [options] Further options * @param [options.decimalSep]='.' Allow decimal separator. * @param [options.thousandSep]="," Strip this character from the number. * @param [options.negative]=false Allow negative numbers. * @param [options.decimalPlaces]=null Maximum number of decimal places. Use `0` for an integer number. * @param [options.max]=null Maximum number * @param [options.min]=null Minimum number * @param [options.returnNumber]=false When this option is true, return the number itself when the value is valid. * @sample Ink_Util_Validator_number.html */ number: function (numb, inOptions) { numb = numb + ''; var options = Ink.extendObj({ decimalSep: '.', thousandSep: '', negative: true, decimalPlaces: null, maxDigits: null, max: null, min: null, returnNumber: false }, inOptions || {}); // smart recursion thing sets up aliases for options. if (options.thousandSep) { numb = numb.replace(new RegExp('\\' + options.thousandSep, 'g'), ''); options.thousandSep = ''; return Validator.number(numb, options); } if (options.negative === false) { options.min = 0; options.negative = true; return Validator.number(numb, options); } if (options.decimalSep !== '.') { numb = numb.replace(new RegExp('\\' + options.decimalSep, 'g'), '.'); } if (!/^(-)?(\d+)?(\.\d+)?$/.test(numb) || numb === '') { return false; // forbidden character found } var split; if (options.decimalSep && numb.indexOf(options.decimalSep) !== -1) { split = numb.split(options.decimalSep); if (options.decimalPlaces !== null && split[1].length > options.decimalPlaces) { return false; } } else { split = ['' + numb, '']; } if (options.maxDigits!== null) { if (split[0].replace(/-/g, '').length > options.maxDigits) { return split; } } // Now look at the actual float var ret = parseFloat(numb); if (options.maxExcl !== null && ret >= options.maxExcl || options.minExcl !== null && ret <= options.minExcl) { return false; } if (options.max !== null && ret > options.max || options.min !== null && ret < options.min) { return false; } if (options.returnNumber) { return ret; } else { return true; } }, /** * Checks if a year is Leap "Bissexto" * * @method _isLeapYear * @param {Number} year Year to be checked * @return {Boolean} True if it is a leap year. * @private * @static * @example * Ink.requireModules(['Ink.Util.Validator_1'], function( InkValidator ){ * console.log( InkValidator._isLeapYear( 2004 ) ); // Result: true * console.log( InkValidator._isLeapYear( 2006 ) ); // Result: false * }); */ _isLeapYear: function(year){ var yearRegExp = /^\d{4}$/; if(yearRegExp.test(year)){ return ((year%4) ? false: ((year%100) ? true : ((year%400)? false : true)) ); } return false; }, /** * Object with the date formats available for validation * * @property _dateParsers * @type {Object} * @private * @static * @readOnly */ _dateParsers: { 'yyyy-mm-dd': {day:5, month:3, year:1, sep: '-', parser: /^(\d{4})(\-)(\d{1,2})(\-)(\d{1,2})$/}, 'yyyy/mm/dd': {day:5, month:3, year:1, sep: '/', parser: /^(\d{4})(\/)(\d{1,2})(\/)(\d{1,2})$/}, 'yy-mm-dd': {day:5, month:3, year:1, sep: '-', parser: /^(\d{2})(\-)(\d{1,2})(\-)(\d{1,2})$/}, 'yy/mm/dd': {day:5, month:3, year:1, sep: '/', parser: /^(\d{2})(\/)(\d{1,2})(\/)(\d{1,2})$/}, 'dd-mm-yyyy': {day:1, month:3, year:5, sep: '-', parser: /^(\d{1,2})(\-)(\d{1,2})(\-)(\d{4})$/}, 'dd/mm/yyyy': {day:1, month:3, year:5, sep: '/', parser: /^(\d{1,2})(\/)(\d{1,2})(\/)(\d{4})$/}, 'dd-mm-yy': {day:1, month:3, year:5, sep: '-', parser: /^(\d{1,2})(\-)(\d{1,2})(\-)(\d{2})$/}, 'dd/mm/yy': {day:1, month:3, year:5, sep: '/', parser: /^(\d{1,2})(\/)(\d{1,2})(\/)(\d{2})$/} }, /** * Gets the number of days in a given month of a given year * * @method _daysInMonth * @param {Number} _m Month (1 to 12) * @param {Number} _y Year * @return {Number} Returns the number of days in a given month of a given year * @private * @static * @example * Ink.requireModules(['Ink.Util.Validator_1'], function( InkValidator ){ * console.log( InkValidator._daysInMonth( 2, 2004 ) ); // Result: 29 * console.log( InkValidator._daysInMonth( 2, 2006 ) ); // Result: 28 * }); */ _daysInMonth: function(_m,_y){ var nDays=0; _m = parseInt(_m, 10); _y = parseInt(_y, 10); if(_m===1 || _m===3 || _m===5 || _m===7 || _m===8 || _m===10 || _m===12) { nDays= 31; } else if ( _m===4 || _m===6 || _m===9 || _m===11) { nDays = 30; } else if (_m===2) { if((_y%400===0) || (_y%4===0 && _y%100!==0)) { nDays = 29; } else { nDays = 28; } } return nDays; }, /** * Checks if a date is valid * * @method _isValidDate * @param {Number} year * @param {Number} month * @param {Number} day * @return {Boolean} True if valid * @private * @static * @example * Ink.requireModules(['Ink.Util.Validator_1'], function( InkValidator ){ * console.log( InkValidator._isValidDate( 2004, 2, 29 ) ); // Result: true * console.log( InkValidator._isValidDate( 2006, 2, 29 ) ); // Result: false * }); */ _isValidDate: function(year, month, day){ var yearRegExp = /^\d{4}$/; var validOneOrTwo = /^\d{1,2}$/; if(yearRegExp.test(year) && validOneOrTwo.test(month) && validOneOrTwo.test(day)){ if(month>=1 && month<=12 && day>=1 && this._daysInMonth(month,year)>=day){ return true; } } return false; }, /** * Checks if an email is valid * * @method mail * @param {String} email * @return {Boolean} True if it's valid * @public * @static * @sample Ink_Util_Validator_mail.html */ email: function(email) { var emailValido = new RegExp("^[_a-z0-9-]+((\\.|\\+)[_a-z0-9-]+)*@([\\w]*-?[\\w]*\\.)+[a-z]{2,4}$", "i"); if(!emailValido.test(email)) { return false; } else { return true; } }, /** * Deprecated. Alias for email(). Use it instead. * * @method mail * @public * @static * @private */ mail: function (mail) { return Validator.email(mail); }, /** * Checks if an url is valid * * @method url * @param {String} url URL to be checked * @param {Boolean} [full] If true, validates a full URL (one that should start with 'http') * @return {Boolean} True if valid * @public * @static * @sample Ink_Util_Validator_url.html */ url: function(url, full) { if(typeof full === "undefined" || full === false) { var reHTTP = new RegExp("(^(http\\:\\/\\/|https\\:\\/\\/)(.+))", "i"); if(reHTTP.test(url) === false) { url = 'http://'+url; } } var reUrl = new RegExp("^(http:\\/\\/|https:\\/\\/)([\\w]*(-?[\\w]*)*\\.)+[a-z]{2,4}", "i"); if(reUrl.test(url) === false) { return false; } else { return true; } }, /** * Checks if a phone is valid in Portugal * * @method isPTPhone * @param {Number} phone Phone number to be checked * @return {Boolean} True if it's a valid Portuguese Phone * @public * @static * @sample Ink_Util_Validator_isPTPhone.html */ isPTPhone: function(phone) { phone = phone.toString(); var aInd = []; for(var i in this._indicativosPT) { if(typeof(this._indicativosPT[i]) === 'string') { aInd.push(i); } } var strInd = aInd.join('|'); var re351 = /^(00351|\+351)/; if(re351.test(phone)) { phone = phone.replace(re351, ""); } var reSpecialChars = /(\s|\-|\.)+/g; phone = phone.replace(reSpecialChars, ''); //var reInt = new RegExp("\\d", "i"); var reInt = /[\d]{9}/i; if(phone.length === 9 && reInt.test(phone)) { var reValid = new RegExp("^("+strInd+")"); if(reValid.test(phone)) { return true; } } return false; }, /** * Alias function for isPTPhone * * @method isPortuguesePhone * @param {Number} phone Phone number to be checked * @return {Boolean} True if it's a valid Portuguese Phone * @public * @static */ isPortuguesePhone: function(phone) { return this.isPTPhone(phone); }, /** * Checks if a phone is valid in Cabo Verde * * @method isCVPhone * @param {Number} phone Phone number to be checked * @return {Boolean} True if it's a valid Cape Verdean Phone * @public * @static * @sample Ink_Util_Validator_isCVPhone.html */ isCVPhone: function(phone) { phone = phone.toString(); var aInd = []; for(var i in this._indicativosCV) { if(typeof(this._indicativosCV[i]) === 'string') { aInd.push(i); } } var strInd = aInd.join('|'); var re238 = /^(00238|\+238)/; if(re238.test(phone)) { phone = phone.replace(re238, ""); } var reSpecialChars = /(\s|\-|\.)+/g; phone = phone.replace(reSpecialChars, ''); //var reInt = new RegExp("\\d", "i"); var reInt = /[\d]{7}/i; if(phone.length === 7 && reInt.test(phone)) { var reValid = new RegExp("^("+strInd+")"); if(reValid.test(phone)) { return true; } } return false; }, /** * Checks if a phone is valid in Angola * * @method isAOPhone * @param {Number} phone Phone number to be checked * @return {Boolean} True if it's a valid Angolan Phone * @public * @static * @sample Ink_Util_Validator_isAOPhone.html */ isAOPhone: function(phone) { phone = phone.toString(); var aInd = []; for(var i in this._indicativosAO) { if(typeof(this._indicativosAO[i]) === 'string') { aInd.push(i); } } var strInd = aInd.join('|'); var re244 = /^(00244|\+244)/; if(re244.test(phone)) { phone = phone.replace(re244, ""); } var reSpecialChars = /(\s|\-|\.)+/g; phone = phone.replace(reSpecialChars, ''); //var reInt = new RegExp("\\d", "i"); var reInt = /[\d]{9}/i; if(phone.length === 9 && reInt.test(phone)) { var reValid = new RegExp("^("+strInd+")"); if(reValid.test(phone)) { return true; } } return false; }, /** * Checks if a phone is valid in Mozambique * * @method isMZPhone * @param {Number} phone Phone number to be checked * @return {Boolean} True if it's a valid Mozambican Phone * @public * @static * @sample Ink_Util_Validator_isMZPhone.html */ isMZPhone: function(phone) { phone = phone.toString(); var aInd = []; for(var i in this._indicativosMZ) { if(typeof(this._indicativosMZ[i]) === 'string') { aInd.push(i); } } var strInd = aInd.join('|'); var re258 = /^(00258|\+258)/; if(re258.test(phone)) { phone = phone.replace(re258, ""); } var reSpecialChars = /(\s|\-|\.)+/g; phone = phone.replace(reSpecialChars, ''); //var reInt = new RegExp("\\d", "i"); var reInt = /[\d]{8,9}/i; if((phone.length === 9 || phone.length === 8) && reInt.test(phone)) { var reValid = new RegExp("^("+strInd+")"); if(reValid.test(phone)) { if(phone.indexOf('2') === 0 && phone.length === 8) { return true; } else if(phone.indexOf('8') === 0 && phone.length === 9) { return true; } } } return false; }, /** * Checks if a phone is valid in Timor * * @method isTLPhone * @param {Number} phone Phone number to be checked * @return {Boolean} True if it's a valid phone from Timor-Leste * @public * @static * @sample Ink_Util_Validator_isTLPhone.html */ isTLPhone: function(phone) { phone = phone.toString(); var aInd = []; for(var i in this._indicativosTL) { if(typeof(this._indicativosTL[i]) === 'string') { aInd.push(i); } } var strInd = aInd.join('|'); var re670 = /^(00670|\+670)/; if(re670.test(phone)) { phone = phone.replace(re670, ""); } var reSpecialChars = /(\s|\-|\.)+/g; phone = phone.replace(reSpecialChars, ''); //var reInt = new RegExp("\\d", "i"); var reInt = /[\d]{7}/i; if(phone.length === 7 && reInt.test(phone)) { var reValid = new RegExp("^("+strInd+")"); if(reValid.test(phone)) { return true; } } return false; }, /** * Checks if a number is a phone number. * This method validates the number in all country codes available the ones set in the second param * * @method isPhone * @param {String} phone Phone number to validate * @param {String|Array} [countryCode] Country code or array of countries to validate * @return {Boolean} True if it's a valid phone in any country available * @public * @static * @sample Ink_Util_Validator_isPhone.html */ isPhone: function(){ var index; if(arguments.length===0){ return false; } var phone = arguments[0]; if(arguments.length>1){ if(arguments[1].constructor === Array){ var func; for(index=0; index= 0 && match[i-1] <= 100){ valid = true; } else { return false; } } // check 0 to 255 values if(i===1 || i===3 || i===5 && (typeof match[i+1] === "undefined" || match[i+1] === "")){ if(typeof match[i] !== "undefined" && match[i] >= 0 && match[i] <= 255){ valid = true; } else { return false; } } } } // hsl range check if((match = hsl.exec(str)) !== null || (match = hsla.exec(str)) !== null){ i = match.length; while(i--){ // check percentage values if(i===3 || i===5){ if(typeof match[i-1] !== "undefined" && typeof match[i] !== "undefined" && match[i] !== "" && match[i-1] >= 0 && match[i-1] <= 100){ valid = true; } else { return false; } } // check 0 to 360 value if(i===1){ if(typeof match[i] !== "undefined" && match[i] >= 0 && match[i] <= 360){ valid = true; } else { return false; } } } } return valid; }, /** * Checks if the value is a valid IP. * * @method isIP * @param {String} value Value to be checked * @param {String} ipType Type of IP to be validated. The values are: ipv4, ipv6. By default is ipv4. * @return {Boolean} True if the value is a valid IP address. False if not. * @sample Ink_Util_Validator_isIP.html */ isIP: function( value, ipType ){ if( typeof value !== 'string' ){ return false; } ipType = (ipType || 'ipv4').toLowerCase(); switch( ipType ){ case 'ipv4': return (/^(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)$/).test(value); case 'ipv6': return (/^\s*((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)(\.(25[0-5]|2[0-4]\d|1\d\d|[1-9]?\d)){3}))|:)))(%.+)?\s*$/).test(value); default: return false; } }, /** * Credit Card specifications, to be used in the credit card verification. * * @property _creditCardSpecs * @type {Object} * @private */ _creditCardSpecs: { 'default': { 'length': '13,14,15,16,17,18,19', 'prefix': /^.+/, 'luhn': true }, 'american express': { 'length': '15', 'prefix': /^3[47]/, 'luhn' : true }, 'diners club': { 'length': '14,16', 'prefix': /^36|55|30[0-5]/, 'luhn' : true }, 'discover': { 'length': '16', 'prefix': /^6(?:5|011)/, 'luhn' : true }, 'jcb': { 'length': '15,16', 'prefix': /^3|1800|2131/, 'luhn' : true }, 'maestro': { 'length': '16,18', 'prefix': /^50(?:20|38)|6(?:304|759)/, 'luhn' : true }, 'mastercard': { 'length': '16', 'prefix': /^5[1-5]/, 'luhn' : true }, 'visa': { 'length': '13,16', 'prefix': /^4/, 'luhn' : true } }, /** * Luhn function, to be used when validating credit cards * */ _luhn: function (num){ num = parseInt(num,10); if ( (typeof num !== 'number') && (num % 1 !== 0) ){ // Luhn can only be used on nums! return false; } num = num+''; // Check num length var length = num.length; // Checksum of the card num var i, checksum = 0 ; for (i = length - 1; i >= 0; i -= 2) { // Add up every 2nd digit, starting from the right checksum += parseInt(num.substr(i, 1),10); } for (i = length - 2; i >= 0; i -= 2) { // Add up every 2nd digit doubled, starting from the right var dbl = parseInt(num.substr(i, 1) * 2,10); // Subtract 9 from the dbl where value is greater than 10 checksum += (dbl >= 10) ? (dbl - 9) : dbl; } // If the checksum is a multiple of 10, the number is valid return (checksum % 10 === 0); }, /** * Checks if a number is of a specific credit card type * @method isCreditCard * @param {String} num Number to be validates * @param {String|Array} creditCardType Credit card type. See _creditCardSpecs for the list of supported values. * @return {Boolean} * @sample Ink_Util_Validator_isCreditCard.html */ isCreditCard: function(num, creditCardType){ if ( /\d+/.test(num) === false ){ return false; } if ( typeof creditCardType === 'undefined' ){ creditCardType = 'default'; } else if ( creditCardType instanceof Array ){ var i, ccLength = creditCardType.length; for ( i=0; i < ccLength; i++ ){ // Test each type for validity if (this.isCreditCard(num, creditCardType[i]) ){ return true; } } return false; } // Check card type creditCardType = creditCardType.toLowerCase(); if ( typeof this._creditCardSpecs[creditCardType] === 'undefined' ){ return false; } // Check card number length var length = num.length+''; // Validate the card length by the card type if ( this._creditCardSpecs[creditCardType]['length'].split(",").indexOf(length) === -1 ){ return false; } // Check card number prefix if ( !this._creditCardSpecs[creditCardType]['prefix'].test(num) ){ return false; } // No Luhn check required if (this._creditCardSpecs[creditCardType]['luhn'] === false){ return true; } return this._luhn(num); } }; return Validator; }); /** * Animate.css Utility * * This module is a wrapper around animate.css's CSS classes to produce animation. * It contains options to ease common tasks, like listen to the "animationend" event with all necessary prefixes, remove the necessary class names when the animation finishes, or configure the duration of your animation with the necessary browser prefix. * * @module Ink.UI.Animate_1 * @version 1 */ Ink.createModule('Ink.UI.Animate', 1, ['Ink.UI.Common_1', 'Ink.Dom.Event_1', 'Ink.Dom.Css_1'], function (Common, InkEvent, Css) { 'use strict'; var animationPrefix = (function (el) { return ('animationName' in el.style) ? 'animation' : ('oAnimationName' in el.style) ? 'oAnimation' : ('msAnimationName' in el.style) ? 'msAnimation' : ('webkitAnimationName' in el.style) ? 'webkitAnimation' : null; }(document.createElement('div'))); var animationEndEventName = { animation: 'animationend', oAnimation: 'oanimationend', msAnimation: 'MSAnimationEnd', webkitAnimation: 'webkitAnimationEnd' }[animationPrefix]; /** * @class Ink.UI.Animate_1 * @constructor * * @param {DOMElement} element Animated element * @param {Object} options Options object * @param {String} options.animation Animation name * @param {String|Number} [options.duration] Duration name (fast|medium|slow) or duration in milliseconds. Defaults to 'medium'. * @param {Boolean} [options.removeClass] Flag to remove the CSS class when finished animating. Defaults to false. * @param {Function} [options.onEnd] Callback for the animation end * * @sample Ink_UI_Animate_1.html * **/ function Animate() { Common.BaseUIComponent.apply(this, arguments); } Animate._name = 'Animate_1'; Animate._optionDefinition = { trigger: ['Element', null], duration: ['String', 'slow'], // Actually a string with a duration name, or a number of ms animation: ['String'], removeClass: ['Boolean', true], onEnd: ['Function', function () {}] }; Animate.prototype._init = function () { if (!isNaN(parseInt(this._options.duration, 10))) { this._options.duration = parseInt(this._options.duration, 10); } if (this._options.trigger) { InkEvent.observe(this._options.trigger, 'click', Ink.bind(function () { this.animate(); }, this)); // later } else { this.animate(); } }; Animate.prototype.animate = function () { Animate.animate(this._element, this._options.animation, this._options); }; Ink.extendObj(Animate, { /** * Browser prefix for the CSS animations. * * @property _animationPrefix * @private **/ _animationPrefix: animationPrefix, /** * Boolean which says whether this browser has CSS3 animation support. * * @property animationSupported **/ animationSupported: !!animationPrefix, /** * Prefixed 'animationend' event name. * * @property animationEndEventName **/ animationEndEventName: animationEndEventName, /** * Animate an element using one of the animate.css classes * * **Note: This is a utility method inside the `Animate` class, which you can access through `Animate.animate()`. Do not mix these up.** * * @static * @method animate * @param element {DOMElement} animated element * @param animation {String} animation name * @param [options] {Object} * @param [options.onEnd=null] {Function} callback for animation end * @param [options.removeClass=false] {Boolean} whether to remove the Css class when finished * @param [options.duration=medium] {String|Number} duration name (fast|medium|slow) or duration in ms * * @sample Ink_UI_Animate_1_animate.html **/ animate: function (element, animation, options) { element = Common.elOrSelector(element); if (typeof options === 'number' || typeof options === 'string') { options = { duration: options }; } else if (!options) { options = {}; } if (typeof arguments[3] === 'function') { options.onEnd = arguments[3]; } if (typeof options.duration !== 'number' && typeof options.duration !== 'string') { options.duration = 400; } if (!Animate.animationSupported) { if (options.onEnd) { setTimeout(function () { options.onEnd(null); }, 0); } return; } if (typeof options.duration === 'number') { element.style[animationPrefix + 'Duration'] = options.duration + 'ms'; } else if (typeof options.duration === 'string') { Css.addClassName(element, options.duration); } Css.addClassName(element, ['animated', animation]); function onAnimationEnd(event) { if (event.target !== element) { return; } if (event.animationName !== animation) { return; } if (options.onEnd) { options.onEnd(event); } if (options.removeClass) { Css.removeClassName(element, animation); } if (typeof options.duration === 'string') { Css.removeClassName(element, options.duration); } element.removeEventListener(animationEndEventName, onAnimationEnd, false); } element.addEventListener(animationEndEventName, onAnimationEnd, false); } }); Common.createUIComponent(Animate); return Animate; }); /** * Flexible Carousel * @module Ink.UI.Carousel_1 * @version 1 */ Ink.createModule('Ink.UI.Carousel', '1', ['Ink.UI.Common_1', 'Ink.Dom.Event_1', 'Ink.Dom.Css_1', 'Ink.Dom.Element_1', 'Ink.UI.Pagination_1', 'Ink.Dom.Browser_1', 'Ink.Dom.Selector_1'], function(Common, InkEvent, Css, InkElement, Pagination, Browser/*, Selector*/) { 'use strict'; /* * TODO: * keyboardSupport */ function limitRange(n, min, max) { return Math.min(max, Math.max(min, n)); } var requestAnimationFrame = window.requestAnimationFrame || window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || function (cb) {return setTimeout(cb, 1000 / 30); }; /** * @class Ink.UI.Carousel_1 * @constructor * * @param {String|DOMElement} selector DOM element or element id * @param {Object} [options] Carousel Options * @param {Integer} [options.autoAdvance] Milliseconds to wait before auto-advancing pages. Set to 0 to disable auto-advance. Defaults to 0. * @param {String} [options.axis] Axis of the carousel. Set to 'y' for a vertical carousel. Defaults to 'x'. * @param {Number} [options.initialPage] Initial index page of the carousel. Defaults to 0. * @param {Boolean} [options.spaceAfterLastSlide=true] If there are not enough slides to fill the full width of the last page, leave white space. Defaults to `true`. * @param {Boolean} [options.swipe] Enable swipe support if available. Defaults to true. * @param {Mixed} [options.pagination] Either an ul element to add pagination markup to or an `Ink.UI.Pagination` instance to use. * @param {Function} [options.onChange] Callback to be called when the page changes. * * @sample Ink_UI_Carousel_1.html */ function Carousel() { Common.BaseUIComponent.apply(this, arguments); } Carousel._name = 'Carousel_1'; Carousel._optionDefinition = { autoAdvance: ['Integer', 0], axis: ['String', 'x'], initialPage: ['Integer', 0], spaceAfterLastSlide: ['Boolean', true], hideLast: ['Boolean', false], // [3.1.0] Deprecate "center". It is only needed when things are of unknown widths. center: ['Boolean', false], keyboardSupport:['Boolean', false], pagination: ['String', null], onChange: ['Function', null], onInit: ['Function', function () {}], swipe: ['Boolean', true] // TODO exponential swipe // TODO specify break point for next page when moving finger }; Carousel.prototype = { _init: function () { this._handlers = { paginationChange: Ink.bindMethod(this, '_onPaginationChange'), windowResize: InkEvent.throttle(Ink.bindMethod(this, 'refit'), 200) }; InkEvent.observe(window, 'resize', this._handlers.windowResize); this._isY = (this._options.axis === 'y'); var ulEl = Ink.s('ul.stage', this._element); this._ulEl = ulEl; InkElement.removeTextNodeChildren(ulEl); if (this._options.pagination == null) { this._currentPage = this._options.initialPage; } this.refit(); // recalculate this._numPages if (this._isY) { // Override white-space: no-wrap which is only necessary to make sure horizontal stuff stays horizontal, but breaks stuff intended to be vertical. this._ulEl.style.whiteSpace = 'normal'; } if (this._options.swipe) { InkEvent.observe(this._element, 'touchstart', Ink.bindMethod(this, '_onTouchStart')); InkEvent.observe(this._element, 'touchmove', Ink.bindMethod(this, '_onTouchMove')); InkEvent.observe(this._element, 'touchend', Ink.bindMethod(this, '_onTouchEnd')); } this._setUpPagination(); this._setUpAutoAdvance(); this._setUpHider(); this._options.onInit.call(this, this); }, /** * Repositions elements around. * Measure the carousel once again, adjusting the involved elements' sizes. This is called automatically when the window resizes, in order to cater for changes from responsive media queries, for instance. * * @method refit * @public */ refit: function() { var _isY = this._isY; var size = function (elm, perpendicular) { if (!elm) { return 0; } if (!perpendicular) { return InkElement.outerDimensions(elm)[_isY ? 1 : 0]; } else { return InkElement.outerDimensions(elm)[_isY ? 0 : 1]; } }; this._liEls = Ink.ss('li.slide', this._ulEl); var numSlides = this._liEls.length; var contRect = this._ulEl.getBoundingClientRect(); this._ctnLength = _isY ? contRect.bottom - contRect.top : contRect.right - contRect.left; this._elLength = size(this._liEls[0]); this._slidesPerPage = Math.floor( this._ctnLength / this._elLength ) || 1; if (!isFinite(this._slidesPerPage)) { this._slidesPerPage = 1; } var numPages = Math.ceil( numSlides / this._slidesPerPage ); var numPagesChanged = this._numPages !== numPages; this._numPages = numPages; this._deltaLength = this._slidesPerPage * this._elLength; this._center(); this._updateHider(); this._IE7(); if (this._pagination && numPagesChanged) { this._pagination.setSize(this._numPages); } this.setPage(limitRange(this.getPage(), 0, this._numPages)); }, _setUpPagination: function () { if (this._options.pagination) { if (Common.isDOMElement(this._options.pagination) || typeof this._options.pagination === 'string') { // if dom element or css selector string... this._pagination = new Pagination(this._options.pagination, { size: this._numPages, onChange: this._handlers.paginationChange }); } else { // assumes instantiated pagination this._pagination = this._options.pagination; this._pagination._options.onChange = this._handlers.paginationChange; this._pagination.setSize(this._numPages); } this._pagination.setCurrent(this._options.initialPage || 0); } else { this._currentPage = this._options.initialPage || 0; } }, _setUpAutoAdvance: function () { if (!this._options.autoAdvance) { return; } var self = this; setTimeout(function autoAdvance() { self.nextPage(true /* wrap */); setTimeout(autoAdvance, self._options.autoAdvance); }, this._options.autoAdvance); }, _setUpHider: function () { if (this._options.hideLast) { var hiderEl = InkElement.create('div', { className: 'hider', insertBottom: this._element }); hiderEl.style.position = 'absolute'; hiderEl.style[ this._isY ? 'left' : 'top' ] = '0'; // fix to top.. hiderEl.style[ this._isY ? 'right' : 'bottom' ] = '0'; // and bottom... hiderEl.style[ this._isY ? 'bottom' : 'right' ] = '0'; // and move to the end. this._hiderEl = hiderEl; } }, // [3.1.0] Deprecate this already _center: function() { if (!this._options.center) { return; } var gap = Math.floor( (this._ctnLength - (this._elLength * this._slidesPerPage) ) / 2 ); var pad; if (this._isY) { pad = [gap, 'px 0']; } else { pad = ['0 ', gap, 'px']; } this._ulEl.style.padding = pad.join(''); }, // [3.1.0] Deprecate this already _updateHider: function() { if (!this._hiderEl) { return; } if (this.getPage() === 0) { var gap = Math.floor( this._ctnLength - (this._elLength * this._slidesPerPage) ); if (this._options.center) { gap /= 2; } this._hiderEl.style[ this._isY ? 'height' : 'width' ] = gap + 'px'; } else { this._hiderEl.style[ this._isY ? 'height' : 'width' ] = '0px'; } }, /** * Refits elements for IE7 because it doesn't support inline-block. * * @method _IE7 * @private */ _IE7: function () { if (Browser.IE && '' + Browser.version.split('.')[0] === '7') { // var numPages = this._numPages; var slides = Ink.ss('li.slide', this._ulEl); var stl = function (prop, val) {slides[i].style[prop] = val; }; for (var i = 0, len = slides.length; i < len; i++) { stl('position', 'absolute'); stl(this._isY ? 'top' : 'left', (i * this._elLength) + 'px'); } } }, _onTouchStart: function (event) { if (event.touches.length > 1) { return; } this._swipeData = { x: InkEvent.pointerX(event), y: InkEvent.pointerY(event) }; var ulRect = this._ulEl.getBoundingClientRect(); this._swipeData.firstUlPos = ulRect[this._isY ? 'top' : 'left']; this._swipeData.inUlX = this._swipeData.x - ulRect.left; this._swipeData.inUlY = this._swipeData.y - ulRect.top; setTransitionProperty(this._ulEl, 'none'); this._touchMoveIsFirstTouchMove = true; }, _onTouchMove: function (event) { if (event.touches.length > 1) { return; /* multitouch event, not my problem. */ } var pointerX = InkEvent.pointerX(event); var pointerY = InkEvent.pointerY(event); var deltaY = Math.abs(pointerY - this._swipeData.y); var deltaX = Math.abs(pointerX - this._swipeData.x); if (this._touchMoveIsFirstTouchMove) { this._touchMoveIsFirstTouchMove = undefined; this._scrolling = this._isY ? deltaX > deltaY : deltaY > deltaX ; if (!this._scrolling) { this._onAnimationFrame(); } } if (!this._scrolling && this._swipeData) { InkEvent.stopDefault(event); this._swipeData.pointerPos = this._isY ? pointerY : pointerX; } }, _onAnimationFrame: function () { var swipeData = this._swipeData; if (!swipeData || this._scrolling || this._touchMoveIsFirstTouchMove) { return; } var elRect = this._element.getBoundingClientRect(); var newPos; if (!this._isY) { newPos = swipeData.pointerPos - swipeData.inUlX - elRect.left; } else { newPos = swipeData.pointerPos - swipeData.inUlY - elRect.top; } this._ulEl.style[this._isY ? 'top' : 'left'] = newPos + 'px'; requestAnimationFrame(Ink.bindMethod(this, '_onAnimationFrame')); }, _onTouchEnd: function (event) { if (this._swipeData && this._swipeData.pointerPos && !this._scrolling && !this._touchMoveIsFirstTouchMove) { var snapToNext = 0.1; // swipe 10% of the way to change page var relProgress = this._swipeData.firstUlPos - this._ulEl.getBoundingClientRect()[this._isY ? 'top' : 'left']; var curPage = this.getPage(); // How many pages were advanced? May be fractional. var progressInPages = relProgress / this._elLength / this._slidesPerPage; // Have we advanced enough to change page? if (Math.abs(progressInPages) > snapToNext) { curPage += Math[ relProgress < 0 ? 'floor' : 'ceil' ](progressInPages); } // If something used to calculate progressInPages was zero, we get NaN here. if (!isNaN(curPage)) { this.setPage(curPage); } InkEvent.stopDefault(event); } setTransitionProperty(this._ulEl, null /* transition: left, top */); this._swipeData = null; this._touchMoveIsFirstTouchMove = undefined; this._scrolling = undefined; }, _onPaginationChange: function(pgn) { this._setPage(pgn.getCurrent()); }, /** * Gets the current page index * @method getPage * @return The current page number **/ getPage: function () { if (this._pagination) { return this._pagination.getCurrent(); } else { return this._currentPage || 0; } }, /** * Sets the current page index * @method setPage * @param {Number} page Index of the destination page. * @param {Boolean} [wrap] Flag to activate circular counting. **/ setPage: function (page, wrap) { if (wrap) { // Pages outside the range [0..this._numPages] are wrapped. page = page % this._numPages; if (page < 0) { page = this._numPages - page; } } page = limitRange(page, 0, this._numPages - 1); if (this._pagination) { this._pagination.setCurrent(page); // _setPage is called by pagination because it listens to its Change event. } else { this._setPage(page); } }, _setPage: function (page) { var _lengthToGo = page * this._deltaLength; var isLastPage = page === (this._numPages - 1); if (!this._options.spaceAfterLastSlide && isLastPage && page > 0) { var _itemsInLastPage = this._liEls.length - (page * this._slidesPerPage); if(_itemsInLastPage < this._slidesPerPage) { _lengthToGo = ((page - 1) * this._deltaLength) + (_itemsInLastPage * this._elLength); } } this._ulEl.style[ this._isY ? 'top' : 'left'] = ['-', _lengthToGo, 'px'].join(''); if (this._options.onChange) { this._options.onChange.call(this, page); } this._currentPage = page; this._updateHider(); }, /** * Goes to the next page * @method nextPage * @param {Boolean} [wrap] Flag to loop from last page to first page. **/ nextPage: function (wrap) { this.setPage(this.getPage() + 1, wrap); }, /** * Goes to the previous page * @method previousPage * @param {Boolean} [wrap] Flag to loop from first page to last page. **/ previousPage: function (wrap) { this.setPage(this.getPage() - 1, wrap); }, /** * Returns how many slides fit into a page * @method getSlidesPerPage * @return {Number} The number of slides per page * @public */ getSlidesPerPage: function() { return this._slidesPerPage; }, /** * Get the amount of pages in the carousel. * @method getTotalPages * @return {Number} The number of pages * @public */ getTotalPages: function() { return this._numPages; }, /** * Get the stage element (your UL with the class ".stage"). * @method getStageElm * @public * @return {DOMElement} Stage element **/ getStageElm: function() { return this._ulEl; }, /** * Get a list of your slides (elements with the ".slide" class inside your stage) * @method getSlidesList * @return {DOMElement[]} Array containing the slides. * @public */ getSlidesList: function() { return this._liEls; }, /** * Get the total number of slides * @method getTotalSlides * @return {Number} The number of slides * @public */ getTotalSlides: function() { return this.getSlidesList().length; } }; function setTransitionProperty(el, newTransition) { el.style.transitionProperty = el.style.oTransitionProperty = el.style.msTransitionProperty = el.style.mozTransitionProperty = el.style.webkitTransitionProperty = newTransition; } Common.createUIComponent(Carousel); return Carousel; }); /** * Closing utilities * @module Ink.UI.Close_1 * @version 1 */ Ink.createModule('Ink.UI.Close', '1', ['Ink.Dom.Event_1','Ink.Dom.Element_1'], function(InkEvent, InkElement) { 'use strict'; /** * Subscribes clicks on the document.body. * Whenever an element with the classes ".ink-close" or ".ink-dismiss" is clicked, this module finds an ancestor ".ink-alert" or ".ink-alert-block" element and removes it from the DOM. * This module should be created only once per page. * * @class Ink.UI.Close * @constructor * @example * * * @sample Ink_UI_Close_1.html */ var Close = function() { InkEvent.observe(document.body, 'click', function(ev) { var el = InkEvent.element(ev); el = InkElement.findUpwardsByClass(el, 'ink-close') || InkElement.findUpwardsByClass(el, 'ink-dismiss'); if (!el) { return; // ink-close or ink-dismiss class not found } var toRemove = InkElement.findUpwardsByClass(el, 'ink-alert') || InkElement.findUpwardsByClass(el, 'ink-alert-block') || el; if (toRemove) { InkEvent.stop(ev); InkElement.remove(toRemove); } }); }; Close._name = 'Close_1'; return Close; }); /** * Auxiliar utilities for UI Modules * @module Ink.UI.Common_1 * @version 1 */ Ink.createModule('Ink.UI.Common', '1', ['Ink.Dom.Element_1', 'Ink.Net.Ajax_1','Ink.Dom.Css_1','Ink.Dom.Selector_1','Ink.Util.Url_1'], function(InkElement, Ajax,Css,Selector,Url) { 'use strict'; var nothing = {} /* a marker, for reference comparison. */; var keys = Object.keys || function (obj) { var ret = []; for (var k in obj) if (obj.hasOwnProperty(k)) { ret.push(k); } return ret; }; var es6WeakMapSupport = 'WeakMap' in window; var instances = es6WeakMapSupport ? new WeakMap() : null; var domRegistry = { get: function get(el) { return es6WeakMapSupport ? instances.get(el) : el.__InkInstances; }, set: function set(el, thing) { if (es6WeakMapSupport) { instances.set(el, thing); } else { el.__InkInstances = thing; } } }; /** * @namespace Ink.UI.Common_1 */ var Common = { /** * Supported Ink Layouts * * @property Layouts * @type Object * @readOnly */ Layouts: { TINY: 'tiny', SMALL: 'small', MEDIUM: 'medium', LARGE: 'large', XLARGE: 'xlarge' }, /** * Checks if an item is a valid DOM Element. * * @method isDOMElement * @static * @param {Mixed} o The object to be checked. * @return {Boolean} True if it's a valid DOM Element. * @example * var el = Ink.s('#element'); * if( Ink.UI.Common.isDOMElement( el ) === true ){ * // It is a DOM Element. * } else { * // It is NOT a DOM Element. * } */ isDOMElement: InkElement.isDOMElement, /** * Checks if an item is a valid integer. * * @method isInteger * @static * @param {Mixed} n The value to be checked. * @return {Boolean} True if it's a valid integer. * @example * var value = 1; * if( Ink.UI.Common.isInteger( value ) === true ){ * // It is an integer. * } else { * // It is NOT an integer. * } */ isInteger: function(n) { return (typeof n === 'number' && n % 1 === 0); }, /** * Gets a DOM Element. * * @method elOrSelector * @static * @param {DOMElement|String} elOrSelector DOM Element or CSS Selector * @param {String} fieldName The name of the field. Commonly used for debugging. * @return {DOMElement} Returns the DOMElement passed or the first result of the CSS Selector. Otherwise it throws an exception. * @example * // In case there are several .myInput, it will retrieve the first found * var el = Ink.UI.Common.elOrSelector('.myInput','My Input'); */ elOrSelector: function(elOrSelector, fieldName) { if (!this.isDOMElement(elOrSelector)) { var t = Selector.select(elOrSelector); if (t.length === 0) { Ink.warn(fieldName + ' must either be a DOM Element or a selector expression!\nThe script element must also be after the DOM Element itself.'); return null; } return t[0]; } return elOrSelector; }, /** * Alias for `elOrSelector` but returns an array of elements. * * @method elsOrSelector * * @static * @param {DOMElement|String} elOrSelector DOM Element or CSS Selector * @param {String} fieldName The name of the field. Commonly used for debugging. * @return {DOMElement} Returns the DOMElement passed or the first result of the CSS Selector. Otherwise it throws an exception. * @param {Boolean} required Flag to accept an empty array as output. * @return {Array} The selected DOM Elements. * @example * var elements = Ink.UI.Common.elsOrSelector('input.my-inputs', 'My Input'); */ elsOrSelector: function(elsOrSelector, fieldName, required) { var ret; if (typeof elsOrSelector === 'string') { ret = Selector.select(elsOrSelector); } else if (Common.isDOMElement(elsOrSelector)) { ret = [elsOrSelector]; } else if (elsOrSelector && typeof elsOrSelector === 'object' && typeof elsOrSelector.length === 'number') { ret = elsOrSelector; } if (ret && ret.length) { return ret; } else { if (required) { throw new TypeError(fieldName + ' must either be a DOM Element, an Array of elements, or a selector expression!\nThe script element must also be after the DOM Element itself.'); } else { return []; } } }, /** * Gets options an object and element's metadata. * * The element's data attributes take precedence. Values from the element's data-atrributes are coerced into the required type. * * @method options * * @param {Object} [fieldId] Name to be used in debugging features. * @param {Object} defaults Object with the options' types and defaults. * @param {Object} overrides Options to override the defaults. Usually passed when instantiating an UI module. * @param {DOMElement} [element] Element with data-attributes * * @example * * this._options = Ink.UI.Common.options('MyComponent', { * 'anobject': ['Object', null], // Defaults to null * 'target': ['Element', null], * 'stuff': ['Number', 0.1], * 'stuff2': ['Integer', 0], * 'doKickFlip': ['Boolean', false], * 'targets': ['Elements'], // Required option since no default was given * 'onClick': ['Function', null] * }, options || {}, elm) * * @example * * ### Note about booleans * * Here is how options are read from the markup * data-attributes, for several values`data-a-boolean`. * * Options considered true: * * - `data-a-boolean="true"` * - (Every other value which is not on the list below.) * * Options considered false: * * - `data-a-boolean="false"` * - `data-a-boolean=""` * - `data-a-boolean` * * Options which go to default: * * - (no attribute). When `data-a-boolean` is ommitted, the * option is not considered true nor false, and as such * defaults to what is in the `defaults` argument. * **/ options: function (fieldId, defaults, overrides, element) { if (typeof fieldId !== 'string') { element = overrides; overrides = defaults; defaults = fieldId; fieldId = ''; } overrides = overrides || {}; var out = {}; var dataAttrs = element ? InkElement.data(element) : {}; var fromDataAttrs; var type; var lType; var defaultVal; var invalidStr = function (str) { if (fieldId) { str = fieldId + ': "' + ('' + str).replace(/"/, '\\"') + '"'; } return str; }; var quote = function (str) { return '"' + ('' + str).replace(/"/, '\\"') + '"'; }; var invalidThrow = function (str) { throw new Error(invalidStr(str)); }; var invalid = function (str) { Ink.error(invalidStr(str) + '. Ignoring option.'); }; function optionValue(key) { type = defaults[key][0]; lType = type.toLowerCase(); defaultVal = defaults[key].length === 2 ? defaults[key][1] : nothing; if (!type) { invalidThrow('Ink.UI.Common.options: Always specify a type!'); } if (!(lType in Common._coerce_funcs)) { invalidThrow('Ink.UI.Common.options: ' + defaults[key][0] + ' is not a valid type. Use one of ' + keys(Common._coerce_funcs).join(', ')); } if (!defaults[key].length || defaults[key].length > 2) { invalidThrow('the "defaults" argument must be an object mapping option names to [typestring, optional] arrays.'); } if (key in dataAttrs) { fromDataAttrs = Common._coerce_from_string(lType, dataAttrs[key], key, fieldId); // (above can return `nothing`) } else { fromDataAttrs = nothing; } if (fromDataAttrs !== nothing) { if (!Common._options_validate(fromDataAttrs, lType)) { invalid('(' + key + ' option) Invalid ' + lType + ' ' + quote(fromDataAttrs)); return defaultVal; } else { return fromDataAttrs; } } else if (key in overrides) { return overrides[key]; } else if (defaultVal !== nothing) { return defaultVal; } else { invalidThrow('Option ' + key + ' is required!'); } } for (var key in defaults) { if (defaults.hasOwnProperty(key)) { out[key] = optionValue(key); } } return out; }, _coerce_from_string: function (type, val, paramName, fieldId) { if (type in Common._coerce_funcs) { return Common._coerce_funcs[type](val, paramName, fieldId); } else { return val; } }, _options_validate: function (val, type) { if (type in Common._options_validate_types) { return Common._options_validate_types[type].call(Common, val); } else { // 'object' options cannot be passed through data-attributes. // Json you say? Not any good to embed in HTML. return false; } }, _coerce_funcs: (function () { var ret = { element: function (val) { return Common.elOrSelector(val, ''); }, elements: function (val) { return Common.elsOrSelector(val, '', false /*not required, so don't throw an exception now*/); }, object: function (val) { return val; }, number: function (val) { return parseFloat(val); }, 'boolean': function (val) { return !(val === 'false' || val === '' || val === null); }, string: function (val) { return val; }, 'function': function (val, paramName, fieldId) { Ink.error(fieldId + ': You cannot specify the option "' + paramName + '" through data-attributes because it\'s a function'); return nothing; } }; ret['float'] = ret.integer = ret.number; return ret; }()), _options_validate_types: (function () { var types = { string: function (val) { return typeof val === 'string'; }, number: function (val) { return typeof val === 'number' && !isNaN(val) && isFinite(val); }, integer: function (val) { return val === Math.round(val); }, element: function (val) { return Common.isDOMElement(val); }, elements: function (val) { return val && typeof val === 'object' && typeof val.length === 'number' && val.length; }, 'boolean': function (val) { return typeof val === 'boolean'; }, object: function () { return true; } }; types['float'] = types.number; return types; }()), /** * Deep copy (clone) an object. * Note: The object cannot have referece loops. * * @method clone * @static * @param {Object} o The object to be cloned/copied. * @return {Object} Returns the result of the clone/copy. * @example * var originalObj = { * key1: 'value1', * key2: 'value2', * key3: 'value3' * }; * var cloneObj = Ink.UI.Common.clone( originalObj ); */ clone: function(o) { try { return JSON.parse( JSON.stringify(o) ); } catch (ex) { throw new Error('Given object cannot have loops!'); } }, /** * Gets an element's one-base index relative to its parent. * * @method childIndex * @static * @param {DOMElement} childEl Valid DOM Element. * @return {Number} Numerical position of an element relatively to its parent. * @example * *
    *
  • One
  • *
  • Two
  • *
  • Three
  • *
  • Four
  • *
* * */ childIndex: function(childEl) { if( Common.isDOMElement(childEl) ){ var els = Selector.select('> *', childEl.parentNode); for (var i = 0, f = els.length; i < f; ++i) { if (els[i] === childEl) { return i; } } } throw 'not found!'; }, /** * AJAX JSON request shortcut method * It provides a more convenient way to do an AJAX request and expect a JSON response.It also offers a callback option, as third parameter, for better async handling. * * @method ajaxJSON * @static * @async * @param {String} endpoint Valid URL to be used as target by the request. * @param {Object} params This field is used in the thrown Exception to identify the parameter. * @param {Function} cb Callback for the request. * @example * // In case there are several .myInput, it will retrieve the first found * var el = Ink.UI.Common.elOrSelector('.myInput','My Input'); */ ajaxJSON: function(endpoint, params, cb) { new Ajax( endpoint, { evalJS: 'force', method: 'POST', parameters: params, onSuccess: function( r) { try { r = r.responseJSON; if (r.status !== 'ok') { throw 'server error: ' + r.message; } cb(null, r); } catch (ex) { cb(ex); } }, onFailure: function() { cb('communication failure'); } } ); }, /** * Gets the current Ink layout. * * @method currentLayout * @static * @return {String} A string representation of the current layout name. * @example * var inkLayout = Ink.UI.Common.currentLayout(); * if (inkLayout === 'small') { * // ... * } */ currentLayout: function() { var i, f, k, v, el, detectorEl = Selector.select('#ink-layout-detector')[0]; if (!detectorEl) { detectorEl = document.createElement('div'); detectorEl.id = 'ink-layout-detector'; for (k in this.Layouts) { if (this.Layouts.hasOwnProperty(k)) { v = this.Layouts[k]; el = document.createElement('div'); el.className = 'show-' + v + ' hide-all'; el.setAttribute('data-ink-layout', v); detectorEl.appendChild(el); } } document.body.appendChild(detectorEl); } for (i = 0, f = detectorEl.children.length; i < f; ++i) { el = detectorEl.children[i]; if (Css.getStyle(el, 'display') === 'block') { return el.getAttribute('data-ink-layout'); } } return 'large'; }, /** * Sets the location's hash (window.location.hash). * * @method hashSet * @static * @param {Object} o Object with the info to be placed in the location's hash. * @example * // It will set the location's hash like: #key1=value1&key2=value2&key3=value3 * Ink.UI.Common.hashSet({ * key1: 'value1', * key2: 'value2', * key3: 'value3' * }); */ hashSet: function(o) { if (typeof o !== 'object') { throw new TypeError('o should be an object!'); } var hashParams = Url.getAnchorString(); hashParams = Ink.extendObj(hashParams, o); window.location.hash = Url.genQueryString('', hashParams).substring(1); }, /** * Removes children nodes from a given object. * This method was initially created to help solve a problem in Internet Explorer(s) that occurred when trying to set the innerHTML of some specific elements like 'table'. * * @method cleanChildren * @static * @param {DOMElement} parentEl Valid DOM Element * @example * *
    *
  • One
  • *
  • Two
  • *
  • Three
  • *
  • Four
  • *
* * * * *
    */ cleanChildren: function(parentEl) { if( !Common.isDOMElement(parentEl) ){ throw 'Please provide a valid DOMElement'; } var prevEl, el = parentEl.lastChild; while (el) { prevEl = el.previousSibling; parentEl.removeChild(el); el = prevEl; } }, /** * Stores the id and/or classes of an element in an object. * * @method storeIdAndClasses * @static * @param {DOMElement} fromEl Valid DOM Element to get the id and classes from. * @param {Object} inObj Object where the id and classes will be saved. * @example *
    * * */ storeIdAndClasses: function(fromEl, inObj) { if( !Common.isDOMElement(fromEl) ){ throw 'Please provide a valid DOMElement as first parameter'; } var id = fromEl.id; if (id) { inObj._id = id; } var classes = fromEl.className; if (classes) { inObj._classes = classes; } }, /** * Sets the id and className properties of an element based * * @method restoreIdAndClasses * @static * @param {DOMElement} toEl Valid DOM Element to set the id and classes on. * @param {Object} inObj Object where the id and classes to be set are. This method uses the same format as the one given in `storeIdAndClasses` * @example *
    * * * * *
    */ restoreIdAndClasses: function(toEl, inObj) { if( !Common.isDOMElement(toEl) ){ throw 'Please provide a valid DOMElement as first parameter'; } if (inObj._id && toEl.id !== inObj._id) { toEl.id = inObj._id; } if (inObj._classes && toEl.className.indexOf(inObj._classes) === -1) { if (toEl.className) { toEl.className += ' ' + inObj._classes; } else { toEl.className = inObj._classes; } } if (inObj._instanceId && !toEl.getAttribute('data-instance')) { toEl.setAttribute('data-instance', inObj._instanceId); } }, _warnDoubleInstantiation: function (elm, newInstance) { var instances = Common.getInstance(elm); if (getName(newInstance) === '') { return; } if (!instances) { return; } var nameWithoutVersion = getName(newInstance); if (!nameWithoutVersion) { return; } for (var i = 0, len = instances.length; i < len; i++) { if (nameWithoutVersion === getName(instances[i])) { // Yes, I am using + to concatenate and , to split // arguments. // // Elements can't be concatenated with strings, but if // they are passed in an argument, modern debuggers will // pretty-print them and make it easy to find them in the // element inspector. // // On the other hand, if strings are passed as different // arguments, they get pretty printed. And the pretty // print of a string has quotes around it. // // If some day people find out that strings are not // just text and they start preserving contextual // information, then by all means change this to a // regular concatenation. // // But they won't. So don't change this. Ink.warn('Creating more than one ' + nameWithoutVersion + '.', '(Was creating a ' + nameWithoutVersion + ' on:', elm, ').'); return false; } } function getName(thing) { return ((thing.constructor && (thing.constructor._name)) || thing._name || '').replace(/_.*?$/, ''); } return true; }, /** * Saves a component's instance reference for later retrieval. * * @method registerInstance * @static * @param {Object} inst Object that holds the instance. * @param {DOMElement} el DOM Element to associate with the object. */ registerInstance: function(inst, el) { if (!inst) { return; } if (!Common.isDOMElement(el)) { throw new TypeError('Ink.UI.Common.registerInstance: The element passed in is not a DOM element!'); } // [todo] this belongs in the BaseUIComponent's initialization if (Common._warnDoubleInstantiation(el, inst) === false) { return false; } var instances = domRegistry.get(el); if (!instances) { instances = []; domRegistry.set(el, instances); } instances.push(inst); return true; }, /** * Deletes an instance with a given id. * * @method unregisterInstance * @static * @param {String} id Id of the instance to be destroyed. */ unregisterInstance: function(inst) { if (!inst || !inst._element) { return; } var instances = domRegistry.get(inst._element); for (var i = 0, len = instances.length; i < len; i++) { if (instances[i] === inst) { instances.splice(i, 1); } } }, /** * Gets an UI instance from an element or instance id. * * @method getInstance * @static * @param {String|DOMElement} el DOM Element from which we want the instances. * @return {Object|Array} Returns an instance or a collection of instances. */ getInstance: function(el, UIComponent) { el = Common.elOrSelector(el); var instances = domRegistry.get(el); if (!instances) { instances = []; } if (typeof UIComponent !== 'function') { return instances; } for (var i = 0, len = instances.length; i < len; i++) { if (instances[i] instanceof UIComponent) { return instances[i]; } } return null; }, /** * Gets an instance based on a selector. * * @method getInstanceFromSelector * @static * @param {String} selector CSS selector to get the instances from. * @return {Object|Array} Returns an instance or a collection of instances. */ getInstanceFromSelector: function(selector) { return Common.getInstance(Common.elOrSelector(selector)); }, /** * Gets all the instance ids * * @method getInstanceIds * @static * @return {Array} Collection of instance ids */ getInstanceIds: function() { var res = []; for (var id in instances) { if (instances.hasOwnProperty(id)) { res.push( id ); } } return res; }, /** * Gets all the instances * * @method getInstances * @static * @return {Array} Collection of existing instances. */ getInstances: function() { var res = []; for (var id in instances) { if (instances.hasOwnProperty(id)) { res.push( instances[id] ); } } return res; }, /** * Boilerplate method to destroy a component. * Components should copy this method as its destroy method and modify it. * * @method destroyComponent * @static */ destroyComponent: function() { Common.unregisterInstance(this); this._element.parentNode.removeChild(this._element); } }; /** * Ink UI Base Class **/ function warnStub() { /* jshint validthis: true */ if (!this || this === window || typeof this.constructor !== 'function') { return; } Ink.warn('You called a method on an incorrectly instantiated ' + this.constructor._name + ' component. Check the warnings above to see what went wrong.'); } function stub(prototype, obj) { for (var k in prototype) if (prototype.hasOwnProperty(k)) { if (k === 'constructor') { continue; } if (typeof obj[k] === 'function') { obj[k] = warnStub; } } } /** * Ink UI Base Class * * You don't use this class directly, or inherit from it directly. * * See createUIComponent() (in this module) for how to create a UI component and inherit from this. It's not plain old JS inheritance, for several reasons. * * @class Ink.UI.Common.BaseUIComponent * @constructor * * @param element * @param options **/ function BaseUIComponent(element, options) { var constructor = this.constructor; var _name = constructor._name; if (!this || this === window) { throw new Error('Use "new InkComponent()" instead of "InkComponent()"'); } if (this && !(this instanceof BaseUIComponent)) { throw new Error('You forgot to call Ink.UI.Common.createUIComponent() on this module!'); } if (!element && !constructor._componentOptions.elementIsOptional) { Ink.error(new Error(_name + ': You need to pass an element or a selector as the first argument to "new ' + _name + '()"')); return; } else { this._element = Common.elsOrSelector(element, _name + ': An element with the selector "' + element + '" was not found!')[0]; } if (!this._element && !constructor._componentOptions.elementIsOptional) { isValidInstance = false; Ink.error(new Error(element + ' does not match an element on the page. You need to pass a valid selector to "new ' + _name + '".')); } // TODO Change Common.options's signature? the below looks better, more manageable // var options = Common.options({ // element: this._element, // modName: constructor._name, // options: constructor._optionDefinition, // defaults: constructor._globalDefaults // }); this._options = Common.options(_name, constructor._optionDefinition, options, this._element); var isValidInstance = BaseUIComponent._validateInstance(this) === true; if (isValidInstance && typeof this._init === 'function') { try { this._init.apply(this, arguments); } catch(e) { isValidInstance = false; Ink.error(e); } } if (!isValidInstance) { BaseUIComponent._stubInstance(this, constructor, _name); } else if (this._element) { Common.registerInstance(this, this._element); } } /** * Calls the `instance`'s _validate() method so it can validate itself. * * Returns false if the method exists, was called, but no Error was returned or thrown. * * @method _validateInstance * @private */ BaseUIComponent._validateInstance = function (instance) { var err; if (typeof instance._validate !== 'function') { return true; } try { err = instance._validate(); } catch (e) { err = e; } if (err instanceof Error) { instance._validationError = err; return false; } return true; }; /** * Replaces every method in the instance with stub functions which just call Ink.warn(). * * This avoids breaking the page when there are errors. * * @method _stubInstance * @param instance * @param constructor * @param name * @private */ BaseUIComponent._stubInstance = function (instance, constructor, name) { stub(constructor.prototype, instance); stub(BaseUIComponent.prototype, instance); Ink.warn(name + ' was not correctly created. ' + (instance._validationError || '')); }; // TODO BaseUIComponent.setGlobalOptions = function () {} // TODO BaseUIComponent.createMany = function (selector) {} BaseUIComponent.getInstance = function (elOrSelector) { elOrSelector = Common.elOrSelector(elOrSelector); return Common.getInstance(elOrSelector, this /* get instance by constructor */); }; Ink.extendObj(BaseUIComponent.prototype, { /** * Get an UI component's option's value. * * @method getOption * @param name * * @return The option value, or undefined if nothing is found. * * @example * * var myUIComponent = new Modal('#element', { trigger: '#trigger' }); // or anything else inheriting BaseUIComponent * myUIComponent.getOption('trigger'); // -> The trigger element (not the selector string, mind you) * **/ getOption: function (name) { if (this.constructor && !(name in this.constructor._optionDefinition)) { Ink.error('"' + name + '" is not an option for ' + this.constructor._name); return undefined; } return this._options[name]; }, /** * Sets an option's value * * @method getOption * @param name * @param value * * @example * * var myUIComponent = new Modal(...); * myUIComponent.setOption('trigger', '#some-element'); **/ setOption: function (name, value) { if (this.constructor && !(name in this.constructor._optionDefinition)) { Ink.error('"' + name + ' is not an option for ' + this.constructor._name); return; } this._options[name] = value; }, /** * Get the element associated with an UI component (IE the one you used in the constructor) * * @method getElement * @return {Element} The component's element. * * @example * var myUIComponent = new Modal('#element'); // or anything else inheriting BaseUIComponent * myUIComponent.getElement(); // -> The '#element' (not the selector string, mind you). * **/ getElement: function () { return this._element; } }); Common.BaseUIComponent = BaseUIComponent; /** * @method createUIComponent * @param theConstructor UI component constructor. It should have an _init function in its prototype, an _optionDefinition object, and a _name property indicating its name. * @param options * @param [options.elementIsOptional=false] Whether the element argument is optional (For example, when the component might work on existing markup or create its own). **/ Common.createUIComponent = function createUIComponent(theConstructor, options) { theConstructor._componentOptions = options || {}; function assert(test, msg) { if (!test) { throw new Error('Ink.UI_1.createUIComponent: ' + msg); } } function assertProp(prop, propType, message) { var propVal = theConstructor[prop]; // Check that the property was passed assert(typeof propVal !== 'undefined', theConstructor + ' doesn\'t have a "' + prop + '" property. ' + message); // Check that its type is correct assert(propType && typeof propVal === propType, 'typeof ' + theConstructor + '.' + prop + ' is not "' + propType + '". ' + message); } assert(typeof theConstructor === 'function', 'constructor argument is not a function!'); assertProp('_name', 'string', 'This property is used for error ' + 'messages. Set it to the full module path and version (Ink.My.Module_1).'); assertProp('_optionDefinition', 'object', 'This property contains the ' + 'option names, types and defaults. See Ink.UI.Common.options() for reference.'); // Extend the instance methods and props var _oldProto = theConstructor.prototype; if (typeof Object.create === 'function') { theConstructor.prototype = Object.create(BaseUIComponent.prototype); } else { theConstructor.prototype = (function hideF() { function F() {} F.prototype = BaseUIComponent.prototype; return new F(); }()); } Ink.extendObj(theConstructor.prototype, _oldProto); theConstructor.prototype.constructor = theConstructor; // Extend static methods Ink.extendObj(theConstructor, BaseUIComponent); }; return Common; }); /** * Date selector * @module Ink.UI.DatePicker_1 * @version 1 */ Ink.createModule('Ink.UI.DatePicker', '1', ['Ink.UI.Common_1','Ink.Dom.Event_1','Ink.Dom.Css_1','Ink.Dom.Element_1','Ink.Dom.Selector_1','Ink.Util.Array_1','Ink.Util.Date_1', 'Ink.Dom.Browser_1'], function(Common, Event, Css, InkElement, Selector, InkArray, InkDate ) { 'use strict'; // Clamp a number into a min/max limit function clamp(n, min, max) { if (n > max) { n = max; } if (n < min) { n = min; } return n; } function dateishFromYMDString(YMD) { var split = YMD.split('-'); return dateishFromYMD(+split[0], +split[1] - 1, +split[2]); } function dateishFromYMD(year, month, day) { return {_year: year, _month: month, _day: day}; } function dateishFromDate(date) { return {_year: date.getFullYear(), _month: date.getMonth(), _day: date.getDate()}; } /** * @class Ink.UI.DatePicker * @constructor * @version 1 * * @param {String|DOMElement} selector * @param {Object} [options] Options * @param {Boolean} [options.autoOpen] Flag to automatically open the datepicker. * @param {String} [options.cleanText] Text for the clean button. Defaults to 'Clear'. * @param {String} [options.closeText] Text for the close button. Defaults to 'Close'. * @param {String} [options.cssClass] CSS class to be applied on the datepicker * @param {String|DOMElement} [options.pickerField] (if not using in an input[type="text"]) Element which displays the DatePicker when clicked. Defaults to an "open" link. * @param {String} [options.dateRange] Enforce limits to year, month and day for the Date, ex: '1990-08-25:2020-11' * @param {Boolean} [options.displayInSelect] Flag to display the component in a select element. * @param {String|DOMElement} [options.dayField] (if using options.displayInSelect) `select` field with days. * @param {String|DOMElement} [options.monthField] (if using options.displayInSelect) `select` field with months. * @param {String|DOMElement} [options.yearField] (if using options.displayInSelect) `select` field with years. * @param {String} [options.format] Date format string * @param {Object} [options.month] Hash of month names. Defaults to portuguese month names. January is 1. * @param {String} [options.nextLinkText] Text for the previous button. Defaults to '«'. * @param {String} [options.ofText] Text to show between month and year. Defaults to ' of '. * @param {Boolean} [options.onFocus] If the datepicker should open when the target element is focused. Defaults to true. * @param {Function} [options.onMonthSelected] Callback to execute when the month is selected. * @param {Function} [options.onSetDate] Callback to execute when the date is set. * @param {Function} [options.onYearSelected] Callback to execute when the year is selected. * @param {String} [options.position] Position for the datepicker. Either 'right' or 'bottom'. Defaults to 'right'. * @param {String} [options.prevLinkText] Text for the previous button. Defaults to '«'. * @param {Boolean} [options.showClean] If the clean button should be visible. Defaults to true. * @param {Boolean} [options.showClose] If the close button should be visible. Defaults to true. * @param {Boolean} [options.shy] If the datepicker should start automatically. Defaults to true. * @param {String} [options.startDate] Date to define initial month. Must be in yyyy-mm-dd format. * @param {Number} [options.startWeekDay] First day of the week. Sunday is zero. Defaults to 1 (Monday). * @param {Function} [options.validYearFn] Callback to execute when 'rendering' the month (in the month view) * @param {Function} [options.validMonthFn] Callback to execute when 'rendering' the month (in the month view) * @param {Function} [options.validDayFn] Callback to execute when 'rendering' the day (in the month view) * @param {Function} [options.nextValidDateFn] Function to calculate the next valid date, given the current. Useful when there's invalid dates or time frames. * @param {Function} [options.prevValidDateFn] Function to calculate the previous valid date, given the current. Useful when there's invalid dates or time frames. * @param {Object} [options.wDay] Hash of week day names. Sunday is 0. Defaults to { 0:'Sunday', 1:'Monday', etc... * @param {String} [options.yearRange] Enforce limits to year for the Date, ex: '1990:2020' (deprecated) * * @sample Ink_UI_DatePicker_1.html */ var DatePicker = function() { Common.BaseUIComponent.apply(this, arguments); }; DatePicker._name = 'DatePicker_1'; DatePicker._optionDefinition = { autoOpen: ['Boolean', false], cleanText: ['String', 'Clear'], closeText: ['String', 'Close'], pickerField: ['Element', null], containerElement:['Element', null], cssClass: ['String', 'ink-calendar bottom'], dateRange: ['String', null], // use this in a * * By applying this UI class to the above input, you get a tag field with the tags "initial" and "value". The class preserves the original input element. It remains hidden and is updated with new tag information dynamically, so regular HTML form logic still applies. * * Below "input" refers to the current value of the input tag (updated as the user enters text, of course), and "output" refers to the value which this class writes back to said input tag. * * @class Ink.UI.TagField * @version 1 * @constructor * @param {String|DOMElement} element Selector or DOM Input Element. * @param {Object} [options] Options object * @param {String|Array} [options.tags] Initial tags in the input * @param {Boolean} [options.allowRepeated] Flag to allow user to input several tags. Defaults to true. * @param {RegExp} [options.separator] Split the input by this RegExp. Defaults to /[,;(space)]+/g (spaces, commas and semicolons) * @param {String} [options.outSeparator] Use this string to separate each tag from the next in the output. Defaults to ','. * @param {Boolean} [options.autoSplit] Flag to activate tag creation when the user types a separator. Defaults to true. * @param {Integer} [options.maxTags] Maximum number of tags allowed. Set to -1 for no limit. Defaults to -1. * @example */ function TagField() { Common.BaseUIComponent.apply(this, arguments); } TagField._name = 'TagField_1'; TagField._optionDefinition = { tags: ['String', []], tagQuery: ['Object', null], tagQueryAsync: ['Object', null], allowRepeated: ['Boolean', false], maxTags: ['Integer', -1], outSeparator: ['String', ','], separator: ['String', /[,; ]+/g], autoSplit: ['Boolean', true] }; TagField.prototype = { /** * Init function called by the constructor * * @method _init * @private */ _init: function() { var o = this._options; if (typeof o.separator === 'string') { o.separator = new RegExp(o.separator, 'g'); } if (typeof o.tags === 'string') { // coerce to array using the separator o.tags = this._readInput(o.tags); } Css.addClassName(this._element, 'hide-all'); this._viewElm = InkElement.create('div', { className: 'ink-tagfield', insertAfter: this._element }); this._input = InkElement.create('input', { type: 'text', className: 'new-tag-input', insertBottom: this._viewElm }); var tags = [].concat(o.tags, this._tagsFromMarkup(this._element)); this._tags = []; InkArray.each(tags, Ink.bindMethod(this, '_addTag')); InkEvent.observe(this._input, 'keyup', Ink.bindEvent(this._onKeyUp, this)); InkEvent.observe(this._input, 'change', Ink.bindEvent(this._onKeyUp, this)); InkEvent.observe(this._input, 'keydown', Ink.bindEvent(this._onKeyDown, this)); InkEvent.observe(this._input, 'blur', Ink.bindEvent(this._onBlur, this)); InkEvent.observe(this._viewElm, 'click', Ink.bindEvent(this._refocus, this)); }, destroy: function () { InkElement.remove(this._viewElm); Css.removeClassName(this._element, 'hide-all'); }, _tagsFromMarkup: function (element) { var tagname = element.tagName.toLowerCase(); if (tagname === 'input') { return this._readInput(element.value); } else if (tagname === 'select') { return InkArray.map(element.getElementsByTagName('option'), function (option) { return InkElement.textContent(option); }); } else { throw new Error('Cannot read tags from a ' + tagname + ' tag. Unknown tag'); } }, _tagsToMarkup: function (tags, element) { var tagname = element.tagName.toLowerCase(); if (tagname === 'input') { if (this._options.separator) { element.value = tags.join(this._options.outSeparator); } } else if (tagname === 'select') { element.innerHTML = ''; InkArray.each(tags, function (tag) { var opt = InkElement.create('option', {selected: 'selected'}); InkElement.setTextContent(opt, tag); element.appendChild(opt); }); } else { throw new Error('TagField: Cannot read tags from a ' + tagname + ' tag. Unknown tag'); } }, _addTag: function (tag) { if (this._options.maxTags !== -1 && this._tags.length >= this._options.maxTags) { return; } if ((!this._options.allowRepeated && InkArray.inArray(tag, this._tags, tag)) || !tag) { return false; } var elm = InkElement.create('span', { className: 'ink-tag', setTextContent: tag + ' ' }); var remove = InkElement.create('span', { className: 'remove fa fa-times', insertBottom: elm }); InkEvent.observe(remove, 'click', Ink.bindEvent(this._removeTag, this, null)); var spc = document.createTextNode(' '); this._tags.push(tag); this._viewElm.insertBefore(elm, this._input); this._viewElm.insertBefore(spc, this._input); this._tagsToMarkup(this._tags, this._element); }, _readInput: function (text) { if (this._options.separator) { return InkArray.filter(text.split(this._options.separator), isTruthy); } else { return [text]; } }, _onKeyUp: function () { // TODO control input box size if (!this._options.autoSplit) { return; } var split = this._input.value.split(this._options.separator); if (split.length <= 1) { return; } var last = split[split.length - 1]; split = split.splice(0, split.length - 1); split = InkArray.filter(split, isTruthy); InkArray.each(split, Ink.bind(this._addTag, this)); this._input.value = last; }, _onKeyDown: function (event) { if (event.which === enterKey) { return this._onEnterKeyDown(event); } else if (event.which === backspaceKey) { return this._onBackspaceKeyDown(); } else if (this._removeConfirm) { // user pressed another key, cancel removal from a backspace key this._unsetRemovingVisual(this._tags.length - 1); } }, /** * When the user presses backspace twice on the empty input, we delete the last tag on the field. * @method onBackspaceKeyDown * @private */ _onBackspaceKeyDown: function () { if (this._input.value) { return; } if (this._removeConfirm) { this._unsetRemovingVisual(this._tags.length - 1); this._removeTag(this._tags.length - 1); this._removeConfirm = null; } else { this._setRemovingVisual(this._tags.length - 1); } }, _onEnterKeyDown: function (event) { var tag = this._input.value; if (tag) { this._addTag(tag); this._input.value = ''; } InkEvent.stopDefault(event); }, _onBlur: function () { this._addTag(this._input.value); this._input.value = ''; }, /* For when the user presses backspace. * Set the style of the tag so that it seems like it's going to be removed * if they press backspace again. */ _setRemovingVisual: function (tagIndex) { var elm = this._viewElm.children[tagIndex]; if (!elm) { return; } Css.addClassName(elm, 'tag-deleting'); this._removeRemovingVisualTimeout = setTimeout(Ink.bindMethod(this, '_unsetRemovingVisual', tagIndex), 4000); InkEvent.observe(this._input, 'blur', Ink.bindMethod(this, '_unsetRemovingVisual', tagIndex)); this._removeConfirm = true; }, _unsetRemovingVisual: function (tagIndex) { var elm = this._viewElm.children[tagIndex]; if (elm) { Css.removeClassName(elm, 'tag-deleting'); clearTimeout(this._removeRemovingVisualTimeout); } this._removeConfirm = null; }, _removeTag: function (event) { var index; if (typeof event === 'object') { // click event on close button var elm = InkEvent.element(event).parentNode; index = InkElement.parentIndexOf(this._viewElm, elm); } else if (typeof event === 'number') { // manual removal index = event; } this._tags = InkArray.remove(this._tags, index, 1); InkElement.remove(this._viewElm.children[index]); this._tagsToMarkup(this._tags, this._element); }, _refocus: function (event) { this._input.focus(); InkEvent.stop(event); return false; } }; Common.createUIComponent(TagField); return TagField; }); /** * Toggle the visibility of elements. * @module Ink.UI.Toggle_1 * @version 1 */ Ink.createModule('Ink.UI.Toggle', '1', ['Ink.UI.Common_1','Ink.Dom.Event_1','Ink.Dom.Css_1','Ink.Dom.Element_1','Ink.Dom.Selector_1','Ink.Util.Array_1'], function(Common, InkEvent, Css, InkElement, Selector, InkArray ) { 'use strict'; /** * * You need two elements to use Toggle: the `trigger` element, and the `target` element (or elements). The default behaviour is to toggle the `target`(s) when you click the `trigger`. * * The toggle has a state. It is either "on" or "off". It works by switching between the CSS classes in `classNameOn` and `classNameOff` according to the current state. * * When you initialize the Toggle, it will check if the targets are visible to figure out what the initial state is. You can force the toggle to consider itself turned "on" or "off" by setting the `initialState` option to `true` or `false`, respectively. * * You can get the current state of the Toggle by calling `getState`, or by checking if your `trigger` element has the "active" class. * The state can be changed through JavaScript. Just call `setState(true)` * to turn the Toggle on (or `setState(false)` to turn it off). * * @class Ink.UI.Toggle * @constructor * @version 1 * @param {String|DOMElement} selector Trigger element. By clicking this, the target (or targets) are triggered. * @param {Object} [options] Options object, containing: * * @param {String} options.target CSS Selector that specifies the elements that this component will toggle * @param {String} [options.classNameOn] CSS class to toggle when on. Defaults to 'show-all'. * @param {String} [options.classNameOff] CSS class to toggle when off. Defaults to 'hide-all'. * @param {String} [options.triggerEvent] Event that will trigger the toggling. Defaults to 'click'. * @param {Boolean} [options.closeOnClick] Flag to toggle the targe off when clicking outside the toggled content. Defaults to true. * @param {String} [options.closeOnInsideClick] Toggle off when a child element matching this selector is clicked. Set to null to deactivate the check. Defaults to 'a[href]'. * @param {Boolean} [options.initialState] Flag to define initial state. false: off, true: on, null: markup. Defaults to null. * @param {Function} [options.onChangeState] Callback when the toggle state changes. Return `false` to cancel the event. * * @sample Ink_UI_Toggle_1_constructor.html */ function Toggle(){ Common.BaseUIComponent.apply(this, arguments); } Toggle._name = 'Toggle_1'; Toggle._optionDefinition = { target: ['Elements'], triggerEvent: ['String', 'click'], closeOnClick: ['Boolean', true], isAccordion: ['Boolean', false], initialState: ['Boolean', null], // May be true, false, or null to be what it is right now classNameOn: ['String', 'show-all'], classNameOff: ['String', 'hide-all'], closeOnInsideClick: ['String', 'a[href]'], // closes the toggle when a target is clicked and it is a link onChangeState: ['Function', null] }; Toggle.prototype = { /** * Init function called by the constructor * * @method _init * @private */ _init: function(){ var i, len; this._targets = Common.elsOrSelector(this._options.target); // Boolean option handling this._options.closeOnClick = this._options.closeOnClick.toString() === 'true'; // Actually a throolean if (this._options.initialState !== null){ this._options.initialState = this._options.initialState.toString() === 'true'; } else { this._options.initialState = Css.getStyle(this._targets[0], 'display') !== 'none'; } if (this._options.classNameOn !== 'show-all' || this._options.classNameOff !== 'hide-all') { for (i = 0, len = this._targets.length; i < len; i++) { Css.removeClassName(this._targets[i], 'show-all'); Css.removeClassName(this._targets[i], 'hide-all'); } } this._accordion = ( Css.hasClassName(this._element.parentNode,'accordion') || Css.hasClassName(this._targets[0].parentNode,'accordion') ); this._firstTime = true; this._bindEvents(); if (this._options.initialState !== null) { this.setState(this._options.initialState, true); } else { // Add initial classes matching the current "display" of the object. var state = Css.getStyle(this._targets[0], 'display') !== 'none'; this.setState(state, true); } // Aditionally, remove any inline "display" style. for (i = 0, len = this._targets.length; i < len; i++) { if (this._targets[i].style.display) { this._targets[i].style.display = ''; // becomes default } } this._element.setAttribute('data-is-toggle-trigger', 'true'); }, /** * @method _bindEvents * @private */ _bindEvents: function () { if ( this._options.triggerEvent ) { InkEvent.observe( this._element, this._options.triggerEvent, Ink.bind(this._onTriggerEvent, this)); } if( this._options.closeOnClick ){ InkEvent.observe( document, 'click', Ink.bind(this._onOutsideClick, this)); } if( this._options.closeOnInsideClick && this._options.closeOnInsideClick !== 'false') { var sel = this._options.closeOnInsideClick; if (sel.toString() === 'true') { sel = '*'; } InkEvent.observeMulti(this._targets, 'click', Ink.bind(function (e) { if ( InkElement.findUpwardsBySelector(InkEvent.element(e), sel) ) { this.setState(false, true); } }, this)); } }, /** * Event handler. It's responsible for handling the `triggerEvent` as defined in the options. * * This will trigger the toggle. * * @method _onTriggerEvent * @param {Event} event * @private */ _onTriggerEvent: function( event ){ // When the togglee is a child of the toggler, we get the togglee's events here. We have to check that this event is for us. var target = InkEvent.element(event); var isAncestorOfClickedElement = InkArray.some(this._targets, function (thisOne) { return thisOne === target || InkElement.isAncestorOf(thisOne, target); }); if (isAncestorOfClickedElement) { return; } if (this._accordion) { this._updateAccordion(); } var has = this.getState(); this.setState(!has, true); if (!has && this._firstTime) { this._firstTime = false; } InkEvent.stopDefault(event); }, /** * Be compatible with accordions * * @method _updateAccordion **/ _updateAccordion: function () { var elms, accordionElement; if( Css.hasClassName(this._targets[0].parentNode,'accordion') ){ accordionElement = this._targets[0].parentNode; } else { accordionElement = this._targets[0].parentNode.parentNode; } elms = Selector.select('.toggle, .ink-toggle',accordionElement); for(var i=0; i 0) && (targetElm[0] !== this._targets[0]) ){ targetElm[0].style.display = 'none'; } } }, /** * Click handler. Will handle clicks outside the toggle component. * * @method _onOutsideClick * @param {Event} event * @private */ _onOutsideClick: function( event ){ var tgtEl = InkEvent.element(event), shades; if (InkElement.findUpwardsBySelector(tgtEl, '[data-is-toggle-trigger="true"]')) return; var ancestorOfTargets = InkArray.some(this._targets, function (target) { return InkElement.isAncestorOf(target, tgtEl) || target === tgtEl; }); if( (this._element === tgtEl) || InkElement.isAncestorOf(this._element, tgtEl) || ancestorOfTargets) { return; } else if( (shades = Ink.ss('.ink-shade')).length ) { var shadesLength = shades.length; for( var i = 0; i < shadesLength; i++ ){ if( InkElement.isAncestorOf(shades[i],tgtEl) && InkElement.isAncestorOf(shades[i],this._element) ){ return; } } } this.setState(false, true); // dismiss }, /** * Sets the state of the toggle. (on/off) * * @method setState * @param newState {Boolean} New state (on/off) */ setState: function (on, callHandler) { if (on === this.getState()) { return; } if (callHandler && typeof this._options.onChangeState === 'function') { var ret = this._options.onChangeState(on); if (ret === false) { return false; } // Canceled by the event handler } for (var i = 0, len = this._targets.length; i < len; i++) { Css.addRemoveClassName(this._targets[i], this._options.classNameOn, on); Css.addRemoveClassName(this._targets[i], this._options.classNameOff, !on); } Css.addRemoveClassName(this._element, 'active', on); }, /** * Gets the state of the toggle. (on/off) * * @method getState * * @return {Boolean} whether the toggle is toggled on. */ getState: function () { return Css.hasClassName(this._element, 'active'); } }; Common.createUIComponent(Toggle); return Toggle; }); /** * Content Tooltips * @module Ink.UI.Tooltip_1 * @version 1 */ Ink.createModule('Ink.UI.Tooltip', '1', ['Ink.UI.Common_1', 'Ink.Dom.Event_1', 'Ink.Dom.Element_1', 'Ink.Dom.Selector_1', 'Ink.Util.Array_1', 'Ink.Dom.Css_1', 'Ink.Dom.Browser_1'], function (Common, InkEvent, InkElement, Selector, InkArray, Css) { 'use strict'; /** * Tooltips are useful as a means to display information about functionality while avoiding clutter. * * Tooltips show up when you hover elements which "have" tooltips. * * This class will "give" a tooltip to many elements, selected by its first argument (`target`). This is contrary to the other UI modules in Ink, which are created once per element. * * You can define options either through the second argument of the Tooltip constructor, or as data-attributes in each `target` element. Options set through data-attributes all start with "data-tip", and override options passed into the Tooltip constructor. * * @class Ink.UI.Tooltip * @constructor * * @param {DOMElement|String} target Target element or selector of elements, to display the tooltips on. * @param {Object} [options] Options object * @param {String} [options.text] Text content for the tooltip. * @param {String} [options.html] HTML for the tooltip. Same as above, but won't escape HTML. * @param {String} [options.where] Positioning for the tooltip. Options are 'up', 'down', 'left', 'right', 'mousemove' (follows the cursor), and 'mousefix' (stays fixed). Defaults to 'up'. * * @param {String} [options.color] Color of the tooltip. Options are red, orange, blue, green and black. Default is white. * @param {Number} [options.fade] Number of seconds to fade in/out. Defaults to 0.3. * @param {Boolean} [options.forever] Flag to prevent the tooltip from being erased when the mouse hovers away from the target. * @param {Number} [options.timeout] Number of seconds the tooltip will stay open. Useful together with options.forever. Defaults to 0. * @param {Number} [options.delay] Time the tooltip waits until it is displayed. Useful to avoid getting the attention of the user unnecessarily * @param {DOMElement|Selector} [options.template] Element or selector containing HTML to be cloned into the tooltips. Can be a hidden element, because CSS `display` is set to `block`. * @param {String} [options.templatefield] Selector within the template element to choose where the text is inserted into the tooltip. Useful when a wrapper DIV is required. * @param {Number} [options.left] Spacing from the target to the tooltip, when `where` is `mousemove` or `mousefix`. Defaults to 10. * @param {Number} [options.top] Spacing from the target to the tooltip, when `where` is `mousemove` or `mousefix`. Defaults to 10. * @param {Number} [options.spacing] Spacing between the tooltip and the target element, when `where` is not `mousemove` or `mousefix`. Defaults to 8. * * @sample Ink_UI_Tooltip_1.html */ function Tooltip(element, options) { this._init(element, options || {}); } function EachTooltip(root, elm) { this._init(root, elm); } var transitionDurationName, transitionPropertyName, transitionTimingFunctionName; (function () { // Feature detection var test = document.createElement('DIV'); var names = ['transition', 'oTransition', 'msTransition', 'mozTransition', 'webkitTransition']; for (var i = 0; i < names.length; i++) { if (typeof test.style[names[i] + 'Duration'] !== 'undefined') { transitionDurationName = names[i] + 'Duration'; transitionPropertyName = names[i] + 'Property'; transitionTimingFunctionName = names[i] + 'TimingFunction'; break; } } }()); // Body or documentElement var bodies = document.getElementsByTagName('body'); var body = bodies.length ? bodies[0] : document.documentElement; Tooltip.prototype = { _init: function(element, options) { var elements; this.options = Ink.extendObj({ where: 'up', zIndex: 10000, left: 10, top: 10, spacing: 8, forever: 0, color: '', timeout: 0, delay: 0, template: null, templatefield: null, fade: 0.3, text: '' }, options || {}); if (typeof element === 'string') { elements = Selector.select(element); } else if (typeof element === 'object') { elements = [element]; } else { throw 'Element expected'; } this.tooltips = []; for (var i = 0, len = elements.length; i < len; i++) { this.tooltips[i] = new EachTooltip(this, elements[i]); } }, /** * Destroys the tooltips created by this instance * * @method destroy */ destroy: function () { InkArray.each(this.tooltips, function (tooltip) { tooltip._destroy(); }); this.tooltips = null; this.options = null; } }; EachTooltip.prototype = { _oppositeDirections: { left: 'right', right: 'left', up: 'down', down: 'up' }, _init: function(root, elm) { InkEvent.observe(elm, 'mouseover', Ink.bindEvent(this._onMouseOver, this)); InkEvent.observe(elm, 'mouseout', Ink.bindEvent(this._onMouseOut, this)); InkEvent.observe(elm, 'mousemove', Ink.bindEvent(this._onMouseMove, this)); this.root = root; this.element = elm; this._delayTimeout = null; this.tooltip = null; Common.registerInstance(this, this.element); }, _makeTooltip: function (mousePosition) { if (!this._getOpt('text') && !this._getOpt('html') && !InkElement.hasAttribute(this.element, 'title')) { return false; } var tooltip = this._createTooltipElement(); if (this.tooltip) { this._removeTooltip(); } this.tooltip = tooltip; this._fadeInTooltipElement(tooltip); this._placeTooltipElement(tooltip, mousePosition); InkEvent.observe(tooltip, 'mouseover', Ink.bindEvent(this._onTooltipMouseOver, this)); var timeout = this._getFloatOpt('timeout'); if (timeout) { setTimeout(Ink.bind(function () { if (this.tooltip === tooltip) { this._removeTooltip(); } }, this), timeout * 1000); } }, _createTooltipElement: function () { var template = this._getOpt('template'), // User template instead of our HTML templatefield = this._getOpt('templatefield'), tooltip, // The element we float field; // Element where we write our message. Child or same as the above if (template) { // The user told us of a template to use. We copy it. var temp = document.createElement('DIV'); temp.innerHTML = Common.elOrSelector(template, 'options.template').outerHTML; tooltip = temp.firstChild; if (templatefield) { field = Selector.select(templatefield, tooltip); if (field) { field = field[0]; } else { throw 'options.templatefield must be a valid selector within options.template'; } } else { field = tooltip; // Assume same element if user did not specify a field } } else { // We create the default structure tooltip = document.createElement('DIV'); Css.addClassName(tooltip, 'ink-tooltip'); Css.addClassName(tooltip, this._getOpt('color')); field = document.createElement('DIV'); Css.addClassName(field, 'content'); tooltip.appendChild(field); } if (this._getOpt('html')) { field.innerHTML = this._getOpt('html'); } else if (this._getOpt('text')) { InkElement.setTextContent(field, this._getOpt('text')); } else { InkElement.setTextContent(field, this.element.getAttribute('title')); } tooltip.style.display = 'block'; tooltip.style.position = 'absolute'; tooltip.style.zIndex = this._getIntOpt('zIndex'); return tooltip; }, _fadeInTooltipElement: function (tooltip) { var fadeTime = this._getFloatOpt('fade'); if (transitionDurationName && fadeTime) { tooltip.style.opacity = '0'; tooltip.style[transitionDurationName] = fadeTime + 's'; tooltip.style[transitionPropertyName] = 'opacity'; tooltip.style[transitionTimingFunctionName] = 'ease-in-out'; setTimeout(function () { tooltip.style.opacity = '1'; }, 0); // Wait a tick } }, _placeTooltipElement: function (tooltip, mousePosition) { var where = this._getOpt('where'); if (where === 'mousemove' || where === 'mousefix') { var mPos = mousePosition; this._setPos(mPos[0], mPos[1]); body.appendChild(tooltip); } else if (where.match(/(up|down|left|right)/)) { body.appendChild(tooltip); var targetElementPos = InkElement.offset(this.element); var tleft = targetElementPos[0], ttop = targetElementPos[1]; var centerh = (InkElement.elementWidth(this.element) / 2) - (InkElement.elementWidth(tooltip) / 2), centerv = (InkElement.elementHeight(this.element) / 2) - (InkElement.elementHeight(tooltip) / 2); var spacing = this._getIntOpt('spacing'); var tooltipDims = InkElement.elementDimensions(tooltip); var elementDims = InkElement.elementDimensions(this.element); var maxX = InkElement.scrollWidth() + InkElement.viewportWidth(); var maxY = InkElement.scrollHeight() + InkElement.viewportHeight(); where = this._getWhereValueInsideViewport(where, { left: tleft - tooltipDims[0], right: tleft + tooltipDims[0], top: ttop + tooltipDims[1], bottom: ttop + tooltipDims[1] }, { right: maxX, bottom: maxY }); if (where === 'up') { ttop -= tooltipDims[1]; ttop -= spacing; tleft += centerh; } else if (where === 'down') { ttop += elementDims[1]; ttop += spacing; tleft += centerh; } else if (where === 'left') { tleft -= tooltipDims[0]; tleft -= spacing; ttop += centerv; } else if (where === 'right') { tleft += elementDims[0]; tleft += spacing; ttop += centerv; } var arrow = null; if (where.match(/(up|down|left|right)/)) { arrow = document.createElement('SPAN'); Css.addClassName(arrow, 'arrow'); Css.addClassName(arrow, this._oppositeDirections[where]); tooltip.appendChild(arrow); } var tooltipLeft = tleft; var tooltipTop = ttop; var toBottom = (tooltipTop + tooltipDims[1]) - maxY; var toRight = (tooltipLeft + tooltipDims[0]) - maxX; var toLeft = 0 - tooltipLeft; var toTop = 0 - tooltipTop; if (toBottom > 0) { if (arrow) { arrow.style.top = (tooltipDims[1] / 2) + toBottom + 'px'; } tooltipTop -= toBottom; } else if (toTop > 0) { if (arrow) { arrow.style.top = (tooltipDims[1] / 2) - toTop + 'px'; } tooltipTop += toTop; } else if (toRight > 0) { if (arrow) { arrow.style.left = (tooltipDims[0] / 2) + toRight + 'px'; } tooltipLeft -= toRight; } else if (toLeft > 0) { if (arrow) { arrow.style.left = (tooltipDims[0] / 2) - toLeft + 'px'; } tooltipLeft += toLeft; } tooltip.style.left = tooltipLeft + 'px'; tooltip.style.top = tooltipTop + 'px'; } }, /** * Get a value for "where" (left/right/up/down) which doesn't put the * tooltip off the screen * * @method _getWhereValueInsideViewport * @param where {String} "where" value which was given by the user and we might change * @param bbox {BoundingBox} A bounding box like what you get from getBoundingClientRect ({top, bottom, left, right}) with pixel positions from the top left corner of the viewport. * @param viewport {BoundingBox} Bounding box for the viewport. "top" and "left" are omitted because these coordinates are relative to the top-left corner of the viewport so they are zero. * * @TODO: we can't use getBoundingClientRect in this case because it returns {0,0,0,0} on our uncreated tooltip. */ _getWhereValueInsideViewport: function (where, bbox, viewport) { if (where === 'left' && bbox.left < 0) { return 'right'; } else if (where === 'right' && bbox.right > viewport.right) { return 'left'; } else if (where === 'up' && bbox.top < 0) { return 'down'; } else if (where === 'down' && bbox.bottom > viewport.bottom) { return 'up'; } return where; }, _removeTooltip: function() { var tooltip = this.tooltip; if (!tooltip) {return;} var remove = Ink.bind(InkElement.remove, {}, tooltip); if (this._getOpt('where') !== 'mousemove' && transitionDurationName) { tooltip.style.opacity = 0; // remove() will operate on correct tooltip, although this.tooltip === null then setTimeout(remove, this._getFloatOpt('fade') * 1000); } else { remove(); } this.tooltip = null; }, _getOpt: function (option) { var dataAttrVal = InkElement.data(this.element)[InkElement._camelCase('tip-' + option)]; if (dataAttrVal /* either null or "" may signify the absense of this attribute*/) { return dataAttrVal; } var instanceOption = this.root.options[option]; if (typeof instanceOption !== 'undefined') { return instanceOption; } }, _getIntOpt: function (option) { return parseInt(this._getOpt(option), 10); }, _getFloatOpt: function (option) { return parseFloat(this._getOpt(option), 10); }, _destroy: function () { if (this.tooltip) { InkElement.remove(this.tooltip); } this.root = null; // Cyclic reference = memory leaks this.element = null; this.tooltip = null; }, _onMouseOver: function(e) { // on IE < 10 you can't access the mouse event not even a tick after it fired var mousePosition = this._getMousePosition(e); var delay = this._getFloatOpt('delay'); if (delay) { this._delayTimeout = setTimeout(Ink.bind(function () { if (!this.tooltip) { this._makeTooltip(mousePosition); } this._delayTimeout = null; }, this), delay * 1000); } else { this._makeTooltip(mousePosition); } }, _onMouseMove: function(e) { if (this._getOpt('where') === 'mousemove' && this.tooltip) { var mPos = this._getMousePosition(e); this._setPos(mPos[0], mPos[1]); } }, _onMouseOut: function () { if (!this._getIntOpt('forever')) { this._removeTooltip(); } if (this._delayTimeout) { clearTimeout(this._delayTimeout); this._delayTimeout = null; } }, _onTooltipMouseOver: function () { if (this.tooltip) { // If tooltip is already being removed, this has no effect this._removeTooltip(); } }, _setPos: function(left, top) { left += this._getIntOpt('left'); top += this._getIntOpt('top'); var pageDims = this._getPageXY(); if (this.tooltip) { var elmDims = [InkElement.elementWidth(this.tooltip), InkElement.elementHeight(this.tooltip)]; var scrollDim = this._getScroll(); if((elmDims[0] + left - scrollDim[0]) >= (pageDims[0] - 20)) { left = (left - elmDims[0] - this._getIntOpt('left') - 10); } if((elmDims[1] + top - scrollDim[1]) >= (pageDims[1] - 20)) { top = (top - elmDims[1] - this._getIntOpt('top') - 10); } this.tooltip.style.left = left + 'px'; this.tooltip.style.top = top + 'px'; } }, _getPageXY: function() { var cWidth = 0; var cHeight = 0; if( typeof( window.innerWidth ) === 'number' ) { cWidth = window.innerWidth; cHeight = window.innerHeight; } else if( document.documentElement && ( document.documentElement.clientWidth || document.documentElement.clientHeight ) ) { cWidth = document.documentElement.clientWidth; cHeight = document.documentElement.clientHeight; } else if( document.body && ( document.body.clientWidth || document.body.clientHeight ) ) { cWidth = document.body.clientWidth; cHeight = document.body.clientHeight; } return [parseInt(cWidth, 10), parseInt(cHeight, 10)]; }, _getScroll: function() { var dd = document.documentElement, db = document.body; if (dd && (dd.scrollLeft || dd.scrollTop)) { return [dd.scrollLeft, dd.scrollTop]; } else if (db) { return [db.scrollLeft, db.scrollTop]; } else { return [0, 0]; } }, _getMousePosition: function(e) { return [parseInt(InkEvent.pointerX(e), 10), parseInt(InkEvent.pointerY(e), 10)]; } }; return Tooltip; }); /** * Elements in a tree structure * @module Ink.UI.TreeView_1 * @version 1 */ Ink.createModule('Ink.UI.TreeView', '1', ['Ink.UI.Common_1','Ink.Dom.Event_1','Ink.Dom.Css_1','Ink.Dom.Element_1','Ink.Dom.Selector_1','Ink.Util.Array_1'], function(Common, Event, Css, Element, Selector, InkArray ) { 'use strict'; /** * Shows elements in a tree structure which can be expanded and contracted. * A TreeView is built with "node"s and "children". "node"s are `li` tags, and "children" are `ul` tags. * You can build your TreeView out of a regular UL and LI element structure which you already use to display lists with several levels. * If you want a node to be open when the TreeView is built, just add the data-open="true" attribute to it. * * @class Ink.UI.TreeView * @constructor * @version 1 * @param {String|DOMElement} selector Element or selector. * @param {String} [options] Options object, containing: * @param {String} [options.node] Selector for the nodes. Defaults to 'li'. * @param {String} [options.children] Selector for the children. Defaults to 'ul'. * @param {String} [options.parentClass] CSS classes to be added to parent nodes. Defaults to 'parent'. * @param {String} [options.openClass] CSS classes to be added to the icon when a parent is open. Defaults to 'fa fa-minus-circle'. * @param {String} [options.closedClass] CSS classes to be added to the icon when a parent is closed. Defaults to 'fa fa-plus-circle'. * @param {String} [options.hideClass] CSS Class to toggle visibility of the children. Defaults to 'hide-all'. * @param {String} [options.iconTag] The name of icon tag. The component tries to find a tag with that name as a direct child of the node. If it doesn't find it, it creates it. Defaults to 'i'. * @param {Boolean} [options.stopDefault] Flag to stops the default behavior of the click handler. Defaults to true. * @example * * * * @sample Ink_UI_TreeView_1.html */ function TreeView() { Common.BaseUIComponent.apply(this, arguments); } TreeView._name = 'TreeView_1'; TreeView._optionDefinition = { 'node': ['String', 'li'], // [3.0.1] Deprecate this terrible, terrible name 'child': ['String',null], 'children': ['String','ul'], 'parentClass': ['String','parent'], 'openNodeClass': ['String', 'open'], 'openClass': ['String','fa fa-minus-circle'], 'closedClass': ['String','fa fa-plus-circle'], 'hideClass': ['String','hide-all'], 'iconTag': ['String', 'i'], 'stopDefault' : ['Boolean', true] }; TreeView.prototype = { /** * Init function called by the constructor. Sets the necessary event handlers. * * @method _init * @private */ _init: function(){ if (this._options.child) { Ink.warn('Ink.UI.TreeView: options.child is being renamed to options.children.'); this._options.children = this._options.child; } this._handlers = { click: Ink.bindEvent(this._onClick,this) }; Event.on(this._element, 'click', this._options.node, this._handlers.click); InkArray.each(Ink.ss(this._options.node, this._element), Ink.bind(function(item){ if( this.isParent(item) ) { Css.addClassName(item, this._options.parentClass); var isOpen = this.isOpen(item); if( !this._getIcon(item) ){ Element.create(this._options.iconTag, { insertTop: item }); } this._setNodeOpen(item, isOpen); } },this)); }, _getIcon: function (node) { return Ink.s('> ' + this._options.iconTag, node); }, /** * Checks if a node is open. * * @method isOpen * @param {DOMElement} node The tree node to check **/ isOpen: function (node) { if (!this._getChild(node)) { throw new Error('not a node!'); } return Element.data(node).open === 'true' || Css.hasClassName(node, this._options.openNodeClass); }, /** * Checks if a node is a parent. * * @method isParent * @param {DOMElement} node Node to check **/ isParent: function (node) { return Css.hasClassName(node, this._options.parentClass) || this._getChild(node) != null; }, _setNodeOpen: function (node, beOpen) { var child = this._getChild(node); if (child) { Css.setClassName(child, this._options.hideClass, !beOpen); var icon = this._getIcon(node); node.setAttribute('data-open', beOpen); /* * Don't refactor this to * * setClassName(el, className, status); setClassName(el, className, !status); * * because it won't work with multiple classes. * * Doing: * setClassName(el, 'fa fa-whatever', true);setClassName(el, 'fa fa-whatever-else', false); * * will remove 'fa' although it is a class we want. */ var toAdd = beOpen ? this._options.openClass : this._options.closedClass; var toRemove = beOpen ? this._options.closedClass : this._options.openClass; Css.removeClassName(icon, toRemove); Css.addClassName(icon, toAdd); Css.setClassName(node, this._options.openNodeClass, beOpen); } else { Ink.error('Ink.UI.TreeView: node', node, 'is not a node!'); } }, /** * Opens one of the tree nodes * * Make sure you pass the node's DOMElement * @method open * @param {DOMElement} node The node you wish to open. **/ open: function (node) { this._setNodeOpen(node, true); }, /** * Closes one of the tree nodes * * Make sure you pass the node's DOMElement * @method close * @param {DOMElement} node The node you wish to close. **/ close: function (node) { this._setNodeOpen(node, false); }, /** * Toggles a node state * * @method toggle * @param {DOMElement} node The node to toggle. **/ toggle: function (node) { if (this.isOpen(node)) { this.close(node); } else { this.open(node); } }, _getChild: function (node) { return Selector.select(this._options.children, node)[0] || null; }, /** * Handles the click event (as specified in the _init function). * * @method _onClick * @param {Event} event * @private */ _onClick: function(ev){ /** * Summary: * If the clicked element is a "node" as defined in the options, will check if it has any "child". * If so, will toggle its state and stop the event's default behavior if the stopDefault option is true. **/ if (!this.isParent(ev.currentTarget) || Selector.matchesSelector(ev.target, this._options.node) || Selector.matchesSelector(ev.target, this._options.children)) { return; } if (this._options.stopDefault){ ev.preventDefault(); } this.toggle(ev.currentTarget); } }; Common.createUIComponent(TreeView); return TreeView; }); Ink.createModule('Ink.UI.Upload', '1', [ 'Ink.Dom.Event_1', 'Ink.Dom.Element_1', 'Ink.Dom.Browser_1', 'Ink.UI.Common_1' ], function(Event, Element, Browser, Common) { 'use strict'; var DirectoryReader = function(options) { this.init(options); }; DirectoryReader.prototype = { init: function(options) { this._options = Ink.extendObj({ entry: undefined, maxDepth: 10 }, options || {}); try { this._read(); } catch(e) { Ink.error(e); } }, _read: function() { if(!this._options.entry) { Ink.error('You must specify the entry!'); return; } try { this._readDirectories(); } catch(e) { Ink.error(e); } }, _readDirectories: function() { var entries = [], running = false, maxDepth = 0; /* TODO return as tree because much better well */ var _readEntries = Ink.bind(function(currentEntry) { var dir = currentEntry.createReader(); running = true; dir.readEntries(Ink.bind(function(res) { if(res.length > 0) { for(var i = 0, len = res.length; i=0; i--) { if(typeof(arr[i]) === 'undefined' || arr[i] === null || arr[i] === '') { arr.splice(i, 1); } } return arr; } }; var Queue = { lists: [], items: [], /** * Create new queue list * @function create * @public * @param {String} list name * @param {Function} function to iterate on items * @return {Object} list id */ create: function(name) { var id; name = String(name); this.lists.push({name: name}); id = this.lists.length - 1; return id; }, getItems: function(parentId) { if(!parentId) { return this.items; } var items = []; for(var i = 0, len = this.items.length; i=0; i--) { if(this.items[i] && id === this.items[i].parentId) { this.remove(this.items[i].parentId, this.items[i].pid); } } if(!keepList) { this.lists.splice(id, 1); } return true; } catch(e) { Ink.error('Purge: invalid id'); return false; } }, /** * add an item to a list * @function add * @public * @param {String} name * @param {Object} item * @return {Number} pid */ add: function(parentId, item, priority) { if(!this.lists[parentId]) { return false; } if(typeof(item) !== 'object') { item = String(item); } var pid = parseInt(Math.round(Math.random() * 100000) + "" + Math.round(Math.random() * 100000), 10); priority = priority || 0; this.items.push({parentId: parentId, item: item, priority: priority || 0, pid: pid}); return pid; }, /** * View list * @function view * @public * @param {Number} list id * @param {Number} process id * @return {Object} item */ view: function(parentId, pid) { var id = this._searchByPid(parentId, pid); if(id === false) { return false; } return this.items[id]; }, /** * Remove an item * @function remove * @public * @param {Object} item * @return {Object|Boolean} removed item or false if not found */ remove: function(parentId, pid) { try { var id = this._searchByPid(parentId, pid); if(id === false) { return false; } this.items.splice(id, 1); return true; } catch(e) { Ink.error('Remove: invalid id'); return false; } }, _searchByPid: function(parentId, pid) { if(!parentId && typeof(parentId) === 'boolean' || !pid) { return false; } parentId = parseInt(parentId, 10); pid = parseInt(pid, 10); if(isNaN(parentId) || isNaN(pid)) { return false; } for(var i = 0, len = this.items.length; i this._options.minSizeToUseChunks; }, _dropEventHandler: function(ev) { Event.stop(ev); this.publish('DropComplete', ev.dataTransfer); var data = ev.dataTransfer; if(!data || !data.files || !data.files.length) { return false; } this._files = data.files; this._files = Array.prototype.slice.call(this._files || [], 0); // check if webkitGetAsEntry exists on first item if(data.items && data.items[0] && data.items[0].webkitGetAsEntry) { if(!this._options.foldersEnabled) { return setTimeout(Ink.bind(this._addFilesToQueue, this, this._files), 0); } var entry, folders = []; for(var i = ev.dataTransfer.items.length-1; i>=0; i--) { entry = ev.dataTransfer.items[i].webkitGetAsEntry(); if(entry && entry.isDirectory) { folders.push(entry); this._files[i].isDirectory = true; this._files.splice(i, 1); } } // starting callback hell this._addFolderToQueue(folders, Ink.bind(function() { setTimeout(Ink.bind(this._addFilesToQueue, this, this._files), 0); }, this)); } else { setTimeout(Ink.bind(this._addFilesToQueue, this, this._files), 0); } return true; }, _addFolderToQueue: function(folders, cb) { var files = [], invalidFolders = {}; if(!folders || !folders.length) { cb(); return files; } var getFiles = function(entries) { var files = []; for(var i = 0, len = entries.length; i this._options.maxFilesize) { this.publish('MaxSizeFailure', file, this._options.maxFilesize); continue; } fileID = parseInt(Math.round(Math.random() * 100000) + "" + Math.round(Math.random() * 100000), 10); o = { id: i, data: file, fileID: fileID, directory: file.isDirectory }; Queue.add(this._queueId, o); this.publish('FileAddedToQueue', o); } this._processQueue(true); this._files = []; }, _processQueue: function(internalUpload) { if(this._queueRunning) { return false; } this.running = 0; var max = 1, i = 0, items, queueLen = Queue.items.length; this._queueRunning = true; this.interval = setInterval(Ink.bind(function() { if(Queue.items.length === i && this.running === 0) { Queue.purge(this._queueId, true); this._queueRunning = false; clearInterval(this.interval); this.publish('QueueEnd', this._queueId, queueLen); } items = Queue.getItems(this._queueId); if(this.running < max && items[i]) { if(!items[i].canceled) { _doRequest.call(this, items[i].pid, items[i].item.data, items[i].item.fileID, items[i].item.directory, internalUpload); this.running++; i++; } else { var j = i; while(items[j] && items[j].canceled) { i++; j++; } } return true; } return false; }, this), 100); var _doRequest = function(pid, data, fileID, directory, internalUpload) { var o = { file: data, fileID: fileID, cb: Ink.bind(function() { this.running--; }, this) }; if(internalUpload) { if(directory) { // do magic o.cb(); } else { this._upload(o); } } }; return true; }, _upload: function(o) { var file = o.file, xhr = new XMLHttpRequest(), fileID = o.fileID; this.publish('BeforeUpload', file, this._options.extraData, fileID, xhr, this._supportChunks(file.size)); var forceAbort = function(showError) { if(o.cb && typeof(o.cb === 'function')) { o.cb(); } this.publish('OnProgress', { length: file.size, lengthComputable: true, loaded: file.size, total: file.size }, file, fileID); this.publish('EndUpload', file, fileID, (showError ? { error: true } : true)); this.publish('InvalidFile', file, 'name'); xhr.abort(); }; if(this._options.INVALID_FILE_NAME && this._options.INVALID_FILE_NAME instanceof RegExp) { if(this._options.INVALID_FILE_NAME.test(o.file.name)) { forceAbort.call(this); return; } } // If file was renamed, abort it // FU OPERA: Opera always return lastModified date as null if(!file.lastModifiedDate && !Ink.Dom.Browser.OPERA) { forceAbort.call(this, true); return; } xhr.upload.onprogress = Ink.bind(this.publish, this, 'OnProgress', file, fileID); var endpoint, method; if(this._supportChunks(file.size)) { if(file.size <= file.chunk_offset) { endpoint = this._options.endpointChunkCommit; method = 'POST'; } else { endpoint = this._options.endpointChunk; if(file.chunk_upload_id) { endpoint += '?upload_id=' + file.chunk_upload_id; } if(file.chunk_offset) { endpoint += '&offset=' + file.chunk_offset; } method = 'PUT'; } } else { endpoint = this._options.endpoint; method = 'POST'; } xhr.open(method, endpoint, true); xhr.withCredentials = true; xhr.setRequestHeader("x-requested-with", "XMLHttpRequest"); if(this._supportChunks(file.size)) { xhr.setRequestHeader('Content-type', 'application/x-www-form-urlencoded'); } var fd = new FormData(), blob; if("Blob" in window && typeof Blob === 'function') { blob = new Blob([file], { type: file.type }); if(this._supportChunks(file.size)) { file.chunk_offset = file.chunk_offset || 0; blob = blob.slice(file.chunk_offset, file.chunk_offset + this._options.chunkSize); } else { fd.append(this._options.fileFormName, blob, file.name); } } else { fd.append(this._options.fileFormName, file); } if(!this._supportChunks(file.size)) { for(var k in this._options.extraData) { if(this._options.extraData.hasOwnProperty(k)) { fd.append(k, this._options.extraData[k]); } } } else { fd.append('upload_id', file.chunk_upload_id); fd.append('path', file.upload_path); } if(!file.hasParent) { if(!this._supportChunks(file.size)) { xhr.send(fd); } else { if(file.size <= file.chunk_offset) { xhr.send('upload_id=' + file.chunk_upload_id + '&path=' + file.upload_path + '/' + file.name); } else { xhr.send(blob); } } } else { this.publish('cbCreateFolder', file.parentID, file.fullPath, this._options.extraData, this._folders, file.rootPath, Ink.bind(function() { if(!this._supportChunks(file.size)) { xhr.send(fd); } else { if(file.size <= file.chunk_offset) { xhr.send('upload_id=' + file.chunk_upload_id + '&path=' + file.upload_path + '/' + file.name); } else { xhr.send(blob); } } }, this)); } xhr.onload = Ink.bindEvent(function() { /* jshint boss:true */ if(this._supportChunks(file.size) && file.size > file.chunk_offset) { if(xhr.response) { var response = JSON.parse(xhr.response); // check expected offset var invalidOffset = file.chunk_offset && response.offset !== (file.chunk_offset + this._options.chunkSize) && file.size !== response.offset; if(invalidOffset) { if(o.cb) { o.cb(); } this.publish('ErrorUpload', file, fileID); } else { file.chunk_upload_id = response.upload_id; file.chunk_offset = response.offset; file.chunk_expires = response.expires; this._upload(o); } } else { if(o.cb) { o.cb(); } this.publish('ErrorUpload', file, fileID); } return (xhr = null); } if(o.cb) { o.cb(); } if(xhr.responseText && xhr['status'] < 400) { this.publish('EndUpload', file, fileID, xhr.responseText); } else { this.publish('ErrorUpload', file, fileID); } return (xhr = null); }, this); xhr.onerror = Ink.bindEvent(function() { if(o.cb) { o.cb(); } this.publish('ErrorUpload', file, fileID); }, this); xhr.onabort = Ink.bindEvent(function() { if(o.cb) { o.cb(); } this.publish('AbortUpload', file, fileID, { abortAll: Ink.bind(this.abortAll, this), abortOne: Ink.bind(this.abortOne, this) }); }, this); }, abortAll: function() { if(!this._queueRunning) { return false; } clearInterval(this.interval); this._queueRunning = false; Queue.purge(this._queueId, true); return true; }, abortOne: function(id, cb) { var items = Queue.getItems(0), o; for(var i = 0, len = items.length; i