mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-01 14:45:29 +00:00
753 lines
20 KiB
JavaScript
753 lines
20 KiB
JavaScript
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is iq.js.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Ian Gilman <ian@iangilman.com>.
|
|
* Portions created by the Initial Developer are Copyright (C) 2010
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Aza Raskin <aza@mozilla.com>
|
|
* Michael Yoshitaka Erlewine <mitcho@mitcho.com>
|
|
*
|
|
* This file incorporates work from:
|
|
* jQuery JavaScript Library v1.4.2: http://code.jquery.com/jquery-1.4.2.js
|
|
* This incorporated work is covered by the following copyright and
|
|
* permission notice:
|
|
* Copyright 2010, John Resig
|
|
* Dual licensed under the MIT or GPL Version 2 licenses.
|
|
* http://jquery.org/license
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
// **********
|
|
// Title: iq.js
|
|
// Various helper functions, in the vein of jQuery.
|
|
|
|
// ----------
|
|
// Function: iQ
|
|
// Returns an iQClass object which represents an individual element or a group
|
|
// of elements. It works pretty much like jQuery(), with a few exceptions,
|
|
// most notably that you can't use strings with complex html,
|
|
// just simple tags like '<div>'.
|
|
function iQ(selector, context) {
|
|
// The iQ object is actually just the init constructor 'enhanced'
|
|
return new iQClass(selector, context);
|
|
};
|
|
|
|
// A simple way to check for HTML strings or ID strings
|
|
// (both of which we optimize for)
|
|
let quickExpr = /^[^<]*(<[\w\W]+>)[^>]*$|^#([\w-]+)$/;
|
|
|
|
// Match a standalone tag
|
|
let rsingleTag = /^<(\w+)\s*\/?>(?:<\/\1>)?$/;
|
|
|
|
// ##########
|
|
// Class: iQClass
|
|
// The actual class of iQ result objects, representing an individual element
|
|
// or a group of elements.
|
|
//
|
|
// ----------
|
|
// Function: iQClass
|
|
// You don't call this directly; this is what's called by iQ().
|
|
let iQClass = function(selector, context) {
|
|
|
|
// Handle $(""), $(null), or $(undefined)
|
|
if (!selector) {
|
|
return this;
|
|
}
|
|
|
|
// Handle $(DOMElement)
|
|
if (selector.nodeType) {
|
|
this.context = selector;
|
|
this[0] = selector;
|
|
this.length = 1;
|
|
return this;
|
|
}
|
|
|
|
// The body element only exists once, optimize finding it
|
|
if (selector === "body" && !context) {
|
|
this.context = document;
|
|
this[0] = document.body;
|
|
this.selector = "body";
|
|
this.length = 1;
|
|
return this;
|
|
}
|
|
|
|
// Handle HTML strings
|
|
if (typeof selector === "string") {
|
|
// Are we dealing with HTML string or an ID?
|
|
|
|
let match = quickExpr.exec(selector);
|
|
|
|
// Verify a match, and that no context was specified for #id
|
|
if (match && (match[1] || !context)) {
|
|
|
|
// HANDLE $(html) -> $(array)
|
|
if (match[1]) {
|
|
let doc = (context ? context.ownerDocument || context : document);
|
|
|
|
// If a single string is passed in and it's a single tag
|
|
// just do a createElement and skip the rest
|
|
let ret = rsingleTag.exec(selector);
|
|
|
|
if (ret) {
|
|
if (Utils.isPlainObject(context)) {
|
|
Utils.assert('does not support HTML creation with context', false);
|
|
} else {
|
|
selector = [doc.createElement(ret[1])];
|
|
}
|
|
|
|
} else {
|
|
Utils.assert('does not support complex HTML creation', false);
|
|
}
|
|
|
|
return Utils.merge(this, selector);
|
|
|
|
// HANDLE $("#id")
|
|
} else {
|
|
let elem = document.getElementById(match[2]);
|
|
|
|
if (elem) {
|
|
this.length = 1;
|
|
this[0] = elem;
|
|
}
|
|
|
|
this.context = document;
|
|
this.selector = selector;
|
|
return this;
|
|
}
|
|
|
|
// HANDLE $("TAG")
|
|
} else if (!context && /^\w+$/.test(selector)) {
|
|
this.selector = selector;
|
|
this.context = document;
|
|
selector = document.getElementsByTagName(selector);
|
|
return Utils.merge(this, selector);
|
|
|
|
// HANDLE $(expr, $(...))
|
|
} else if (!context || context.iq) {
|
|
return (context || iQ(document)).find(selector);
|
|
|
|
// HANDLE $(expr, context)
|
|
// (which is just equivalent to: $(context).find(expr)
|
|
} else {
|
|
return iQ(context).find(selector);
|
|
}
|
|
|
|
// HANDLE $(function)
|
|
// Shortcut for document ready
|
|
} else if (typeof selector == "function") {
|
|
Utils.log('iQ does not support ready functions');
|
|
return null;
|
|
}
|
|
|
|
if (typeof selector.selector !== "undefined") {
|
|
this.selector = selector.selector;
|
|
this.context = selector.context;
|
|
}
|
|
|
|
let ret = this || [];
|
|
if (selector != null) {
|
|
// The window, strings (and functions) also have 'length'
|
|
if (selector.length == null || typeof selector == "string" || selector.setInterval) {
|
|
Array.push(ret, selector);
|
|
} else {
|
|
Utils.merge(ret, selector);
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
iQClass.prototype = {
|
|
|
|
// Start with an empty selector
|
|
selector: "",
|
|
|
|
// The current version of iQ being used
|
|
iq: "1.4.2",
|
|
|
|
// The default length of a iQ object is 0
|
|
length: 0,
|
|
|
|
// ----------
|
|
// Function: each
|
|
// Execute a callback for every element in the matched set.
|
|
each: function(callback) {
|
|
if (typeof callback != "function") {
|
|
Utils.assert("each's argument must be a function", false);
|
|
return null;
|
|
}
|
|
for (let i = 0; this[i] != null; i++) {
|
|
callback(this[i]);
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: addClass
|
|
// Adds the given class(es) to the receiver.
|
|
addClass: function(value) {
|
|
if (typeof value != "string" || !value) {
|
|
Utils.assert('requires a valid string argument', false);
|
|
return null;
|
|
}
|
|
|
|
let length = this.length;
|
|
for (let i = 0; i < length; i++) {
|
|
let elem = this[i];
|
|
if (elem.nodeType === 1) {
|
|
value.split(/\s+/).forEach(function(className) {
|
|
elem.classList.add(className);
|
|
});
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: removeClass
|
|
// Removes the given class(es) from the receiver.
|
|
removeClass: function(value) {
|
|
if (typeof value != "string" || !value) {
|
|
Utils.assert('does not support function argument', false);
|
|
return null;
|
|
}
|
|
|
|
let length = this.length;
|
|
for (let i = 0; i < length; i++) {
|
|
let elem = this[i];
|
|
if (elem.nodeType === 1 && elem.className) {
|
|
value.split(/\s+/).forEach(function(className) {
|
|
elem.classList.remove(className);
|
|
});
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: hasClass
|
|
// Returns true is the receiver has the given css class.
|
|
hasClass: function(singleClassName) {
|
|
let length = this.length;
|
|
for (let i = 0; i < length; i++) {
|
|
if (this[i].classList.contains(singleClassName)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
// ----------
|
|
// Function: find
|
|
// Searches the receiver and its children, returning a new iQ object with
|
|
// elements that match the given selector.
|
|
find: function(selector) {
|
|
let ret = [];
|
|
let length = 0;
|
|
|
|
let l = this.length;
|
|
for (let i = 0; i < l; i++) {
|
|
length = ret.length;
|
|
try {
|
|
Utils.merge(ret, this[i].querySelectorAll(selector));
|
|
} catch(e) {
|
|
Utils.log('iQ.find error (bad selector)', e);
|
|
}
|
|
|
|
if (i > 0) {
|
|
// Make sure that the results are unique
|
|
for (let n = length; n < ret.length; n++) {
|
|
for (let r = 0; r < length; r++) {
|
|
if (ret[r] === ret[n]) {
|
|
ret.splice(n--, 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return iQ(ret);
|
|
},
|
|
|
|
// ----------
|
|
// Function: remove
|
|
// Removes the receiver from the DOM.
|
|
remove: function() {
|
|
for (let i = 0; this[i] != null; i++) {
|
|
let elem = this[i];
|
|
if (elem.parentNode) {
|
|
elem.parentNode.removeChild(elem);
|
|
}
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: empty
|
|
// Removes all of the reciever's children and HTML content from the DOM.
|
|
empty: function() {
|
|
for (let i = 0; this[i] != null; i++) {
|
|
let elem = this[i];
|
|
while (elem.firstChild) {
|
|
elem.removeChild(elem.firstChild);
|
|
}
|
|
}
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: width
|
|
// Returns the width of the receiver.
|
|
width: function() {
|
|
return parseInt(this.css('width'));
|
|
},
|
|
|
|
// ----------
|
|
// Function: height
|
|
// Returns the height of the receiver.
|
|
height: function() {
|
|
return parseInt(this.css('height'));
|
|
},
|
|
|
|
// ----------
|
|
// Function: position
|
|
// Returns an object with the receiver's position in left and top
|
|
// properties.
|
|
position: function() {
|
|
return {
|
|
left: parseInt(this.css('left')),
|
|
top: parseInt(this.css('top'))
|
|
};
|
|
},
|
|
|
|
// ----------
|
|
// Function: bounds
|
|
// Returns a <Rect> with the receiver's bounds.
|
|
bounds: function() {
|
|
let p = this.position();
|
|
return new Rect(p.left, p.top, this.width(), this.height());
|
|
},
|
|
|
|
// ----------
|
|
// Function: data
|
|
// Pass in both key and value to attach some data to the receiver;
|
|
// pass in just key to retrieve it.
|
|
data: function(key, value) {
|
|
let data = null;
|
|
if (typeof value === "undefined") {
|
|
Utils.assert('does not yet support multi-objects (or null objects)', this.length == 1);
|
|
data = this[0].iQData;
|
|
if (data)
|
|
return data[key];
|
|
else
|
|
return null;
|
|
}
|
|
|
|
for (let i = 0; this[i] != null; i++) {
|
|
let elem = this[i];
|
|
data = elem.iQData;
|
|
|
|
if (!data)
|
|
data = elem.iQData = {};
|
|
|
|
data[key] = value;
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: html
|
|
// Given a value, sets the receiver's innerHTML to it; otherwise returns
|
|
// what's already there.
|
|
html: function(value) {
|
|
Utils.assert('does not yet support multi-objects (or null objects)', this.length == 1);
|
|
if (typeof value === "undefined")
|
|
return this[0].innerHTML;
|
|
|
|
this[0].innerHTML = value;
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: text
|
|
// Given a value, sets the receiver's textContent to it; otherwise returns
|
|
// what's already there.
|
|
text: function(value) {
|
|
Utils.assert('does not yet support multi-objects (or null objects)', this.length == 1);
|
|
if (typeof value === "undefined") {
|
|
return this[0].textContent;
|
|
}
|
|
|
|
return this.empty().append((this[0] && this[0].ownerDocument || document).createTextNode(value));
|
|
},
|
|
|
|
// ----------
|
|
// Function: val
|
|
// Given a value, sets the receiver's value to it; otherwise returns what's already there.
|
|
val: function(value) {
|
|
Utils.assert('does not yet support multi-objects (or null objects)', this.length == 1);
|
|
if (typeof value === "undefined") {
|
|
return this[0].value;
|
|
}
|
|
|
|
this[0].value = value;
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: appendTo
|
|
// Appends the receiver to the result of iQ(selector).
|
|
appendTo: function(selector) {
|
|
Utils.assert('does not yet support multi-objects (or null objects)', this.length == 1);
|
|
iQ(selector).append(this);
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: append
|
|
// Appends the result of iQ(selector) to the receiver.
|
|
append: function(selector) {
|
|
let object = iQ(selector);
|
|
Utils.assert('does not yet support multi-objects (or null objects)', object.length == 1 && this.length == 1);
|
|
this[0].appendChild(object[0]);
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: attr
|
|
// Sets or gets an attribute on the element(s).
|
|
attr: function(key, value) {
|
|
try {
|
|
Utils.assert('string key', typeof key === 'string');
|
|
if (typeof value === "undefined") {
|
|
Utils.assert('retrieval does not support multi-objects (or null objects)', this.length == 1);
|
|
return this[0].getAttribute(key);
|
|
}
|
|
for (let i = 0; this[i] != null; i++) {
|
|
this[i].setAttribute(key, value);
|
|
}
|
|
} catch(e) {
|
|
Utils.log(e);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: css
|
|
// Sets or gets CSS properties on the receiver. When setting certain numerical properties,
|
|
// will automatically add "px".
|
|
//
|
|
// Possible call patterns:
|
|
// a: object, b: undefined - sets with properties from a
|
|
// a: string, b: undefined - gets property specified by a
|
|
// a: string, b: string/number - sets property specified by a to b
|
|
css: function(a, b) {
|
|
let properties = null;
|
|
|
|
if (typeof a === 'string') {
|
|
let key = a;
|
|
if (typeof b === "undefined") {
|
|
Utils.assert('retrieval does not support multi-objects (or null objects)', this.length == 1);
|
|
|
|
let substitutions = {
|
|
'MozTransform': '-moz-transform',
|
|
'zIndex': 'z-index'
|
|
};
|
|
|
|
return window.getComputedStyle(this[0], null).getPropertyValue(substitutions[key] || key);
|
|
}
|
|
properties = {};
|
|
properties[key] = b;
|
|
} else {
|
|
properties = a;
|
|
}
|
|
|
|
let pixels = {
|
|
'left': true,
|
|
'top': true,
|
|
'right': true,
|
|
'bottom': true,
|
|
'width': true,
|
|
'height': true
|
|
};
|
|
|
|
for (let i = 0; this[i] != null; i++) {
|
|
let elem = this[i];
|
|
for (let key in properties) {
|
|
let value = properties[key];
|
|
if (pixels[key] && typeof(value) != 'string')
|
|
value += 'px';
|
|
|
|
if (key.indexOf('-') != -1)
|
|
elem.style.setProperty(key, value, '');
|
|
else
|
|
elem.style[key] = value;
|
|
}
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: animate
|
|
// Uses CSS transitions to animate the element.
|
|
//
|
|
// Parameters:
|
|
// css - an object map of the CSS properties to change
|
|
// options - an object with various properites (see below)
|
|
//
|
|
// Possible "options" properties:
|
|
// duration - how long to animate, in milliseconds
|
|
// easing - easing function to use. Possibilities include
|
|
// "tabviewBounce", "easeInQuad". Default is "ease".
|
|
// complete - function to call once the animation is done, takes nothing
|
|
// in, but "this" is set to the element that was animated.
|
|
animate: function(css, options) {
|
|
try {
|
|
Utils.assert('does not yet support multi-objects (or null objects)', this.length == 1);
|
|
|
|
if (!options)
|
|
options = {};
|
|
|
|
let easings = {
|
|
tabviewBounce: "cubic-bezier(0.0, 0.63, .6, 1.29)",
|
|
easeInQuad: 'ease-in', // TODO: make it a real easeInQuad, or decide we don't care
|
|
fast: 'cubic-bezier(0.7,0,1,1)'
|
|
};
|
|
|
|
let duration = (options.duration || 400);
|
|
let easing = (easings[options.easing] || 'ease');
|
|
|
|
// The latest versions of Firefox do not animate from a non-explicitly
|
|
// set css properties. So for each element to be animated, go through
|
|
// and explicitly define 'em.
|
|
let rupper = /([A-Z])/g;
|
|
this.each(function(elem) {
|
|
let cStyle = window.getComputedStyle(elem, null);
|
|
for (let prop in css) {
|
|
prop = prop.replace(rupper, "-$1").toLowerCase();
|
|
iQ(elem).css(prop, cStyle.getPropertyValue(prop));
|
|
}
|
|
});
|
|
|
|
this.css({
|
|
'-moz-transition-property': 'all', // TODO: just animate the properties we're changing
|
|
'-moz-transition-duration': (duration / 1000) + 's',
|
|
'-moz-transition-timing-function': easing
|
|
});
|
|
|
|
this.css(css);
|
|
|
|
let self = this;
|
|
Utils.timeout(function() {
|
|
self.css({
|
|
'-moz-transition-property': 'none',
|
|
'-moz-transition-duration': '',
|
|
'-moz-transition-timing-function': ''
|
|
});
|
|
|
|
if (typeof options.complete == "function")
|
|
options.complete.apply(self);
|
|
}, duration);
|
|
} catch(e) {
|
|
Utils.log(e);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: fadeOut
|
|
// Animates the receiver to full transparency. Calls callback on completion.
|
|
fadeOut: function(callback) {
|
|
Utils.assert('does not yet support duration', typeof callback == "function"
|
|
|| typeof callback === "undefined");
|
|
|
|
this.animate({
|
|
opacity: 0
|
|
}, {
|
|
duration: 400,
|
|
complete: function() {
|
|
iQ(this).css({display: 'none'});
|
|
if (typeof callback == "function")
|
|
callback.apply(this);
|
|
}
|
|
});
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: fadeIn
|
|
// Animates the receiver to full opacity.
|
|
fadeIn: function() {
|
|
try {
|
|
this.css({display: ''});
|
|
this.animate({
|
|
opacity: 1
|
|
}, {
|
|
duration: 400
|
|
});
|
|
} catch(e) {
|
|
Utils.log(e);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: hide
|
|
// Hides the receiver.
|
|
hide: function() {
|
|
try {
|
|
this.css({display: 'none', opacity: 0});
|
|
} catch(e) {
|
|
Utils.log(e);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: show
|
|
// Shows the receiver.
|
|
show: function() {
|
|
try {
|
|
this.css({display: '', opacity: 1});
|
|
} catch(e) {
|
|
Utils.log(e);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: bind
|
|
// Binds the given function to the given event type. Also wraps the function
|
|
// in a try/catch block that does a Utils.log on any errors.
|
|
bind: function(type, func) {
|
|
Utils.assert('does not support eventData argument', typeof func == "function");
|
|
|
|
let handler = function(event) {
|
|
try {
|
|
return func.apply(this, [event]);
|
|
} catch(e) {
|
|
Utils.log(e);
|
|
}
|
|
};
|
|
|
|
for (let i = 0; this[i] != null; i++) {
|
|
let elem = this[i];
|
|
if (!elem.iQEventData)
|
|
elem.iQEventData = {};
|
|
|
|
if (!elem.iQEventData[type])
|
|
elem.iQEventData[type] = [];
|
|
|
|
elem.iQEventData[type].push({
|
|
original: func,
|
|
modified: handler
|
|
});
|
|
|
|
elem.addEventListener(type, handler, false);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
// ----------
|
|
// Function: one
|
|
// Binds the given function to the given event type, but only for one call;
|
|
// automatically unbinds after the event fires once.
|
|
one: function(type, func) {
|
|
Utils.assert('does not support eventData argument', typeof func == "function");
|
|
|
|
let handler = function(e) {
|
|
iQ(this).unbind(type, handler);
|
|
return func.apply(this, [e]);
|
|
};
|
|
|
|
return this.bind(type, handler);
|
|
},
|
|
|
|
// ----------
|
|
// Function: unbind
|
|
// Unbinds the given function from the given event type.
|
|
unbind: function(type, func) {
|
|
Utils.assert('Must provide a function', typeof func == "function");
|
|
|
|
for (let i = 0; this[i] != null; i++) {
|
|
let elem = this[i];
|
|
let handler = func;
|
|
if (elem.iQEventData && elem.iQEventData[type]) {
|
|
let count = elem.iQEventData[type].length;
|
|
for (let a = 0; a < count; a++) {
|
|
let pair = elem.iQEventData[type][a];
|
|
if (pair.original == func) {
|
|
handler = pair.modified;
|
|
elem.iQEventData[type].splice(a, 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
elem.removeEventListener(type, handler, false);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
};
|
|
|
|
// ----------
|
|
// Create various event aliases
|
|
let events = [
|
|
'keyup',
|
|
'keydown',
|
|
'mouseup',
|
|
'mousedown',
|
|
'mouseover',
|
|
'mouseout',
|
|
'mousemove',
|
|
'click',
|
|
'resize',
|
|
'change',
|
|
'blur',
|
|
'focus'
|
|
];
|
|
|
|
events.forEach(function(event) {
|
|
iQClass.prototype[event] = function(func) {
|
|
return this.bind(event, func);
|
|
};
|
|
});
|