OpenAuth.Net/OpenAuth.Mvc/Content/ace/js/ace-extra.js

448 lines
12 KiB
JavaScript
Raw Normal View History

2016-10-14 11:22:16 +08:00
(function(){
if( !('ace' in window) ) window['ace'] = {}
ace.config = {
storage_method: 0, //0 means use localStorage if available otherwise cookies, 1 means localStorage, 2 means cookies
cookie_expiry : 604800, //(cookie only) 1 week duration for saved settings
cookie_path: ''//(cookie only)
}
if( !('vars' in window['ace']) ) window['ace'].vars = {}
ace.vars['very_old_ie'] = !('querySelector' in document.documentElement);
ace.settings = {
saveState : function(element, attrName, attrVal, append) {
if( !element || (typeof element == 'string' && !(element = document.getElementById(element))) || !element.hasAttribute('id') ) return false;
if( !ace.hasClass(element, 'ace-save-state') ) return false;
var attrName = attrName || 'class';
var id = element.getAttribute('id');
var attrList = ace.data.get('state', 'id-'+id) || {};
if(typeof attrList == 'string') {
try {
attrList = JSON.parse(attrList);
}
catch(e) {
attrList = {}
}
}
var newVal, hasCustomVal = typeof attrVal !== 'undefined', $delete = false;
var re1 = /class/i
var re2 = /checked|disabled|readonly|value/i
if(re2.test(attrName)) newVal = hasCustomVal ? attrVal : element[attrName];
else {
if(element.hasAttribute(attrName)) {
newVal = hasCustomVal ? attrVal : element.getAttribute(attrName);
}
else if(!hasCustomVal) $delete = true;
//delete this, because element has no such attribute and we haven't given a custom value! (no attrVal)
}
if($delete) {
delete attrList[attrName];
}
else {
//save class names as an object which indicated which classes should be included or excluded (true/false)
if( re1.test(attrName) ) {//class
if( !attrList.hasOwnProperty(attrName) ) attrList[attrName] = {}
if(append === true) {
//append to previous value
attrList[attrName][newVal] = 1;
}
else if(append === false) {
//remove from previous value
attrList[attrName][newVal] = -1;
}
else {
attrList[attrName]['className'] = newVal;
}
}
else {
attrList[attrName] = newVal;
}
}
ace.data.set('state', 'id-'+id , JSON.stringify(attrList));
},
loadState : function(element, attrName) {
if( !element || (typeof element == 'string' && !(element = document.getElementById(element))) || !element.hasAttribute('id') ) return false;
var id = element.getAttribute('id');
var attrList = ace.data.get('state', 'id-'+id) || {};
if(typeof attrList == 'string') {
try {
attrList = JSON.parse(attrList);
}
catch(e) {
attrList = {}
}
}
var setAttr = function(element, attr, val) {
var re1 = /class/i
var re2 = /checked|disabled|readonly|value/i
if(re1.test(attr)) {
if(typeof val === 'object') {
if('className' in val) element.setAttribute('class', val['className']);
for(var key in val) if(val.hasOwnProperty(key)) {
var append = val[key];
if(append == 1) ace.addClass(element, key);
else if(append == -1) ace.removeClass(element, key);
}
}
//else if(typeof ace.addClass(element, val);
}
else if(re2.test(attr)) element[attr] = val;
else element.setAttribute(attr, val);
}
if(attrName !== undefined) {
if(attrList.hasOwnProperty(attrName) && attrList[attrName] !== null) setAttr(element, attrName, attrList[attrName]);
}
else {
for(var name in attrList) {
if(attrList.hasOwnProperty(name) && attrList[name] !== null) setAttr(element, name, attrList[name]);
}
}
},
clearState : function(element) {
var id = null;
if(typeof element === 'string') {
id = element;
}
else if('hasAttribute' in element && element.hasAttribute('id')) {
id = element.getAttribute('id');
}
if(id) ace.data.remove('state', 'id-'+id);
}
};
(function() {
//detect if it is supported
//https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_Animations/Detecting_CSS_animation_support
var animationSupport = function() {
var animation = false,
animationstring = 'animation',
keyframeprefix = '',
domPrefixes = 'Webkit Moz O ms Khtml'.split(' '),
pfx = '',
elm = document.createElement('div');
if( elm.style.animationName !== undefined ) { animation = true; }
if( animation === false ) {
for( var i = 0; i < domPrefixes.length; i++ ) {
if( elm.style[ domPrefixes[i] + 'AnimationName' ] !== undefined ) {
pfx = domPrefixes[ i ];
animationstring = pfx + 'Animation';
keyframeprefix = '-' + pfx.toLowerCase() + '-';
animation = true;
break;
}
}
}
return animation;
}
ace.vars['animation'] = animationSupport();
if( ace.vars['animation'] ) {
//based on http://www.backalleycoder.com/2012/04/25/i-want-a-damnodeinserted/
var animationCSS = "@keyframes nodeInserted{from{outline-color:#fff}to{outline-color:#000}}@-moz-keyframes nodeInserted{from{outline-color:#fff}to{outline-color:#000}}@-webkit-keyframes nodeInserted{from{outline-color:#fff}to{outline-color:#000}}@-ms-keyframes nodeInserted{from{outline-color:#fff}to{outline-color:#000}}@-o-keyframes nodeInserted{from{outline-color:#fff}to{outline-color:#000}}.ace-save-state{animation-duration:10ms;-o-animation-duration:10ms;-ms-animation-duration:10ms;-moz-animation-duration:10ms;-webkit-animation-duration:10ms;animation-delay:0s;-o-animation-delay:0s;-ms-animation-delay:0s;-moz-animation-delay:0s;-webkit-animation-delay:0s;animation-name:nodeInserted;-o-animation-name:nodeInserted;-ms-animation-name:nodeInserted;-moz-animation-name:nodeInserted;-webkit-animation-name:nodeInserted}";
var animationNode = document.createElement('style');
animationNode.innerHTML = animationCSS;
document.head.appendChild(animationNode);
var domInsertEvent = function(event) {
var element = event.target;
if( !element || !ace.hasClass(element, 'ace-save-state') ) return;
ace.settings.loadState(element);
}
document.addEventListener('animationstart', domInsertEvent, false);
document.addEventListener('MSAnimationStart', domInsertEvent, false);
document.addEventListener('webkitAnimationStart', domInsertEvent, false);
}
else {
//if animation events are not supported, wait for document ready event
var documentReady = function() {
var list = document.querySelectorAll('.ace-save-state');
for(var i = 0 ; i < list.length ; i++) ace.settings.loadState(list[i]);
}
if(document.readyState == 'complete') documentReady();
else if(document.addEventListener) document.addEventListener('DOMContentLoaded', documentReady, false);
else if(document.attachEvent) document.attachEvent('onreadystatechange', function(){
if (document.readyState == 'complete') documentReady();
});
}
})();
//save/retrieve data using localStorage or cookie
//method == 1, use localStorage
//method == 2, use cookies
//method not specified, use localStorage if available, otherwise cookies
ace.data_storage = function(method, undefined) {
var prefix = 'ace_';
var storage = null;
var type = 0;
if((method == 1 || method === undefined || method == 0) && 'localStorage' in window && window['localStorage'] !== null) {
storage = ace.storage;
type = 1;
}
else if(storage == null && (method == 2 || method === undefined) && 'cookie' in document && document['cookie'] !== null) {
storage = ace.cookie;
type = 2;
}
this.set = function(namespace, key, value, path, is_obj, undefined) {
if(!storage) return;
if(value === undefined) {//no namespace here?
value = key;
key = namespace;
if(value == null) storage.remove(prefix+key)
else {
if(type == 1)
storage.set(prefix+key, value)
else if(type == 2)
storage.set(prefix+key, value, ace.config.cookie_expiry, path || ace.config.cookie_path)
}
}
else {
if(type == 1) {//localStorage
if(value == null) storage.remove(prefix+namespace+'_'+key)
else {
if(is_obj && typeof value == 'object') {
value = JSON.stringify(value);
}
storage.set(prefix+namespace+'_'+key, value);
}
}
else if(type == 2) {//cookie
var val = storage.get(prefix+namespace);
var tmp = val ? JSON.parse(val) : {};
if(value == null) {
delete tmp[key];//remove
if(ace.sizeof(tmp) == 0) {//no other elements in this cookie, so delete it
storage.remove(prefix+namespace);
return;
}
}
else {
tmp[key] = value;
}
storage.set(prefix+namespace , JSON.stringify(tmp), ace.config.cookie_expiry, path || ace.config.cookie_path)
}
}
}
this.get = function(namespace, key, is_obj, undefined) {
if(!storage) return null;
if(key === undefined) {//no namespace here?
key = namespace;
return storage.get(prefix+key);
}
else {
if(type == 1) {//localStorage
var value = storage.get(prefix+namespace+'_'+key);
if(is_obj && value) {
try { value = JSON.parse(value) } catch(e) {}
}
return value;
}
else if(type == 2) {//cookie
var val = storage.get(prefix+namespace);
var tmp = val ? JSON.parse(val) : {};
return key in tmp ? tmp[key] : null;
}
}
}
this.remove = function(namespace, key, undefined) {
if(!storage) return;
if(key === undefined) {
key = namespace
this.set(key, null);
}
else {
this.set(namespace, key, null);
}
}
}
//cookie storage
ace.cookie = {
// The following settingFunction are from Cookie.js class in TinyMCE, Moxiecode, used under LGPL.
/**
* Get a cookie.
*/
get : function(name) {
var cookie = document.cookie, e, p = name + "=", b;
if ( !cookie )
return;
b = cookie.indexOf("; " + p);
if ( b == -1 ) {
b = cookie.indexOf(p);
if ( b != 0 )
return null;
} else {
b += 2;
}
e = cookie.indexOf(";", b);
if ( e == -1 )
e = cookie.length;
return decodeURIComponent( cookie.substring(b + p.length, e) );
},
/**
* Set a cookie.
*
* The 'expires' arg can be either a JS Date() object set to the expiration date (back-compat)
* or the number of seconds until expiration
*/
set : function(name, value, expires, path, domain, secure) {
var d = new Date();
if ( typeof(expires) == 'object' && expires.toGMTString ) {
expires = expires.toGMTString();
} else if ( parseInt(expires, 10) ) {
d.setTime( d.getTime() + ( parseInt(expires, 10) * 1000 ) ); // time must be in miliseconds
expires = d.toGMTString();
} else {
expires = '';
}
document.cookie = name + "=" + encodeURIComponent(value) +
((expires) ? "; expires=" + expires : "") +
((path) ? "; path=" + path : "") +
((domain) ? "; domain=" + domain : "") +
((secure) ? "; secure" : "");
},
/**
* Remove a cookie.
*
* This is done by setting it to an empty value and setting the expiration time in the past.
*/
remove : function(name, path) {
this.set(name, '', -1000, path);
}
};
//local storage
ace.storage = {
get: function(key) {
return window['localStorage'].getItem(key);
},
set: function(key, value) {
window['localStorage'].setItem(key , value);
},
remove: function(key) {
window['localStorage'].removeItem(key);
}
};
//count the number of properties in an object
//useful for getting the number of elements in an associative array
ace.sizeof = function(obj) {
var size = 0;
for(var key in obj) if(obj.hasOwnProperty(key)) size++;
return size;
}
//because jQuery may not be loaded at this stage, we use our own toggleClass
ace.hasClass = function(elem, className) { return (" " + elem.className + " ").indexOf(" " + className + " ") > -1; }
ace.addClass = function(elem, className) {
var parts = className.split(/\s+/);
for(var p = 0; p < parts.length; p++) {
if ( parts[p].length > 0 && !ace.hasClass(elem, parts[p]) ) {
var currentClass = elem.className;
elem.className = currentClass + (currentClass.length ? " " : "") + parts[p];
}
}
}
ace.removeClass = function(elem, className) {
var parts = className.split(/\s+/);
for(var p = 0; p < parts.length; p++) {
if( parts[p].length > 0 ) ace.replaceClass(elem, parts[p]);
}
ace.replaceClass(elem, className);
}
ace.replaceClass = function(elem, className, newClass) {
var classToRemove = new RegExp(("(^|\\s)" + className + "(\\s|$)"), "i");
elem.className = elem.className.replace(classToRemove, function (match, p1, p2) {
return newClass ? (p1 + newClass + p2) : " ";
}).replace(/^\s+|\s+$/g, "");
}
ace.toggleClass = function(elem, className) {
if(ace.hasClass(elem, className))
ace.removeClass(elem, className);
else ace.addClass(elem, className);
}
ace.isHTMlElement = function(elem) {
return window.HTMLElement ? elem instanceof HTMLElement : ('nodeType' in elem ? elem.nodeType == 1 : false);
}
//data_storage instance used inside ace.settings etc
ace.data = new ace.data_storage(ace.config.storage_method);
})();