2
0
Fork 0
es5-ff-spa-loader/es5-ff-spa-loader.js

816 lines
26 KiB
JavaScript
Raw Normal View History

2015-12-23 00:16:54 +00:00
/*
* Copyright (c) 2015-2016, Willem Cazander
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification, are permitted provided
* that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice, this list of conditions and the
* following disclaimer.
* * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and
* the following disclaimer in the documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT
* OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
* TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
2016-01-17 20:57:59 +00:00
/* jslint browser: true */
/* global angular,define */
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
/**
* FFSpaLoader is an assets loader for single page applications.
* Its build around the concept the there is only a single static index.html which
* synces all its assets to local cache for offline and or fast page loads.
*
* Features:
* - Assets caching for offline use.
* - Assets hashing for fast syncing.
* - Assets types: js,css,cssData
* - Caching backends: null,localStorage,webSqlDB,sqllite
* - Server url question ui.
* - Loader error ui.
* - Build-in Cordova booting.
* - Build-in AngularJS booting.
2016-01-14 21:42:13 +00:00
*
* @module FFSpaLoader
*/
2016-01-17 20:57:59 +00:00
(function (root, factory) {
if ( typeof define === 'function' && define.amd ) {
define('FFSpaLoader', factory(root));
} else if ( typeof exports === 'object' ) {
module.exports = factory(root);
} else {
root.FFSpaLoader = factory(root);
}
})(this || window, /** @lends module:FFSpaLoader */ function (rootWindow) {
'use strict';
2016-01-14 21:42:13 +00:00
var options = {
debug: {
enable: false,
handler: null, // auto filled
prefix: 'FFSpaLoader.'
},
error: {
handler: null, // auto filled
title: 'Loader Error',
style: '.ffError { margin: auto;width: 90%;border: 3px solid red;padding-left: 1em;padding-bottom: 1em;}'
2016-01-14 21:42:13 +00:00
},
boot: {
cordova: {
enable: true,
timeout: 4096, // FIXME: Test really slow devices -> increase ? or add support -1 for disable or ?
flag: 'FFCordovaDevice'
},
angular: {
enable: true,
modules: []
},
2016-01-14 21:42:13 +00:00
},
server: {
url: null,
assets: null,
timeout: 4096,
flag: 'FFServerUrl',
2016-01-14 21:42:13 +00:00
header: {
request: { // TODO: add header support
2016-01-14 21:42:13 +00:00
'X-FFSpaLoader': '42'
},
response: {
}
},
question: {
transport: 'http://',
title: 'Server',
text: 'Please provide the server name;', // TODO: rename .ffAskUrl
style: '.ffAskUrl { font-size: 1em;margin: auto;width: 90%;border: 3px solid #73AD21;padding-left: 1em;padding-bottom: 1em;} .ffAskUrl > div {font-size: 0.8em;color: #ccc;} .ffAskUrl > div > * {} .ffAskUrl > div > input {} .ffAskUrlError{ color: red}',
2016-01-14 21:42:13 +00:00
}
},
cache: {
meta: null,
js: null,
css: null,
cssData: null
}
};
2015-12-23 00:16:54 +00:00
var cacheDB = null; // single instance for websql
2016-01-14 21:42:13 +00:00
var factory = {
detect: {
localStorage: function() {
try {
var testData = 'localStorageDetect';
2016-01-17 20:57:59 +00:00
rootWindow.localStorage.setItem(testData, testData);
rootWindow.localStorage.removeItem(testData);
2016-01-14 21:42:13 +00:00
return true;
} catch(e) {
return false;
}
},
openDatabase: function() {
2016-01-17 20:57:59 +00:00
return 'openDatabase' in rootWindow;
2016-01-14 21:42:13 +00:00
},
sqlitePlugin: function() {
2016-01-17 20:57:59 +00:00
return 'sqlitePlugin' in rootWindow;
2016-01-14 21:42:13 +00:00
},
cordova: function() {
2016-01-17 20:57:59 +00:00
return 'cordova' in rootWindow;
2016-01-14 21:42:13 +00:00
},
cordovaDevice: function() {
2016-01-17 20:57:59 +00:00
return options.boot.cordova.flag in rootWindow;
2016-01-14 21:42:13 +00:00
},
mobileAgent: function() {
2016-01-17 20:57:59 +00:00
return rootWindow.navigator !== undefined && rootWindow.navigator.userAgent.match(/(iPhone|iPod|iPad|Android|BlackBerry|IEMobile)/);
2016-01-14 21:42:13 +00:00
}
},
cache: {
localStorage: function() {
return {
cacheGetValue: function(key, cb) {
try {
2016-01-17 20:57:59 +00:00
var dataRaw = rootWindow.localStorage.getItem(key);
2016-01-14 21:42:13 +00:00
var data = JSON.parse(dataRaw);
cb(null, data);
} catch(e) {
cb(e);
}
},
cacheSetValue: function(key, value, cb) {
try {
2016-01-17 20:57:59 +00:00
rootWindow.localStorage.setItem(key,JSON.stringify(value));
2016-01-14 21:42:13 +00:00
cb(null);
} catch(e) {
cb(e);
}
},
cacheDeleteValue: function(key, cb) {
try {
2016-01-17 20:57:59 +00:00
rootWindow.localStorage.removeItem(key);
2016-01-14 21:42:13 +00:00
cb(null);
} catch(e) {
cb(e);
}
}
2016-01-17 20:57:59 +00:00
};
2016-01-14 21:42:13 +00:00
},
websql: function(opt) {
if (opt === undefined) { opt = {}; }
if (opt.name === undefined) { opt.name = 'ffSpaLoader.db'; }
if (opt.size === undefined) { opt.size = -1; }
if (opt.version === undefined) { opt.version = '1.0'; }
2016-01-17 20:57:59 +00:00
if (opt.openDatabase === undefined) { opt.openDatabase = rootWindow.openDatabase; }
2016-01-14 21:42:13 +00:00
var nullDataHandler = function(cb) {
2016-01-17 20:57:59 +00:00
return function () {
2016-01-14 21:42:13 +00:00
cb(null);
};
};
var sqlErrorHandler = function(cb) {
return function (tx, err) {
cb(err.message+' code: '+err.code);
};
};
var cacheDBInit = function(cb) {
cacheDB.transaction(function(tx) {
2016-01-14 21:42:13 +00:00
var query = 'CREATE TABLE cache_store(id INTEGER PRIMARY KEY AUTOINCREMENT, key TEXT NOT NULL, value TEXT NOT NULL)';
utilDebug('websql.init query '+query);
2016-01-17 20:57:59 +00:00
tx.executeSql(query, [], function(tx) {
2016-01-14 23:36:10 +00:00
var query = 'CREATE UNIQUE INDEX cache_store__key__udx ON cache_store (key)';
utilDebug('websql.init query '+query);
2016-01-17 20:57:59 +00:00
tx.executeSql(query, [], function() {cb(null);}, sqlErrorHandler(cb));
}, sqlErrorHandler(cb));
2016-01-14 21:42:13 +00:00
});
2016-01-17 20:57:59 +00:00
};
2016-01-14 21:42:13 +00:00
return {
cacheOpen: function(cb) {
if (cacheDB !== null) {
cb(null); // open once.
return;
}
2016-01-17 20:57:59 +00:00
cacheDB = rootWindow.openDatabase(opt.name, opt.version, opt.name, opt.size);
cacheDB.transaction(function(tx) {
var query = 'SELECT value FROM cache_store WHERE key = \"test-for-table\"';
utilDebug('websql.cacheOpen query '+query);
2016-01-17 20:57:59 +00:00
tx.executeSql(query, [], function() {
cb(null);
2016-01-17 20:57:59 +00:00
}, function() {
cacheDBInit(cb);
});
});
},
2016-01-14 21:42:13 +00:00
cacheGetValue: function(key, cb) {
cacheDB.transaction(function(tx) {
var query = 'SELECT value FROM cache_store WHERE key = ?';
utilDebug('websql.cacheGetValue query '+query);
2016-01-14 21:42:13 +00:00
tx.executeSql(query,[key], function(tx, res) {
if (res.rows.length === 0) {
cb(null, null);
} else {
var value = res.rows.item(0).value;
cb(null, JSON.parse(value));
}
}, sqlErrorHandler(cb));
});
},
cacheSetValue: function(key, value, cb) {
cacheDB.transaction(function(tx) {
var query = 'SELECT value FROM cache_store WHERE key = ?';
utilDebug('websql.cacheSetValue query '+query);
2016-01-14 21:42:13 +00:00
tx.executeSql(query,[key], function(tx, res) {
if (res.rows.length === 0) {
2016-01-17 20:57:59 +00:00
var queryInsert = 'INSERT INTO cache_store (key,value) VALUES (?,?)';
utilDebug('websql.cacheSetValue query '+queryInsert);
tx.executeSql(queryInsert, [key,JSON.stringify(value)], nullDataHandler(cb), sqlErrorHandler(cb));
2016-01-14 21:42:13 +00:00
} else {
2016-01-17 20:57:59 +00:00
var queryUpdate = 'UPDATE cache_store SET value = ? WHERE key = ?';
utilDebug('websql.cacheSetValue query '+queryUpdate);
tx.executeSql(queryUpdate, [JSON.stringify(value),key], nullDataHandler(cb), sqlErrorHandler(cb));
2016-01-14 21:42:13 +00:00
}
}, sqlErrorHandler(cb));
});
},
cacheDeleteValue: function(key, cb) {
cacheDB.transaction(function(tx) {
var query = 'DELETE FROM cache_store WHERE key = ?';
utilDebug('websql.cacheDeleteValue query '+query);
2016-01-17 20:57:59 +00:00
tx.executeSql(query, [key], function () {
setTimeout(function() {cb(null);}); // return next tick so transaction is flushed before location.reload
}, sqlErrorHandler(cb));
2016-01-14 21:42:13 +00:00
});
}
2016-01-17 20:57:59 +00:00
};
2016-01-14 21:42:13 +00:00
},
}
};
/**
* Prints the debug message with prefix to the options.debug.handler if options.debug.enable is true.
* @param {String} message The message to log.
* @private
*/
var utilDebug = function (message) {
if (options.debug.enable !== true) {
return;
}
options.debug.handler(options.debug.prefix+message);
};
2016-01-14 21:42:13 +00:00
/**
* The default error handler which renders the error in the browser.
* @param {String|Error} err The error message.
* @private
*/
var utilErrorHandler = function(err) {
utilDebug('utilErrorHandler error '+err); // TODO: Add Error object support
2016-01-14 21:42:13 +00:00
var rootTag = document.createElement('div');
rootTag.setAttribute('class','ffError');
var cssTag = document.createElement("style");
cssTag.type = "text/css";
cssTag.innerHTML = options.error.style;
2016-01-14 21:42:13 +00:00
rootTag.appendChild(cssTag);
var titleTag = document.createElement('h1');
titleTag.appendChild(document.createTextNode(options.error.title));
2016-01-14 21:42:13 +00:00
rootTag.appendChild(titleTag);
var questionTag = document.createElement('p');
questionTag.appendChild(document.createTextNode(err));
rootTag.appendChild(questionTag);
document.getElementsByTagName('body')[0].appendChild(rootTag);
};
/**
* Fetches an url resource with a XMLHttpRequest.
* @param {String} url The url to fetch.
* @param {function} cb The callback for error and request if ready.
* @private
*/
2015-12-23 00:16:54 +00:00
var utilHttpFetch = function (url, cb) {
var startTime = new Date().getTime();
var httpRequest = new XMLHttpRequest();
httpRequest.onreadystatechange = function() {
2016-01-17 20:57:59 +00:00
if (httpRequest.readyState === 4 && httpRequest.status === 200) {
utilDebug('utilHttpFetch url \"'+url+'\" done in '+(new Date().getTime()-startTime)+' ms.');
2016-01-14 21:42:13 +00:00
cb(null, httpRequest);
2016-01-17 20:57:59 +00:00
} else if (httpRequest.readyState === 4) {
cb('Wrong status '+httpRequest.status);
2015-12-23 00:16:54 +00:00
}
2016-01-17 20:57:59 +00:00
};
2015-12-23 00:16:54 +00:00
httpRequest.open('GET', url, true);
2016-01-14 21:42:13 +00:00
httpRequest.timeout = options.server.timeout; // ieX: after open()
httpRequest.ontimeout = function() {
cb('timeout after '+options.server.timeout+' of url '+url);
2016-01-14 21:42:13 +00:00
};
2015-12-23 00:16:54 +00:00
httpRequest.send();
};
2016-01-14 21:42:13 +00:00
var cacheGetService = function (type) {
if (options.cache[type]) {
return options.cache[type];
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
return null;
};
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var cacheHasService = function (type) {
return cacheGetService(type) !== null;
};
var cacheCheckType = function (type, cb, action) {
cacheHasService(type)?action():cb('No caching for '+type);
2016-01-14 21:42:13 +00:00
};
var cacheGetValue = function(type, key , cb) {
cacheCheckType(type, cb, function() {
var cacheKey = type+'_'+key;
utilDebug('cacheGetValue key '+cacheKey);
2016-01-14 21:42:13 +00:00
cacheGetService(type).cacheGetValue(cacheKey,cb);
});
};
var cacheSetValue = function(type, key, value, cb) {
cacheCheckType(type, cb, function() {
var cacheKey = type+'_'+key;
utilDebug('cacheSetValue key '+cacheKey);
2016-01-14 21:42:13 +00:00
cacheGetService(type).cacheSetValue(cacheKey,value,cb);
});
};
var cacheDeleteValue = function(type, key, cb) {
cacheCheckType(type, cb, function() {
var cacheKey = type+'_'+key;
utilDebug('cacheDeleteValue key '+cacheKey);
2016-01-14 21:42:13 +00:00
cacheGetService(type).cacheDeleteValue(cacheKey,cb);
});
};
2016-01-17 20:57:59 +00:00
// var cleanupCache = function (resources, cb) {
// utilDebug('cleanupCache TODO cacheList '+0+' resources '+resources.length);
// // TODO: impl for removes in resource lists
// cb(null);
// };
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var injectResourceData = function(resource, data, cb) {
utilDebug('injectResourceData resource '+JSON.stringify(resource)+' data '+data.length);
2016-01-14 21:42:13 +00:00
var tag = null;
if (resource.type === 'css' || resource.type === 'cssData') {
tag = document.createElement('style');
tag.type = 'text/css';
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
if (resource.type === 'js') {
tag = document.createElement('script');
tag.type = 'text/javascript';
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
tag.appendChild(document.createTextNode(data));
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
document.getElementsByTagName('head')[0].appendChild(tag);
//var ref = document.getElementsByTagName('script')[0];
//ref.parentNode.insertBefore(tag, ref); // note in reverse order
cb(null);
};
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var storeResourceKey = function (resource, cb) {
// Add all cache keys in central list so we can clear the cache item if resources are removed.
var cacheKey = resource.type+'_keys';
cacheGetValue('meta',cacheKey,function (err,value) { // TODO: move to resource.type
2016-01-14 21:42:13 +00:00
if (err !== null) {
cb(err);
2015-12-23 00:16:54 +00:00
return;
}
2016-01-14 21:42:13 +00:00
if (value === null) {
value = [];
}
value.push(resource.hash);
cacheSetValue('meta',cacheKey,value, cb);
});
2016-01-17 20:57:59 +00:00
};
2016-01-14 21:42:13 +00:00
var storeResource = function (resource, httpRequest, cb) {
utilDebug('storeResource url '+resource.url+' hash '+resource.hash);
2016-01-14 21:42:13 +00:00
var item = {
resource: resource,
data: httpRequest.responseText
2016-01-17 20:57:59 +00:00
};
2016-01-14 21:42:13 +00:00
cacheSetValue(resource.type, resource.hash, item , function(err) {
if (err !== null) {
cb(err);
2015-12-23 00:16:54 +00:00
} else {
2016-01-14 21:42:13 +00:00
storeResourceKey(resource,cb);
2015-12-23 00:16:54 +00:00
}
});
2016-01-14 21:42:13 +00:00
};
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var loadResource = function (resource, cb) {
var resourceUrl = resource.url;
if (resourceUrl.indexOf('http') === -1) {
resourceUrl = options.server.url + resourceUrl;
}
utilDebug('loadResource '+JSON.stringify(resource));
if (cacheHasService(resource.type)) {
cacheGetValue(resource.type,resource.hash,function(err, value) {
2016-01-17 20:57:59 +00:00
if (err !== null) { return cb(err); }
2016-01-14 21:42:13 +00:00
if (value === null) {
utilDebug('loadResource cache miss'); // + hash mismatch as its the key
2016-01-14 21:42:13 +00:00
} else if (value.resource === undefined) {
utilDebug('loadResource cache wrong obj');
} else {
utilDebug('loadResource cache hit');
injectResourceData(value.resource,value.data,cb);
return;
}
utilHttpFetch(resourceUrl, function(err, httpRequest) {
2016-01-17 20:57:59 +00:00
if (err !== null) { return cb(err); }
storeResource(resource, httpRequest, function (err) {
if (err !== null) { return cb(err); }
injectResourceData(resource,httpRequest.responseText,cb);
});
2016-01-14 21:42:13 +00:00
});
});
} else {
// note: was links but now download + inject so order stays sequenced
utilHttpFetch(resourceUrl, function(err, httpRequest) {
2016-01-17 20:57:59 +00:00
if (err !== null) { return cb(err); }
injectResourceData(resource,httpRequest.responseText,cb);
2016-01-14 21:42:13 +00:00
});
}
};
var loadResources = function(resources, cb) {
2015-12-23 00:16:54 +00:00
var startTime = new Date().getTime();
2016-01-14 21:42:13 +00:00
utilDebug('loadResources');
2015-12-23 00:16:54 +00:00
var resourceStack = resources;
2016-01-14 21:42:13 +00:00
var resourceLoader = function(err) {
2016-01-17 20:57:59 +00:00
if (err !== null) { return cb(err); }
2015-12-23 00:16:54 +00:00
resourceStack = resourceStack.slice(1);
if (resourceStack.length === 0) {
2016-01-14 21:42:13 +00:00
utilDebug('loadResources done in '+(new Date().getTime()-startTime)+' ms.');
cb(null);
2015-12-23 00:16:54 +00:00
} else {
2016-01-14 21:42:13 +00:00
loadResource(resourceStack[0],resourceLoader);
2015-12-23 00:16:54 +00:00
}
};
2016-01-14 21:42:13 +00:00
loadResource(resourceStack[0],resourceLoader);
};
2015-12-23 00:16:54 +00:00
var injectResources = function(resources, cb) {
var startTime = new Date().getTime();
utilDebug('injectResources');
resources.forEach(function (resource) {
2016-01-14 21:42:13 +00:00
cacheGetValue(resource.type,resource.hash,function(err,item) {
// TODO injectResourceData
var tag = null;
if (resource.type === 'css' || resource.type === 'cssData') {
tag = document.createElement('style');
tag.type = 'text/css';
}
if (resource.type === 'js') {
tag = document.createElement('script');
tag.type = 'text/javascript';
}
utilDebug('injectResources from '+JSON.stringify(resource));
2016-01-14 21:42:13 +00:00
tag.appendChild(document.createTextNode(item.data));
//document.getElementsByTagName('head')[0].appendChild(tag);
var ref = document.getElementsByTagName('script')[0];
ref.parentNode.insertBefore(tag, ref); // note in reverse order
});
2015-12-23 00:16:54 +00:00
});
utilDebug('injectResources done in '+(new Date().getTime()-startTime)+' ms.');
2016-01-14 21:42:13 +00:00
cb(null); // FIXME async
};
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
/**
* Starts loader by downloading resource list or using cache version to
* load/refresh/inject the resources.
*
* @param {function} cb Callback gets called when loader is done.
* @private
*/
var startLoader = function (cb) {
if (options.server.url === null) {
cb('Missing options.server.url');
return;
}
if (options.server.assets === null) {
cb('Missing options.server.assets');
2016-01-14 21:42:13 +00:00
return;
}
2016-01-17 20:57:59 +00:00
rootWindow[options.server.flag] = options.server.url;
2016-01-14 21:42:13 +00:00
var resourcesUrl = options.server.url + options.server.assets;
utilDebug('startLoader assets \"'+resourcesUrl+'\"');
2016-01-14 21:42:13 +00:00
utilHttpFetch(resourcesUrl, function(err, httpRequest) {
if (err !== null) {
utilDebug('startLoader fetch error '+err);
2016-01-14 21:42:13 +00:00
if (cacheHasService('meta')) {
cacheGetValue('meta','server_resources',function(err, value) {
if (err !== null) {
cb(err);
} else if (value === null) {
cb('Have no cache of server resouces from '+options.server.url);
2016-01-14 21:42:13 +00:00
} else {
injectResources(value, cb);
}
});
} else {
cb('Could not fetch server resouces from '+options.server.url);
2016-01-14 21:42:13 +00:00
}
return;
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
var resources = [];
JSON.parse(httpRequest.responseText).data.resources.forEach(function (r) {
resources.push(r);
});
utilDebug('startLoader resources '+resources.length);
2016-01-14 21:42:13 +00:00
if (cacheHasService('meta')) {
cacheSetValue('meta','server_resources',resources, function (err) {
2016-01-17 20:57:59 +00:00
if (err !== null) { return cb(err); }
loadResources(resources, cb);
2016-01-14 21:42:13 +00:00
});
2015-12-23 00:16:54 +00:00
} else {
2016-01-14 21:42:13 +00:00
loadResources(resources, cb);
2015-12-23 00:16:54 +00:00
}
});
2016-01-14 21:42:13 +00:00
};
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
/**
* Validates the user input url by downloading it.
*
* @param {function} cb Callback gets called when loader is done.
* @param {Element} deleteTag The dom element to remove from the body tag after success.
* @private
*/
var askUrlValidate = function (cb,deleteTag) {
var inputTag = document.getElementById('serverInput');
var inputErrorTag = document.getElementById('serverInputError');
while (inputErrorTag.firstChild) {
inputErrorTag.removeChild(inputErrorTag.firstChild); // clear error
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
options.server.url = '';
if (options.server.question.transport !== undefined) {
options.server.url += options.server.question.transport;
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
options.server.url += inputTag.value;
var resourcesUrl = options.server.url + options.server.assets;
utilDebug('askUrlStart check assets '+resourcesUrl);
2016-01-14 21:42:13 +00:00
// TODO: add+check headers
2016-01-17 20:57:59 +00:00
utilHttpFetch(resourcesUrl,function(err, httpRequest) {
2016-01-14 21:42:13 +00:00
if (err !== null) {
inputErrorTag.appendChild(document.createTextNode('Error could not get data.'));
return;
}
2016-01-17 20:57:59 +00:00
if (httpRequest.responseText.length === 0) {
inputErrorTag.appendChild(document.createTextNode('Error Got empty data.'));
return;
}
2016-01-14 21:42:13 +00:00
document.getElementsByTagName('body')[0].removeChild(deleteTag);
if (cacheHasService('meta')) {
cacheSetValue('meta','server_url',options.server.url, function(err) {
2016-01-17 20:57:59 +00:00
if (err !== null) { return cb(err); }
startLoader(cb);
2016-01-14 21:42:13 +00:00
});
} else {
startLoader(cb);
}
2016-01-17 20:57:59 +00:00
});
};
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
/**
* Creates the ask url ui.
*
* @param {function} cb Callback gets called when loader is done.
* @private
*/
var askUrl = function (cb) {
utilDebug('askUrl create ui');
var rootTag = document.createElement('div');
rootTag.setAttribute('class','ffAskUrl');
var cssTag = document.createElement("style");
cssTag.type = "text/css";
cssTag.innerHTML = options.server.question.style;
2016-01-14 21:42:13 +00:00
rootTag.appendChild(cssTag);
var titleTag = document.createElement('h1');
titleTag.appendChild(document.createTextNode(options.server.question.title));
rootTag.appendChild(titleTag);
2016-01-14 21:42:13 +00:00
var questionTag = document.createElement('p');
questionTag.appendChild(document.createTextNode(options.server.question.text));
2016-01-14 21:42:13 +00:00
rootTag.appendChild(questionTag);
var formTag = document.createElement('div');
rootTag.appendChild(formTag);
if (options.server.question.transport !== undefined) {
2016-01-14 21:42:13 +00:00
var transportTag = document.createElement('label');
rootTag.setAttribute('for','serverInput');
transportTag.appendChild(document.createTextNode(options.server.question.transport));
2016-01-14 21:42:13 +00:00
formTag.appendChild(transportTag);
2015-12-23 00:16:54 +00:00
}
2016-01-14 21:42:13 +00:00
var inputTag = document.createElement('input');
inputTag.type = 'text';
inputTag.id = 'serverInput';
inputTag.setAttribute('autofocus','');
inputTag.setAttribute('onkeydown','if (event.keyCode == 13) {document.getElementById(\'serverSubmit\').click()}');
formTag.appendChild(inputTag);
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var submitLineTag = document.createElement('br');
formTag.appendChild(submitLineTag);
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var submitTag = document.createElement('input');
submitTag.id = 'serverSubmit';
submitTag.type = 'submit';
submitTag.value = 'Start';
submitTag.onclick = function() {askUrlValidate(cb,rootTag);};
formTag.appendChild(submitTag);
2015-12-23 00:16:54 +00:00
2016-01-14 21:42:13 +00:00
var serverErrorTag = document.createElement('div');
serverErrorTag.id = 'serverInputError';
serverErrorTag.setAttribute('class','ffAskUrlError');
formTag.appendChild(serverErrorTag);
document.getElementsByTagName('body')[0].appendChild(rootTag);
};
var startCacheType = function (type,cb) {
if (options.cache[type] !== null && options.cache[type] === false) {
utilDebug('startCacheType '+type+' disabled');
options.cache[type] = null;
return cb(null);
}
if (options.cache[type] === null) {
if (factory.detect.cordovaDevice() && factory.detect.sqlitePlugin()) {
utilDebug('startCacheType auto sqlitePlugin for '+type);
2016-01-17 20:57:59 +00:00
options.cache[type] = factory.cache.websql({openDatabase: rootWindow.sqlitePlugin});
} else if (factory.detect.openDatabase()) {
utilDebug('startCacheType auto openDatabase for '+type);
options.cache[type] = factory.cache.websql();
} else if (factory.detect.localStorage()) {
utilDebug('startCacheType auto localStorage for '+type);
options.cache[type] = factory.cache.localStorage();
} else {
utilDebug('startCacheType '+type+' none');
}
}
2016-01-17 20:57:59 +00:00
if (options.cache[type] !== null && typeof options.cache[type].cacheOpen === 'function') {
options.cache[type].cacheOpen(cb);
} else {
cb(null);
}
};
var startCache = function (cb) {
startCacheType('meta', function(err) {
if (err !== null) { return cb(err); }
startCacheType('js', function(err) {
if (err !== null) { return cb(err); }
startCacheType('css', function(err) {
if (err !== null) { return cb(err); }
startCacheType('cssData', cb);
});
});
});
};
2016-01-14 21:42:13 +00:00
/**
* Starts the loader.
*
* @param {function} cb Callback gets called when loader is done.
*/
var start = function (cbArgu) {
var startTime = new Date().getTime();
var cb = function(err) {
if (err !== null) {
options.error.handler(err);
2015-12-23 00:16:54 +00:00
} else {
2016-01-14 21:42:13 +00:00
utilDebug('start done in '+(new Date().getTime()-startTime)+' ms.');
bootAngular(function(err) {
if (err !== null) { return options.error.handler(err); }
2016-01-17 20:57:59 +00:00
if (typeof cbArgu === 'function') {
cbArgu();
}
});
2015-12-23 00:16:54 +00:00
}
2016-01-17 20:57:59 +00:00
};
if (options.debug.enable === true) {
var optionsKeys = Object.keys(options);
for (var keyId in optionsKeys) {
var key = optionsKeys[keyId];
utilDebug('start config '+key+' '+JSON.stringify(options[key]));
}
2016-01-14 21:42:13 +00:00
}
bootCordova(function(err) {
if (err !== null) { return cb(err); }
startCache(function(err) {
if (err !== null) { return cb(err); }
if (options.server.url !== null) {
2016-01-14 21:42:13 +00:00
startLoader(cb);
return;
}
if (cacheHasService('meta')) {
cacheGetValue('meta','server_url',function(err, value) {
if (err !== null) {
cb(err);
} else if (value === undefined || value === null || value === '') {
askUrl(cb);
} else {
options.server.url = value;
startLoader(cb);
}
});
} else {
askUrl(cb);
2016-01-14 21:42:13 +00:00
}
});
});
2015-12-23 00:16:54 +00:00
};
var clearServerUrl = function(cb) {
2016-01-14 23:36:10 +00:00
if (cb === undefined) {
cb = function() {};
}
if (cacheHasService('meta')) {
cacheDeleteValue('meta','server_url',cb);
2016-01-14 23:36:10 +00:00
} else {
cb(null);
}
2016-01-17 20:57:59 +00:00
};
2016-01-14 23:36:10 +00:00
var bootAngular = function(cb) {
if (options.boot.angular.enable !== true) {
utilDebug('bootAngular disabled by options');
return cb(null);
2016-01-14 21:42:13 +00:00
}
if (options.boot.angular.modules.length === 0) {
utilDebug('bootAngular disabled by no modules');
return cb(null);
2016-01-14 21:42:13 +00:00
}
utilDebug('bootAngular start '+options.boot.angular.modules);
angular.bootstrap(document, options.boot.angular.modules);
cb(null);
2016-01-17 20:57:59 +00:00
};
2016-01-14 21:42:13 +00:00
/**
* Helper for cordova applications which want to use the sqllite plugin as cache.
* Note: On none cordova page it will callback directly.
* Note: On none cordova device page is will timeout.
*
* @param {function} cb Callback gets called device is ready to use.
* @private
2016-01-14 21:42:13 +00:00
*/
var bootCordova = function(cb) {
if (options.boot.cordova.enable !== true) {
utilDebug('bootCordova disabled by options');
return cb(null);
}
if (factory.detect.cordova() !== true) {
utilDebug('bootCordova disabled by detect');
return cb(null);
}
2016-01-14 21:42:13 +00:00
var startTime = new Date().getTime();
var bootOnce = function() {
var callback = cb;
cb = null;
utilDebug('bootCordova done in '+(new Date().getTime()-startTime)+' ms.');
callback();
2016-01-14 21:42:13 +00:00
};
utilDebug('bootCordova timeout '+options.boot.cordova.timeout);
2016-01-14 21:42:13 +00:00
setTimeout ( function () {
utilDebug('bootCordova timeout');
2016-01-14 21:42:13 +00:00
bootOnce();
}, options.boot.cordova.timeout);
2016-01-14 21:42:13 +00:00
document.addEventListener("deviceready", function () {
2016-01-17 20:57:59 +00:00
rootWindow[options.boot.cordova.flag] = true;
utilDebug('bootCordova '+options.boot.cordova.flag);
2016-01-14 21:42:13 +00:00
bootOnce();
}, false);
2015-12-23 00:16:54 +00:00
};
// Auto fill handlers and return public object.
options.debug.handler = function(msg) {console.log(msg);};
options.error.handler = utilErrorHandler;
2015-12-23 00:16:54 +00:00
return {
2016-01-14 21:42:13 +00:00
options: options,
factory: factory,
start: start,
clearServerUrl: clearServerUrl
2015-12-23 00:16:54 +00:00
};
2016-01-17 20:57:59 +00:00
});