You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
719 lines
22 KiB
719 lines
22 KiB
/*
|
|
* # Semantic - API
|
|
* http://github.com/semantic-org/semantic-ui/
|
|
*
|
|
*
|
|
* Copyright 2013 Contributors
|
|
* Released under the MIT license
|
|
* http://opensource.org/licenses/MIT
|
|
*
|
|
*/
|
|
|
|
;(function ( $, window, document, undefined ) {
|
|
|
|
$.api = $.fn.api = function(parameters) {
|
|
|
|
var
|
|
// use window context if none specified
|
|
$allModules = $.isFunction(this)
|
|
? $(window)
|
|
: $(this),
|
|
moduleSelector = $allModules.selector || '',
|
|
time = new Date().getTime(),
|
|
performance = [],
|
|
|
|
query = arguments[0],
|
|
methodInvoked = (typeof query == 'string'),
|
|
queryArguments = [].slice.call(arguments, 1),
|
|
returnedValue
|
|
;
|
|
|
|
$allModules
|
|
.each(function() {
|
|
var
|
|
settings = $.extend(true, {}, $.fn.api.settings, parameters),
|
|
|
|
// internal aliases
|
|
namespace = settings.namespace,
|
|
metadata = settings.metadata,
|
|
selector = settings.selector,
|
|
error = settings.error,
|
|
className = settings.className,
|
|
|
|
// define namespaces for modules
|
|
eventNamespace = '.' + namespace,
|
|
moduleNamespace = 'module-' + namespace,
|
|
|
|
// element that creates request
|
|
$module = $(this),
|
|
$form = $module.closest(selector.form),
|
|
|
|
// context used for state
|
|
$context = (settings.stateContext)
|
|
? $(settings.stateContext)
|
|
: $module,
|
|
|
|
// request details
|
|
ajaxSettings,
|
|
requestSettings,
|
|
url,
|
|
data,
|
|
|
|
// standard module
|
|
element = this,
|
|
instance = $module.data(moduleNamespace),
|
|
module
|
|
;
|
|
|
|
module = {
|
|
|
|
initialize: function() {
|
|
var
|
|
triggerEvent = module.get.event()
|
|
;
|
|
if(!methodInvoked) {
|
|
if( triggerEvent ) {
|
|
module.debug('Attaching API events to element', triggerEvent);
|
|
$module
|
|
.on(triggerEvent + eventNamespace, module.query)
|
|
;
|
|
}
|
|
else {
|
|
module.query();
|
|
}
|
|
}
|
|
module.instantiate();
|
|
},
|
|
|
|
instantiate: function() {
|
|
module.verbose('Storing instance of module', module);
|
|
instance = module;
|
|
$module
|
|
.data(moduleNamespace, instance)
|
|
;
|
|
},
|
|
|
|
destroy: function() {
|
|
module.verbose('Destroying previous module for', element);
|
|
$module
|
|
.removeData(moduleNamespace)
|
|
.off(eventNamespace)
|
|
;
|
|
},
|
|
|
|
query: function() {
|
|
|
|
// determine if an api event already occurred
|
|
if(module.is.loading() && !settings.allowMultiple) {
|
|
module.debug('Request cancelled previous request is still pending');
|
|
return;
|
|
}
|
|
|
|
// pass element metadata to url (value, text)
|
|
if(settings.defaultData) {
|
|
$.extend(true, settings.urlData, module.get.defaultData());
|
|
}
|
|
|
|
// Add form content
|
|
if(settings.serializeForm) {
|
|
$.extend(true, settings.data, module.get.formData());
|
|
}
|
|
|
|
// call beforesend and get any settings changes
|
|
requestSettings = module.get.settings();
|
|
|
|
// check if beforesend cancelled request
|
|
if(requestSettings === false) {
|
|
module.error(error.beforeSend);
|
|
return;
|
|
}
|
|
|
|
if(settings.url) {
|
|
// override with url if specified
|
|
module.debug('Using specified url', url);
|
|
url = module.add.urlData( settings.url );
|
|
}
|
|
else {
|
|
// otherwise find url from api endpoints
|
|
url = module.add.urlData( module.get.templateURL() );
|
|
module.debug('API url resolved to', url);
|
|
}
|
|
|
|
// exit conditions reached, missing url parameters
|
|
if( !url ) {
|
|
module.error(error.missingURL);
|
|
return;
|
|
}
|
|
|
|
// add loading state
|
|
module.set.loading();
|
|
|
|
// look for jQuery ajax parameters in settings
|
|
ajaxSettings = $.extend(true, {}, settings, {
|
|
type : settings.method || settings.type,
|
|
data : data,
|
|
url : url,
|
|
beforeSend : settings.beforeXHR,
|
|
success : function() {},
|
|
failure : function() {},
|
|
complete : function() {}
|
|
});
|
|
|
|
module.verbose('Creating AJAX request with settings', ajaxSettings);
|
|
|
|
// request provides a wrapper around xhr
|
|
module.request = module.create.request();
|
|
module.xhr = module.create.xhr();
|
|
|
|
},
|
|
|
|
|
|
is: {
|
|
loading: function() {
|
|
return (module.request && module.request.state() == 'pending');
|
|
}
|
|
},
|
|
|
|
was: {
|
|
succesful: function() {
|
|
return (module.request && module.request.state() == 'resolved');
|
|
},
|
|
failure: function() {
|
|
return (module.request && module.request.state() == 'rejected');
|
|
},
|
|
complete: function() {
|
|
return (module.request && (module.request.state() == 'resolved' || module.request.state() == 'rejected') );
|
|
}
|
|
},
|
|
|
|
add: {
|
|
urlData: function(url, urlData) {
|
|
var
|
|
urlVariables
|
|
;
|
|
if(url) {
|
|
urlVariables = url.match(settings.regExpTemplate);
|
|
urlData = urlData || settings.urlData;
|
|
|
|
if(urlVariables) {
|
|
module.debug('Looking for URL variables', urlVariables);
|
|
$.each(urlVariables, function(index, templateValue){
|
|
var
|
|
term = templateValue.substr( 2, templateValue.length - 3),
|
|
termValue = ($.isPlainObject(urlData) && urlData[term] !== undefined)
|
|
? urlData[term]
|
|
: ($module.data(term) !== undefined)
|
|
? $module.data(term)
|
|
: urlData[term]
|
|
;
|
|
module.verbose('Looking for variable', term);
|
|
// remove optional value
|
|
if(termValue === false) {
|
|
module.debug('Removing variable from URL', urlVariables);
|
|
url = url.replace('/' + templateValue, '');
|
|
}
|
|
// undefined condition
|
|
else if(termValue === undefined || !termValue) {
|
|
module.error(error.missingParameter, term, url);
|
|
url = false;
|
|
return false;
|
|
}
|
|
else {
|
|
url = url.replace(templateValue, termValue);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
return url;
|
|
}
|
|
},
|
|
|
|
event: {
|
|
xhr: {
|
|
always: function() {
|
|
// calculate if loading time was below minimum threshold
|
|
},
|
|
done: function(response) {
|
|
var
|
|
context = this,
|
|
elapsedTime = (new Date().getTime() - time),
|
|
loadingDelay = (settings.loadingDuration - elapsedTime >= 0)
|
|
? settings.loadingDuration - elapsedTime
|
|
: 0
|
|
;
|
|
setTimeout(function(){
|
|
module.request.resolveWith(context, [response]);
|
|
}, loadingDelay);
|
|
},
|
|
fail: function(xhr, status, httpMessage) {
|
|
var
|
|
context = this,
|
|
elapsedTime = (new Date().getTime() - time),
|
|
loadingDelay = (settings.loadingDuration - elapsedTime >= 0)
|
|
? settings.loadingDuration - elapsedTime
|
|
: 0
|
|
;
|
|
// page triggers abort on navigation, dont show error
|
|
if(status !== 'abort') {
|
|
setTimeout(function(){
|
|
module.request.rejectWith(context, [xhr, status, httpMessage]);
|
|
}, settings.loadingDelay);
|
|
}
|
|
else {
|
|
module.reset();
|
|
}
|
|
}
|
|
},
|
|
request: {
|
|
complete: function() {
|
|
module.remove.loading();
|
|
$.proxy(settings.complete, $context)();
|
|
},
|
|
done: function(response) {
|
|
module.debug('API request received', response);
|
|
if(settings.dataType == 'json') {
|
|
if( $.isFunction(settings.successTest) ) {
|
|
module.debug('Checking JSON', settings.successTest, response);
|
|
if( settings.success(response) ) {
|
|
$.proxy(settings.success, $context)(response, $module);
|
|
}
|
|
else {
|
|
module.debug('JSON test specified by user and response failed', response);
|
|
$.proxy(settings.failure, $context)(response, $module);
|
|
}
|
|
}
|
|
else {
|
|
$.proxy(settings.success, $context)(response, $module);
|
|
}
|
|
}
|
|
else {
|
|
$.proxy(settings.success, $context)(response, $module);
|
|
}
|
|
},
|
|
error: function(xhr, status, httpMessage) {
|
|
var
|
|
errorMessage = (settings.error[status] !== undefined)
|
|
? settings.error[status]
|
|
: httpMessage,
|
|
response
|
|
;
|
|
// let em know unless request aborted
|
|
if(xhr !== undefined) {
|
|
// readyState 4 = done, anything less is not really sent
|
|
if(xhr.readyState !== undefined && xhr.readyState == 4) {
|
|
|
|
// if http status code returned and json returned error, look for it
|
|
if( xhr.status != 200 && httpMessage !== undefined && httpMessage !== '') {
|
|
module.error(error.statusMessage + httpMessage);
|
|
}
|
|
else {
|
|
if(status == 'error' && settings.dataType == 'json') {
|
|
try {
|
|
response = $.parseJSON(xhr.responseText);
|
|
if(response && response.error !== undefined) {
|
|
errorMessage = response.error;
|
|
}
|
|
}
|
|
catch(er) {
|
|
module.error(error.JSONParse);
|
|
}
|
|
}
|
|
}
|
|
module.remove.loading();
|
|
module.set.error();
|
|
// show error state only for duration specified in settings
|
|
if(settings.errorDuration) {
|
|
setTimeout(module.remove.error, settings.errorDuration);
|
|
}
|
|
module.debug('API Request error:', errorMessage);
|
|
$.proxy(settings.failure, $context)(errorMessage, this);
|
|
}
|
|
else {
|
|
module.debug('Request Aborted (Most likely caused by page change)');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
|
|
create: {
|
|
request: function() {
|
|
return $.Deferred()
|
|
.always(module.event.request.complete)
|
|
.done(module.event.request.done)
|
|
.fail(module.event.request.error)
|
|
;
|
|
},
|
|
xhr: function() {
|
|
$.ajax(ajaxSettings)
|
|
.always(module.event.xhr.always)
|
|
.done(module.event.xhr.done)
|
|
.fail(module.event.xhr.fail)
|
|
;
|
|
}
|
|
},
|
|
|
|
set: {
|
|
error: function() {
|
|
module.verbose('Adding error state to element', $context);
|
|
$context.addClass(className.error);
|
|
},
|
|
loading: function() {
|
|
module.verbose('Adding loading state to element', $context);
|
|
$context.addClass(className.loading);
|
|
}
|
|
},
|
|
|
|
remove: {
|
|
error: function() {
|
|
module.verbose('Removing error state from element', $context);
|
|
$context.removeClass(className.error);
|
|
},
|
|
loading: function() {
|
|
module.verbose('Removing loading state from element', $context);
|
|
$context.removeClass(className.loading);
|
|
}
|
|
},
|
|
|
|
get: {
|
|
request: function() {
|
|
return module.request;
|
|
},
|
|
xhr: function() {
|
|
return module.xhr;
|
|
},
|
|
settings: function() {
|
|
return $.proxy(settings.beforeSend, $module)(settings);
|
|
},
|
|
defaultData: function() {
|
|
var
|
|
data = {}
|
|
;
|
|
if( !$.isWindow(element) ) {
|
|
if( $module.is('input') ) {
|
|
data.value = $module.val();
|
|
}
|
|
else {
|
|
data.text = $module.text();
|
|
}
|
|
}
|
|
return data;
|
|
},
|
|
event: function() {
|
|
if( $.isWindow(element) || settings.on == 'now' ) {
|
|
module.debug('API called without element, no events attached');
|
|
return false;
|
|
}
|
|
else if(settings.on == 'auto') {
|
|
if( $module.is('input') ) {
|
|
return (element.oninput !== undefined)
|
|
? 'input'
|
|
: (element.onpropertychange !== undefined)
|
|
? 'propertychange'
|
|
: 'keyup'
|
|
;
|
|
}
|
|
else {
|
|
return 'click';
|
|
}
|
|
}
|
|
else {
|
|
return settings.on;
|
|
}
|
|
},
|
|
formData: function() {
|
|
var
|
|
formData
|
|
;
|
|
if( $(this).toJSON() === undefined ) {
|
|
module.error(error.missingSerialize);
|
|
return;
|
|
}
|
|
formData = $form.toJSON();
|
|
module.debug('Retrieving form data', formData);
|
|
return $form.toJSON();
|
|
},
|
|
templateURL: function(action) {
|
|
var
|
|
url
|
|
;
|
|
action = action || $module.data(settings.metadata.action) || settings.action || false;
|
|
if(action) {
|
|
module.debug('Looking up url for action', action);
|
|
if(settings.api[action] !== undefined) {
|
|
url = settings.api[action];
|
|
module.debug('Found template url', url);
|
|
}
|
|
else {
|
|
module.error(error.missingAction, settings.action);
|
|
}
|
|
}
|
|
return url;
|
|
}
|
|
},
|
|
|
|
// reset state
|
|
reset: function() {
|
|
module.remove.error();
|
|
module.remove.loading();
|
|
},
|
|
|
|
setting: function(name, value) {
|
|
if( $.isPlainObject(name) ) {
|
|
$.extend(true, settings, name);
|
|
}
|
|
else if(value !== undefined) {
|
|
settings[name] = value;
|
|
}
|
|
else {
|
|
return settings[name];
|
|
}
|
|
},
|
|
internal: function(name, value) {
|
|
if( $.isPlainObject(name) ) {
|
|
$.extend(true, module, name);
|
|
}
|
|
else if(value !== undefined) {
|
|
module[name] = value;
|
|
}
|
|
else {
|
|
return module[name];
|
|
}
|
|
},
|
|
debug: function() {
|
|
if(settings.debug) {
|
|
if(settings.performance) {
|
|
module.performance.log(arguments);
|
|
}
|
|
else {
|
|
module.debug = Function.prototype.bind.call(console.info, console, settings.name + ':');
|
|
module.debug.apply(console, arguments);
|
|
}
|
|
}
|
|
},
|
|
verbose: function() {
|
|
if(settings.verbose && settings.debug) {
|
|
if(settings.performance) {
|
|
module.performance.log(arguments);
|
|
}
|
|
else {
|
|
module.verbose = Function.prototype.bind.call(console.info, console, settings.name + ':');
|
|
module.verbose.apply(console, arguments);
|
|
}
|
|
}
|
|
},
|
|
error: function() {
|
|
module.error = Function.prototype.bind.call(console.error, console, settings.name + ':');
|
|
module.error.apply(console, arguments);
|
|
},
|
|
performance: {
|
|
log: function(message) {
|
|
var
|
|
currentTime,
|
|
executionTime,
|
|
previousTime
|
|
;
|
|
if(settings.performance) {
|
|
currentTime = new Date().getTime();
|
|
previousTime = time || currentTime;
|
|
executionTime = currentTime - previousTime;
|
|
time = currentTime;
|
|
performance.push({
|
|
'Element' : element,
|
|
'Name' : message[0],
|
|
'Arguments' : [].slice.call(message, 1) || '',
|
|
'Execution Time' : executionTime
|
|
});
|
|
}
|
|
clearTimeout(module.performance.timer);
|
|
module.performance.timer = setTimeout(module.performance.display, 100);
|
|
},
|
|
display: function() {
|
|
var
|
|
title = settings.name + ':',
|
|
totalTime = 0
|
|
;
|
|
time = false;
|
|
clearTimeout(module.performance.timer);
|
|
$.each(performance, function(index, data) {
|
|
totalTime += data['Execution Time'];
|
|
});
|
|
title += ' ' + totalTime + 'ms';
|
|
if(moduleSelector) {
|
|
title += ' \'' + moduleSelector + '\'';
|
|
}
|
|
if( (console.group !== undefined || console.table !== undefined) && performance.length > 0) {
|
|
console.groupCollapsed(title);
|
|
if(console.table) {
|
|
console.table(performance);
|
|
}
|
|
else {
|
|
$.each(performance, function(index, data) {
|
|
console.log(data['Name'] + ': ' + data['Execution Time']+'ms');
|
|
});
|
|
}
|
|
console.groupEnd();
|
|
}
|
|
performance = [];
|
|
}
|
|
},
|
|
invoke: function(query, passedArguments, context) {
|
|
var
|
|
maxDepth,
|
|
found,
|
|
response
|
|
;
|
|
passedArguments = passedArguments || queryArguments;
|
|
context = element || context;
|
|
if(typeof query == 'string' && instance !== undefined) {
|
|
query = query.split(/[\. ]/);
|
|
maxDepth = query.length - 1;
|
|
$.each(query, function(depth, value) {
|
|
var camelCaseValue = (depth != maxDepth)
|
|
? value + query[depth + 1].charAt(0).toUpperCase() + query[depth + 1].slice(1)
|
|
: query
|
|
;
|
|
if( $.isPlainObject( instance[value] ) && (depth != maxDepth) ) {
|
|
instance = instance[value];
|
|
}
|
|
else if( $.isPlainObject( instance[camelCaseValue] ) && (depth != maxDepth) ) {
|
|
instance = instance[camelCaseValue];
|
|
}
|
|
else if( instance[value] !== undefined ) {
|
|
found = instance[value];
|
|
return false;
|
|
}
|
|
else if( instance[camelCaseValue] !== undefined ) {
|
|
found = instance[camelCaseValue];
|
|
return false;
|
|
}
|
|
else {
|
|
module.error(error.method, query);
|
|
return false;
|
|
}
|
|
});
|
|
}
|
|
if ( $.isFunction( found ) ) {
|
|
response = found.apply(context, passedArguments);
|
|
}
|
|
else if(found !== undefined) {
|
|
response = found;
|
|
}
|
|
if($.isArray(returnedValue)) {
|
|
returnedValue.push(response);
|
|
}
|
|
else if(returnedValue !== undefined) {
|
|
returnedValue = [returnedValue, response];
|
|
}
|
|
else if(response !== undefined) {
|
|
returnedValue = response;
|
|
}
|
|
return found;
|
|
}
|
|
};
|
|
|
|
if(methodInvoked) {
|
|
if(instance === undefined) {
|
|
module.initialize();
|
|
}
|
|
module.invoke(query);
|
|
}
|
|
else {
|
|
if(instance !== undefined) {
|
|
module.destroy();
|
|
}
|
|
module.initialize();
|
|
}
|
|
})
|
|
;
|
|
|
|
return (returnedValue !== undefined)
|
|
? returnedValue
|
|
: this
|
|
;
|
|
};
|
|
|
|
$.api.settings = {
|
|
|
|
name : 'API',
|
|
namespace : 'api',
|
|
|
|
debug : true,
|
|
verbose : true,
|
|
performance : true,
|
|
|
|
// event binding
|
|
on : 'auto',
|
|
filter : '.disabled, .loading',
|
|
context : false,
|
|
stateContext : false,
|
|
|
|
// templating
|
|
action : false,
|
|
regExpTemplate : /\{\$([A-z]+)\}/g,
|
|
|
|
// data
|
|
url : false,
|
|
urlData : false,
|
|
serializeForm : false,
|
|
|
|
// ui
|
|
defaultData : true,
|
|
throttle : 100,
|
|
allowMultiple : false,
|
|
|
|
// state
|
|
loadingDuration : 1000,
|
|
errorDuration : 2000,
|
|
|
|
// jQ ajax
|
|
method : 'get',
|
|
data : {},
|
|
dataType : 'json',
|
|
cache : true,
|
|
|
|
// callbacks
|
|
beforeSend : function(settings) { return settings; },
|
|
beforeXHR : function(xhr) {},
|
|
|
|
success : function(response) {},
|
|
successText : function(response) { return true; },
|
|
complete : function(response) {},
|
|
failure : function(response) {},
|
|
|
|
// errors
|
|
error : {
|
|
beforeSend : 'The before send function has aborted the request',
|
|
error : 'There was an error with your request',
|
|
exitConditions : 'API Request Aborted. Exit conditions met',
|
|
JSONParse : 'JSON could not be parsed during error handling',
|
|
missingSerialize : 'Serializing a Form requires toJSON to be included',
|
|
missingAction : 'API action used but no url was defined',
|
|
missingParameter : 'Missing an essential URL parameter: ',
|
|
missingURL : 'No URL specified for api event',
|
|
parseError : 'There was an error parsing your request',
|
|
statusMessage : 'Server gave an error: ',
|
|
timeout : 'Your request timed out'
|
|
},
|
|
|
|
className: {
|
|
loading : 'loading',
|
|
error : 'error'
|
|
},
|
|
|
|
selector: {
|
|
form: 'form'
|
|
},
|
|
|
|
metadata: {
|
|
action : 'action',
|
|
request : 'request',
|
|
xhr : 'xhr'
|
|
}
|
|
};
|
|
|
|
|
|
$.api.settings.api = {};
|
|
|
|
|
|
})( jQuery, window , document );
|