app.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312
  1. /**
  2. * jQuery.browser.mobile (http://detectmobilebrowser.com/)
  3. *
  4. * jQuery.browser.mobile will be true if the browser is a mobile device
  5. *
  6. **/
  7. (function(a){(jQuery.browser=jQuery.browser||{}).mobile=/(android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od)|iris|kindle|lge |maemo|midp|mmp|netfront|opera m(ob|in)i|palm( os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series(4|6)0|symbian|treo|up\.(browser|link)|vodafone|wap|windows (ce|phone)|xda|xiino/i.test(a)||/1207|6310|6590|3gso|4thp|50[1-6]i|770s|802s|a wa|abac|ac(er|oo|s\-)|ai(ko|rn)|al(av|ca|co)|amoi|an(ex|ny|yw)|aptu|ar(ch|go)|as(te|us)|attw|au(di|\-m|r |s )|avan|be(ck|ll|nq)|bi(lb|rd)|bl(ac|az)|br(e|v)w|bumb|bw\-(n|u)|c55\/|capi|ccwa|cdm\-|cell|chtm|cldc|cmd\-|co(mp|nd)|craw|da(it|ll|ng)|dbte|dc\-s|devi|dica|dmob|do(c|p)o|ds(12|\-d)|el(49|ai)|em(l2|ul)|er(ic|k0)|esl8|ez([4-7]0|os|wa|ze)|fetc|fly(\-|_)|g1 u|g560|gene|gf\-5|g\-mo|go(\.w|od)|gr(ad|un)|haie|hcit|hd\-(m|p|t)|hei\-|hi(pt|ta)|hp( i|ip)|hs\-c|ht(c(\-| |_|a|g|p|s|t)|tp)|hu(aw|tc)|i\-(20|go|ma)|i230|iac( |\-|\/)|ibro|idea|ig01|ikom|im1k|inno|ipaq|iris|ja(t|v)a|jbro|jemu|jigs|kddi|keji|kgt( |\/)|klon|kpt |kwc\-|kyo(c|k)|le(no|xi)|lg( g|\/(k|l|u)|50|54|\-[a-w])|libw|lynx|m1\-w|m3ga|m50\/|ma(te|ui|xo)|mc(01|21|ca)|m\-cr|me(rc|ri)|mi(o8|oa|ts)|mmef|mo(01|02|bi|de|do|t(\-| |o|v)|zz)|mt(50|p1|v )|mwbp|mywa|n10[0-2]|n20[2-3]|n30(0|2)|n50(0|2|5)|n7(0(0|1)|10)|ne((c|m)\-|on|tf|wf|wg|wt)|nok(6|i)|nzph|o2im|op(ti|wv)|oran|owg1|p800|pan(a|d|t)|pdxg|pg(13|\-([1-8]|c))|phil|pire|pl(ay|uc)|pn\-2|po(ck|rt|se)|prox|psio|pt\-g|qa\-a|qc(07|12|21|32|60|\-[2-7]|i\-)|qtek|r380|r600|raks|rim9|ro(ve|zo)|s55\/|sa(ge|ma|mm|ms|ny|va)|sc(01|h\-|oo|p\-)|sdk\/|se(c(\-|0|1)|47|mc|nd|ri)|sgh\-|shar|sie(\-|m)|sk\-0|sl(45|id)|sm(al|ar|b3|it|t5)|so(ft|ny)|sp(01|h\-|v\-|v )|sy(01|mb)|t2(18|50)|t6(00|10|18)|ta(gt|lk)|tcl\-|tdg\-|tel(i|m)|tim\-|t\-mo|to(pl|sh)|ts(70|m\-|m3|m5)|tx\-9|up(\.b|g1|si)|utst|v400|v750|veri|vi(rg|te)|vk(40|5[0-3]|\-v)|vm40|voda|vulc|vx(52|53|60|61|70|80|81|83|85|98)|w3c(\-| )|webc|whit|wi(g |nc|nw)|wmlb|wonu|x700|yas\-|your|zeto|zte\-/i.test(a.substr(0,4))})(navigator.userAgent||navigator.vendor||window.opera);
  8. /*! jQuery JSON plugin 2.4.0 | code.google.com/p/jquery-json */
  9. (function(jQuery){'use strict';var escape=/["\\\x00-\x1f\x7f-\x9f]/g,meta={'\b':'\\b','\t':'\\t','\n':'\\n','\f':'\\f','\r':'\\r','"':'\\"','\\':'\\\\'},hasOwn=Object.prototype.hasOwnProperty;jQuery.toJSON=typeof JSON==='object'&&JSON.stringify?JSON.stringify:function(o){if(o===null){return'null';}
  10. var pairs,k,name,val,type=jQuery.type(o);if(type==='undefined'){return undefined;}
  11. if(type==='number'||type==='boolean'){return String(o);}
  12. if(type==='string'){return jQuery.quoteString(o);}
  13. if(typeof o.toJSON==='function'){return jQuery.toJSON(o.toJSON());}
  14. if(type==='date'){var month=o.getUTCMonth()+1,day=o.getUTCDate(),year=o.getUTCFullYear(),hours=o.getUTCHours(),minutes=o.getUTCMinutes(),seconds=o.getUTCSeconds(),milli=o.getUTCMilliseconds();if(month<10){month='0'+month;}
  15. if(day<10){day='0'+day;}
  16. if(hours<10){hours='0'+hours;}
  17. if(minutes<10){minutes='0'+minutes;}
  18. if(seconds<10){seconds='0'+seconds;}
  19. if(milli<100){milli='0'+milli;}
  20. if(milli<10){milli='0'+milli;}
  21. return'"'+year+'-'+month+'-'+day+'T'+
  22. hours+':'+minutes+':'+seconds+'.'+milli+'Z"';}
  23. pairs=[];if(jQuery.isArray(o)){for(k=0;k<o.length;k++){pairs.push(jQuery.toJSON(o[k])||'null');}
  24. return'['+pairs.join(',')+']';}
  25. if(typeof o==='object'){for(k in o){if(hasOwn.call(o,k)){type=typeof k;if(type==='number'){name='"'+k+'"';}else if(type==='string'){name=jQuery.quoteString(k);}else{continue;}
  26. type=typeof o[k];if(type!=='function'&&type!=='undefined'){val=jQuery.toJSON(o[k]);pairs.push(name+':'+val);}}}
  27. return'{'+pairs.join(',')+'}';}};jQuery.evalJSON=typeof JSON==='object'&&JSON.parse?JSON.parse:function(str){return eval('('+str+')');};jQuery.secureEvalJSON=typeof JSON==='object'&&JSON.parse?JSON.parse:function(str){var filtered=str.replace(/\\["\\\/bfnrtu]/g,'@').replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,']').replace(/(?:^|:|,)(?:\s*\[)+/g,'');if(/^[\],:{}\s]*jQuery/.test(filtered)){return eval('('+str+')');}
  28. throw new SyntaxError('Error parsing JSON, source is not valid.');};jQuery.quoteString=function(str){if(str.match(escape)){return'"'+str.replace(escape,function(a){var c=meta[a];if(typeof c==='string'){return c;}
  29. c=a.charCodeAt();return'\\u00'+Math.floor(c/16).toString(16)+(c%16).toString(16);})+'"';}
  30. return'"'+str+'"';};}(jQuery));
  31. /**
  32. *
  33. * @author: Malishev Dmitry <dima.malishev@gmail.com>
  34. */
  35. var _DEBUG = true;
  36. var _DEBUG_LEVEL = 'ALL';
  37. // possible levels: ALL, IMPORTANT
  38. var Error = {FATAL: 1, WARNING: 0, NORMAL: -1};
  39. //
  40. // GLOBAL SETTINGS
  41. //
  42. GLOBAL = {};
  43. GLOBAL.FTP_USER_PREFIX = 'admin_';
  44. GLOBAL.DB_USER_PREFIX = 'admin_';
  45. GLOBAL.DB_DBNAME_PREFIX = 'admin_';
  46. GLOBAL.AJAX_URL = '';
  47. /**
  48. * Init debug, grabs console object if accessible, or makes dummy debugger
  49. */
  50. var fb = _DEBUG && 'undefined' != typeof(console) ? console : {
  51. log : function(){},
  52. debug : function(){},
  53. info : function(){},
  54. warn : function(){},
  55. error : function(){},
  56. assert : function(){},
  57. dir : function(){},
  58. dirxml : function(){},
  59. trace : function(){},
  60. group : function(){},
  61. groupEnd : function(){},
  62. time : function(){},
  63. timeEnd : function(){},
  64. profile : function(){},
  65. profileEnd : function(){},
  66. count : function(){},
  67. msg : function(){}
  68. };
  69. //
  70. var App = {
  71. // Main namespases for page specific functions
  72. // Core namespaces
  73. Ajax: { Busy: {} },
  74. Core: {},
  75. // Actions. More widly used funcs
  76. Actions: {
  77. DB: {},
  78. WEB: {}
  79. },
  80. // Utilities
  81. Helpers: {},
  82. HTML: {Build: {}},
  83. Filters: {},
  84. Env: {
  85. lang: GLOBAL.lang,
  86. },
  87. i18n: {},
  88. Listeners: {
  89. DB: {},
  90. WEB: {}
  91. },
  92. View:{
  93. HTML: {
  94. Build: {}
  95. },
  96. // pages related views
  97. },
  98. Cache: {
  99. clear: function(){} // TODO: stub method, will be used later
  100. },
  101. Ref: {},
  102. Tmp: {},
  103. Thread: {
  104. run: function(delay, ref){
  105. setTimeout(function(){
  106. ref();
  107. }, delay*10);
  108. }
  109. },
  110. Settings: { GLOBAL: {}, General: {}},
  111. Templates: {
  112. Templator: null,
  113. Tpl: {},
  114. _indexes: {}
  115. }
  116. };
  117. // Internals
  118. Array.prototype.set = function(key, value){
  119. var index = this[0][key];
  120. this[1][index] = value;
  121. }
  122. Array.prototype.get = function(key){
  123. var index = this[0][key];
  124. return this[1][index];
  125. }
  126. Array.prototype.finalize = function(){
  127. this.shift();
  128. this[0] = this[0].join('');
  129. return this[0];
  130. }
  131. Array.prototype.done = function(){
  132. return this.join('');
  133. }
  134. String.prototype.wrapperize = function(key, ns){
  135. var tpl = App.Templates.get(key, ns);
  136. tpl.set(':content', this);
  137. return tpl.finalize();
  138. }
  139. App.Ajax.request = function(method, data, callback, onError){
  140. // this will prevent multiple ajaxes on user clicks
  141. /*if (App.Helpers.isAjaxBusy(method, data)) {
  142. fb.warn('ajax request ['+method+'] is busy');
  143. return;
  144. }*/
  145. //App.Helpers.setAjaxBusy(method, data);
  146. data = data || {};
  147. jQuery.ajax({
  148. url: GLOBAL.ajax_url,
  149. global: false,
  150. type: data.request_method || "GET",
  151. data: jQuery.extend(data, {'action': method}),
  152. dataType: "text boost",
  153. converters: {
  154. "text boost": function(value) {
  155. value = value.trim();
  156. return jsonParse(value);
  157. }},
  158. async: true,
  159. cache: false,
  160. error: function(jqXHR, textStatus, errorThrown)
  161. {
  162. onError && onError();
  163. if ('undefined' != typeof onError) {
  164. fb.error(textStatus);
  165. }
  166. },
  167. complete: function()
  168. {
  169. //App.Helpers.setAjaxFree(method, data);
  170. },
  171. success: function(reply)
  172. {
  173. //App.Helpers.setAjaxFree(method, data);
  174. try {
  175. callback && callback(reply);
  176. }
  177. catch(e) {
  178. alert('GENERAL ERROR: '+e);
  179. //App.Helpers.generalError();
  180. }
  181. }
  182. });
  183. }
  184. jQuery.extend({
  185. keys: function(obj){
  186. if (!obj) {
  187. return [];
  188. }
  189. var a = [];
  190. jQuery.each(obj, function(k){ a.push(k) });
  191. return a;
  192. }
  193. })
  194. App.Core.create_hidden_form = function(action){
  195. var form = jQuery('<form>', {
  196. id : 'hidden-form',
  197. method : 'post',
  198. action : action
  199. });
  200. jQuery('body').append(form);
  201. return form;
  202. };
  203. App.Core.extend_from_json = function(elm, data, prefix){
  204. elm = jQuery(elm);
  205. var data = App.Core.flatten_json(data, prefix);
  206. var keys = jQuery.keys(data);
  207. for(var i=0, cnt=keys.length; i<cnt; i++)
  208. {
  209. elm.append(jQuery('<input>', {
  210. name : keys[i],
  211. value: data[keys[i]],
  212. type : 'hidden'
  213. }));
  214. }
  215. return elm;
  216. }
  217. App.Core.flatten_json = function(data, prefix){
  218. var keys = jQuery.keys(data);
  219. var result = {};
  220. prefix || (prefix = '');
  221. if(keys.length)
  222. {
  223. for(var i=0, cnt=keys.length; i<cnt; i++)
  224. {
  225. var value = data[keys[i]];
  226. switch(typeof(value))
  227. {
  228. case 'function': break;
  229. case 'object' : result = jQuery.extend(result, App.Core.flatten_json(value, prefix + '[' + keys[i] + ']')); break;
  230. default : result[prefix + '[' + keys[i] + ']'] = value;
  231. }
  232. }
  233. return result;
  234. }
  235. else
  236. {
  237. return false;
  238. }
  239. }
  240. //
  241. // Cookies adapter
  242. // Allow to work old pages realisations of cookie requests
  243. //
  244. function createCookie(name, value, expire_days) {
  245. jQuery.cookie(name, value, { expires: expire_days});
  246. }
  247. function readCookie(name) {
  248. jQuery.cookie(name);
  249. }
  250. function eraseCookie(name) {
  251. jQuery.removeCookie(name);
  252. }
  253. /**
  254. * Timer for profiling
  255. */
  256. var timer = {};
  257. timer.start = function()
  258. {
  259. timer.start_time = new Date();
  260. }
  261. timer.stop = function( msg )
  262. {
  263. timer.stop_time = new Date();
  264. timer.print( msg );
  265. }
  266. timer.print = function( msg )
  267. {
  268. var passed = timer.stop_time - timer.start_time;
  269. fb.info( msg || '' + passed / 1000 );
  270. }
  271. String.prototype.trim = function()
  272. {
  273. var str = this;
  274. str = str.replace(/^\s+/, '');
  275. for (var i = str.length - 1; i >= 0; i--) {
  276. if (/\S/.test(str.charAt(i))) {
  277. str = str.substring(0, i + 1);
  278. break;
  279. }
  280. }
  281. return str;
  282. }