o2.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. /** ***** BEGIN LICENSE BLOCK *****
  2. * |------------------------------------------------------------------------------|
  3. * | O2OA 活力办公 创意无限 o2.js |
  4. * |------------------------------------------------------------------------------|
  5. * | Distributed under the AGPL license: |
  6. * |------------------------------------------------------------------------------|
  7. * | Copyright © 2018, o2oa.net, o2server.io O2 Team |
  8. * | All rights reserved. |
  9. * |------------------------------------------------------------------------------|
  10. *
  11. * This file is part of O2OA.
  12. *
  13. * O2OA is free software: you can redistribute it and/or modify
  14. * it under the terms of the GNU Affero General Public License as published by
  15. * the Free Software Foundation, either version 3 of the License, or
  16. * (at your option) any later version.
  17. *
  18. * O2OA is distributed in the hope that it will be useful,
  19. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  20. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  21. * GNU Affero General Public License for more details.
  22. *
  23. * You should have received a copy of the GNU General Public License
  24. * along with Foobar. If not, see <https://www.gnu.org/licenses/>.
  25. *
  26. * ***** END LICENSE BLOCK ******/
  27. /* load o2 Core
  28. * |------------------------------------------------------------------------------|
  29. * |addReady: o2.addReady(fn), |
  30. * |------------------------------------------------------------------------------|
  31. * |load: o2.load(urls, callback, reload) |
  32. * |loadCss: o2.loadCss(urls, dom, callback, reload, doc) |
  33. * |------------------------------------------------------------------------------|
  34. * |typeOf: o2.typeOf(o) |
  35. * |------------------------------------------------------------------------------|
  36. * |uuid: o2.uuid() |
  37. * |------------------------------------------------------------------------------|
  38. */
  39. (function(){
  40. var _href = window.location.href;
  41. var _debug = (_href.indexOf("debugger")!==-1);
  42. var _par = _href.substr(_href.lastIndexOf("?")+1, _href.length);
  43. var _lp = "zh-cn";
  44. if (_par){
  45. var _parList = _par.split("&");
  46. for (var i=0; i<_parList.length; i++){
  47. var _v = _parList[i];
  48. var _kv = _v.split("=");
  49. if (_kv[0].toLowerCase()==="lg") _lp = _kv[1];
  50. }
  51. }
  52. this.o2 = {
  53. "version": {
  54. "v": '2.0.0',
  55. "build": "2018.11.22",
  56. "info": "O2OA 活力办公 创意无限. Copyright © 2018, o2oa.net O2 Team All rights reserved."
  57. },
  58. "session": {
  59. "isDebugger": _debug,
  60. "path": "/o2_core/o2"
  61. },
  62. "language": _lp,
  63. "splitStr": /(,\s*){1}|(;\s*){1}/g
  64. };
  65. var _attempt = function(){
  66. for (var i = 0, l = arguments.length; i < l; i++){
  67. try {
  68. arguments[i]();
  69. return arguments[i];
  70. } catch (e){}
  71. }
  72. return null;
  73. };
  74. var _typeOf = function(item){
  75. if (item == null) return 'null';
  76. if (item.$family != null) return item.$family();
  77. if (item.constructor == window.Array) return "array";
  78. if (item.nodeName){
  79. if (item.nodeType == 1) return 'element';
  80. if (item.nodeType == 3) return (/\S/).test(item.nodeValue) ? 'textnode' : 'whitespace';
  81. } else if (typeof item.length == 'number'){
  82. if (item.callee) return 'arguments';
  83. }
  84. return typeof item;
  85. };
  86. this.o2.typeOf = _typeOf;
  87. var _addListener = function(dom, type, fn){
  88. if (type == 'unload'){
  89. var old = fn, self = this;
  90. fn = function(){
  91. _removeListener(dom, 'unload', fn);
  92. old();
  93. };
  94. }
  95. if (dom.addEventListener) dom.addEventListener(type, fn, !!arguments[2]);
  96. else dom.attachEvent('on' + type, fn);
  97. };
  98. var _removeListener = function(dom, type, fn){
  99. if (dom.removeEventListener) dom.removeEventListener(type, fn, !!arguments[2]);
  100. else dom.detachEvent('on' + type, fn);
  101. };
  102. //http request class
  103. var _request = (function(){
  104. var XMLHTTP = function(){ return new XMLHttpRequest(); };
  105. var MSXML2 = function(){ return new ActiveXObject('MSXML2.XMLHTTP'); };
  106. var MSXML = function(){ return new ActiveXObject('Microsoft.XMLHTTP'); };
  107. return _attempt(XMLHTTP, MSXML2, MSXML);
  108. })();
  109. var _returnBase = function(number, base) {
  110. return (number).toString(base).toUpperCase();
  111. };
  112. var _getIntegerBits = function(val, start, end){
  113. var base16 = _returnBase(val, 16);
  114. var quadArray = new Array();
  115. var quadString = '';
  116. var i = 0;
  117. for (i = 0; i < base16.length; i++) {
  118. quadArray.push(base16.substring(i, i + 1));
  119. }
  120. for (i = Math.floor(start / 4); i <= Math.floor(end / 4); i++) {
  121. if (!quadArray[i] || quadArray[i] == '')
  122. quadString += '0';
  123. else
  124. quadString += quadArray[i];
  125. }
  126. return quadString;
  127. };
  128. var _rand = function(max) {
  129. return Math.floor(Math.random() * (max + 1));
  130. };
  131. var _uuid = function(){
  132. var dg = new Date(1582, 10, 15, 0, 0, 0, 0);
  133. var dc = new Date();
  134. var t = dc.getTime() - dg.getTime();
  135. var tl = _getIntegerBits(t, 0, 31);
  136. var tm = _getIntegerBits(t, 32, 47);
  137. var thv = _getIntegerBits(t, 48, 59) + '1';
  138. var csar = _getIntegerBits(_rand(4095), 0, 7);
  139. var csl = _getIntegerBits(_rand(4095), 0, 7);
  140. var n = _getIntegerBits(_rand(8191), 0, 7)
  141. + _getIntegerBits(_rand(8191), 8, 15)
  142. + _getIntegerBits(_rand(8191), 0, 7)
  143. + _getIntegerBits(_rand(8191), 8, 15)
  144. + _getIntegerBits(_rand(8191), 0, 15);
  145. return tl + tm + thv + csar + csl + n;
  146. };
  147. this.o2.uuid = _uuid;
  148. var _runCallback = function(callback, key, par){
  149. if (typeOf(callback).toLowerCase() === 'function'){
  150. if (key.toLowerCase()==="success") callback.apply(callback, par);
  151. }else{
  152. if (typeOf(callback).toLowerCase()==='object'){
  153. var name = ("on-"+key).camelCase();
  154. if (callback[name]) callback[name].apply(callback, par);
  155. }
  156. }
  157. };
  158. this.o2.runCallback = _runCallback;
  159. //use framework url
  160. var _frameworks = {
  161. "o2.core": ["/o2_core/o2/o2.core.js"],
  162. "o2.more": ["/o2_core/o2/o2.more.js"],
  163. "ie_adapter": ["/o2_lib/o2/ie_adapter.js"],
  164. "jquery": ["/o2_lib/jquery/jquery.min.js"],
  165. "mootools": ["/o2_lib/mootools/mootools-1.6.0.js"],
  166. "ckeditor": ["/o2_lib/htmleditor/ckeditor/ckeditor.js"],
  167. "raphael": ["/o2_lib/raphael/raphael.js"],
  168. "d3": ["/o2_lib/d3/d3.min.js"],
  169. "ace": ["/o2_lib/ace/src-noconflict/ace.js","/o2_lib/ace/src-noconflict/ext-language_tools.js"],
  170. "JSBeautifier": ["/o2_lib/JSBeautifier/beautify.js"],
  171. "JSBeautifier_css": ["/o2_lib/JSBeautifier/beautify-css.js"],
  172. "JSBeautifier_html": ["/o2_lib/JSBeautifier/beautify-html.js"]
  173. };
  174. var _loaded = {};
  175. var _loadedCss = {};
  176. var _loadSingle = function(url, callback, reload){
  177. var addr_uri = _frameworks[url] || url;
  178. if (!_debug) if (addr_uri.indexOf("o2_lib")===-1) addr_uri = addr_uri.replace(/\.js/, ".min.js");
  179. addr_uri = (addr_uri.indexOf("?")!==-1) ? addr_uri+"&v="+this.o2.version.v : addr_uri+"?v="+this.o2.version.v;
  180. var key = encodeURIComponent(url);
  181. if (!reload) if (_loaded[key]){ if (callback)callback(); return; }
  182. var head = (document.head || document.getElementsByTagName("head")[0] || document.documentElement);
  183. var s = document.createElement('script');
  184. s.src = addr_uri;
  185. head.appendChild(s);
  186. var _checkScriptLoaded = function(_, isAbort){
  187. if (isAbort || !s.readyState || s.readyState === "loaded" || s.readyState === "complete") {
  188. _loaded[key] = true;
  189. _removeListener(s, 'DOMContentLoaded', _checkScriptLoaded);
  190. _removeListener(s, 'readystatechange', _checkScriptLoaded);
  191. if (!isAbort) if (callback)callback();
  192. }
  193. };
  194. if ('onreadystatechange' in s) _addListener(s, 'readystatechange', _checkScriptLoaded);
  195. _addListener(s, 'load', _checkScriptLoaded);
  196. };
  197. var _load = function(urls, callback, reload, doc){
  198. var urltype = _typeOf(urls);
  199. var modules;
  200. if (urltype==="array"){
  201. modules = [];
  202. for (var i=0; i<urls.length; i++){
  203. var url = urls[i];
  204. var module = _frameworks[url] || url;
  205. if (_typeOf(module)==="array"){
  206. modules.concat(module)
  207. }else{
  208. modules.push(url)
  209. }
  210. }
  211. }else{
  212. modules = _frameworks[urls] || urls;
  213. }
  214. var type = _typeOf(modules);
  215. if (type==="array"){
  216. var thisLoaded = [];
  217. for (var i=0; i<modules.length; i++){
  218. _loadSingle(modules[i], function(){
  219. thisLoaded.push(modules[i]);
  220. if (thisLoaded.length===modules.length){
  221. if (callback) callback();
  222. }
  223. }, reload, doc);
  224. }
  225. }
  226. if (type==="string"){
  227. _loadSingle(modules, callback, reload);
  228. }
  229. };
  230. this.o2.load = _load;
  231. var _loadSingleCss = function(url, callback, uuid, reload, sourceDoc){
  232. var key = encodeURIComponent(url);
  233. if (!reload) if (_loadedCss[key]){ if (callback)callback(_loadedCss[key]); return; }
  234. var cssurl = _frameworks[url] || url;
  235. var xhr = new _request();
  236. xhr.open("GET", cssurl, true);
  237. var success = function(xhr){
  238. var cssText = xhr.responseText;
  239. try{
  240. if (cssText){
  241. if (uuid){
  242. var rex = new RegExp("(.+)(?=\\{)", "g");
  243. var match;
  244. while ((match = rex.exec(cssText)) !== null) {
  245. var prefix = "." + uuid + " ";
  246. var rule = prefix + match[0];
  247. cssText = cssText.substring(0, match.index) + rule + cssText.substring(rex.lastIndex, cssText.length)
  248. rex.lastIndex = rex.lastIndex + prefix.length;
  249. }
  250. }
  251. var doc = sourceDoc || document;
  252. var style = doc.createElement("style");
  253. style.setAttribute("type", "text/css");
  254. var head = (document.head || document.getElementsByTagName("head")[0] || document.documentElement);
  255. head.appendChild(style);
  256. if(style.styleSheet){
  257. var setFunc = function(){
  258. style.styleSheet.cssText = cssText;
  259. };
  260. if(style.styleSheet.disabled){
  261. setTimeout(setFunc, 10);
  262. }else{
  263. setFunc();
  264. }
  265. }else{
  266. var cssTextNode = doc.createTextNode(cssText);
  267. style.appendChild(cssTextNode);
  268. }
  269. style.sheet.cssRules[0].cssText = "#layout_xxx {\n" +
  270. " width: 500px;\n" +
  271. " height: 300px;\n" +
  272. " background: #FFCCBA;\n" +
  273. "}"
  274. }
  275. _loadedCss[key] = style;
  276. if (callback) callback(style);
  277. }catch (e){
  278. if (callback) callback();
  279. return;
  280. }
  281. };
  282. var failure = function(xhr){
  283. if (callback) callback();
  284. };
  285. var onreadystatechange= function(){
  286. if (xhr.readyState != 4) return;
  287. var status = xhr.status;
  288. status = (status == 1223) ? 204 : status;
  289. //var response = {text: xhr.responseText || '', xml: xhr.responseXML};
  290. if ((status >= 200 && status < 300))
  291. success(xhr);
  292. else if ((status >= 300 && status < 400))
  293. failure(xhr);
  294. else
  295. failure(xhr);
  296. };
  297. _addListener(xhr, "readystatechange", onreadystatechange);
  298. //xhr.onreadystatechange = onreadystatechange;
  299. xhr.send();
  300. };
  301. var _parseDomString = function(dom, fn, sourceDoc){
  302. var doc = sourceDoc || document;
  303. var list = doc.querySelectorAll(dom);
  304. if (list.length) for (var i=0; i<list.length; i++) _parseDomElement(list[i], fn);
  305. };
  306. var _parseDomElement = function(dom, fn){
  307. if (fn) fn(dom);
  308. };
  309. var _parseDom = function(dom, fn, sourceDoc){
  310. var domType = _typeOf(dom);
  311. if (domType==="string") _parseDomString(dom, fn, sourceDoc);
  312. if (domType==="element") _parseDomElement(dom, fn);
  313. if (domType==="array") for (var i=0; i<dom.length; i++) _parseDom(dom[i], fn, sourceDoc);
  314. };
  315. var _loadCss = function(urls, dom, callback, reload, sourceDoc){
  316. var uuid = "";
  317. if (dom){
  318. uuid = "css"+_uuid();
  319. _parseDom(dom, function(node){ node.className += ((node.className) ? " "+uuid : uuid)}, sourceDoc);
  320. }
  321. modules = _frameworks[urls] || urls;
  322. var type = _typeOf(modules);
  323. if (type==="array"){
  324. var thisLoaded = [];
  325. var styleList = [];
  326. for (var i=0; i<modules.length; i++){
  327. _loadSingleCss(modules[i], function(style){
  328. thisLoaded.push(modules[i]);
  329. if (style) styleList.push(styleList);
  330. if (thisLoaded.length===modules.length){
  331. if (callback) callback(styleList);
  332. }
  333. }, uuid, reload, sourceDoc);
  334. }
  335. }
  336. if (type==="string"){
  337. _loadSingleCss(modules, callback, uuid, reload, sourceDoc);
  338. }
  339. };
  340. this.o2.loadCss = _loadCss;
  341. var _dom = {
  342. ready: false,
  343. loaded: false,
  344. checks: [],
  345. shouldPoll: false,
  346. timer: null,
  347. testElement: document.createElement('div'),
  348. readys: [],
  349. domready: function(){
  350. clearTimeout(_dom.timer);
  351. if (_dom.ready) return;
  352. _dom.loaded = _dom.ready = true;
  353. _removeListener(document, 'DOMContentLoaded', _dom.checkReady);
  354. _removeListener(document, 'readystatechange', _dom.check);
  355. _dom.onReady();
  356. },
  357. check: function(){
  358. for (var i = _dom.checks.length; i--;) if (_dom.checks[i]() && window.MooTools && o2.core && o2.more){
  359. _dom.domready();
  360. return true;
  361. }
  362. return false;
  363. },
  364. poll: function(){
  365. clearTimeout(_dom.timer);
  366. if (!_dom.check()) _dom.timer = setTimeout(_dom.poll, 10);
  367. },
  368. /*<ltIE8>*/
  369. // doScroll technique by Diego Perini http://javascript.nwbox.com/IEContentLoaded/
  370. // testElement.doScroll() throws when the DOM is not ready, only in the top window
  371. doScrollWorks: function(){
  372. try {
  373. _dom.testElement.doScroll();
  374. return true;
  375. } catch (e){}
  376. return false;
  377. },
  378. /*</ltIE8>*/
  379. onReady: function(){
  380. for (var i=0; i<_dom.readys.length; i++){
  381. this.readys[i].apply(window);
  382. }
  383. },
  384. addReady: function(fn){
  385. if (_dom.loaded){
  386. if (fn) fn.apply(window);
  387. }else{
  388. if (fn) _dom.readys.push(fn);
  389. }
  390. return _dom;
  391. },
  392. checkReady: function(){
  393. _dom.checks.push(function(){return true});
  394. _dom.check();
  395. }
  396. };
  397. var _loadO2 = function(){
  398. this.o2.load("o2.core", _dom.check);
  399. this.o2.load("o2.more", _dom.check);
  400. };
  401. _addListener(document, 'DOMContentLoaded', _dom.checkReady);
  402. /*<ltIE8>*/
  403. // If doScroll works already, it can't be used to determine domready
  404. // e.g. in an iframe
  405. if (_dom.testElement.doScroll && !_dom.doScrollWorks()){
  406. _dom.checks.push(_dom.doScrollWorks);
  407. _dom.shouldPoll = true;
  408. }
  409. /*</ltIE8>*/
  410. if (document.readyState) _dom.checks.push(function(){
  411. var state = document.readyState;
  412. return (state == 'loaded' || state == 'complete');
  413. });
  414. if ('onreadystatechange' in document) _addListener(document, 'readystatechange', _dom.check);
  415. else _dom.shouldPoll = true;
  416. if (_dom.shouldPoll) _dom.poll();
  417. if (!window.MooTools) this.o2.load("mootools", function(){ _loadO2(); _dom.check(); });
  418. this.o2.addReady = function(fn){ _dom.addReady.call(_dom, fn); };
  419. })();