WhutBBCode?

This is a cross-browser BBCode helper. It works with both What.CD and Waffles.fm and other sites . . . CDs and waffles, mmm.

当前为 2014-05-16 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @id WhutBBCode
  3. // @name WhutBBCode?
  4. // @namespace hateradio)))
  5. // @author hateradio
  6. // @version 3.0.3
  7. // @description This is a cross-browser BBCode helper. It works with both What.CD and Waffles.fm and other sites . . . CDs and waffles, mmm.
  8. // @homepage https://www.userscripts.org/scripts/show/89544
  9. // -updateURL https://userscripts.org/scripts/source/89544.meta.js
  10. // @icon data:image/png;base64,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
  11. // @icon64 data:image/png;base64,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
  12. // @screenshot http://i.min.us/iMfJGx70.png
  13.  
  14. // @include http*://*what.cd/*
  15. // @exclude http*://*what.cd/*logchecker*
  16. // @exclude http*://*what.cd/user.php?action=notify*
  17. // @exclude http*://*what.cd/reportsv2.php*
  18.  
  19. // @include http*://*waffles.fm/forum/*
  20. // @include http*://*waffles.fm/details.php*
  21. // @include http*://*waffles.fm/my.php*
  22. // @include http*://*waffles.fm/bbcode.php*
  23. // @include http*://*waffles.fm/forums.php*
  24. // @include http*://*waffles.fm/upload.php*
  25.  
  26. // @include http*://*indietorrents.com/*
  27.  
  28. // @update May 15 2014
  29. // @since Sep 30 2010
  30. // 2010+, hateradio
  31. // Please don't modify or edit my script and re-release it. D:
  32. // Send me a message if you want something modified.
  33. // ==/UserScript==
  34.  
  35. (function () {
  36. 'use strict';
  37.  
  38. // helpers
  39. var dom, ie, strg, update;
  40.  
  41. // S T O R A G E HANDLE
  42. strg = {
  43. on: (function () { try { var a, b = localStorage, c = Math.random().toString(16).substr(2, 8); b.setItem(c, c); a = b.getItem(c); return a === c ? !b.removeItem(c) : false; } catch (e) { return false; } }()),
  44. read: function (key) { return this.on ? JSON.parse(localStorage.getItem(key)) : false; },
  45. save: function (key, dat) { return this.on ? !localStorage.setItem(key, JSON.stringify(dat)) : false; },
  46. wipe: function (key) { return this.on ? !localStorage.removeItem(key) : false; },
  47. zero: function (o) { var k; for (k in o) { if (o.hasOwnProperty(k)) { return false; } } return true; },
  48. grab: function (key, def) { var s = strg.read(key); return strg.zero(s) ? def : s; }
  49. };
  50.  
  51. // U P D A T E HANDLE
  52. update = {
  53. name: 'WhutBBCode?',
  54. version: 3020,
  55. key: 'ujs_WBB_UPDT_HR',
  56. callback: 'wbbupdt',
  57. urij: 'https://dl.dropboxusercontent.com/u/14626536/userscripts/updt/wbb/wbb.json',
  58. uric: 'https://dl.dropboxusercontent.com/u/14626536/userscripts/updt/wbb/wbb.js',
  59. checkchrome: false,
  60. interval: 5,
  61. day: (new Date()).getTime(),
  62. time: function () { return new Date(this.day + (1000 * 60 * 60 * 24 * this.interval)).getTime(); },
  63. top: document.head || document.body,
  64. css: function (t) {
  65. if (!this.style) {
  66. this.style = document.createElement('style');
  67. this.style.type = 'text/css';
  68. this.top.appendChild(this.style);
  69. }
  70. if (ie) {
  71. this.style.cssText += t;
  72. } else {
  73. this.style.appendChild(document.createTextNode(t + '\n'));
  74. }
  75. },
  76. js: function (t) {
  77. var j = document.createElement('script');
  78. j.type = 'text/javascript';
  79. j[/^https?\:\/\//i.test(t) ? 'src' : 'textContent'] = t;
  80. this.top.appendChild(j);
  81. },
  82. notification: function (j) {
  83. if (j) {if (this.version < j.version) { window.localStorage.setItem(this.key, JSON.stringify({date: this.time(), version: j.version, page: j.page })); } else { return true; } }
  84. var a = document.createElement('a'), b = JSON.parse(window.localStorage.getItem(this.key));
  85. a.href = b.page || '#';
  86. a.id = 'userscriptupdater';
  87. a.title = 'Update now.';
  88. a.textContent = 'An update for ' + this.name + ' is available.';
  89. document.body.appendChild(a);
  90. return true;
  91. },
  92. check: function (opt) {
  93. if (typeof (GM_updatingEnabled) === 'boolean' || !strg.on) { return; }
  94. var stored = strg.read(this.key), J, page;
  95. this.csstxt();
  96. if (opt || !stored || stored.date < this.day) {
  97. page = stored && stored.page ? stored.page : '#';
  98. strg.save(this.key, {date: this.time(), version: this.version, page: page});
  99. if (!opt && typeof (GM_xmlhttpRequest) === 'function' && !this.chrome()) {
  100. GM_xmlhttpRequest({method: 'GET', url: update.urij, onload: function (r) { update.notification(JSON.parse(r.responseText)); }, onerror: function () { update.check(1); } });
  101. } else {
  102. J = this.notification.toString().replace('function', 'function ' + this.callback).replace('this.version', this.version).replace(/(?:this\.key)/g, "'" + this.key + "'").replace('this.time()', this.time()).replace('this.name', 'j.name');
  103. this.js(J);
  104. this.js(this.uric);
  105. }
  106. } else if (this.version < stored.version) { this.notification(); }
  107. },
  108. chrome: function () {
  109. if (this.checkchrome === true && typeof (chrome) === 'object') { return true; }
  110. },
  111. csstxt: function () {
  112. if (!this.pop) { this.pop = true; this.css('#userscriptupdater,#userscriptupdater:visited{-moz-box-shadow:0 0 6px #787878;-webkit-box-shadow:0 0 6px #787878;box-shadow:0 0 6px #787878;border:1px solid #777;-moz-border-radius:4px;border-radius:4px;cursor:pointer;color:#555;font-family:Arial, Verdana, sans-serif;font-size:11px;font-weight:700;text-align:justify;min-height:45px;position:fixed;z-index:999999;right:10px;top:10px;width:170px;background:#ebebeb url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACsAAACLCAYAAAD4QWAuAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAA2RpVFh0WE1MOmNvbS5hZG9iZS54bXAAAAAAADw/eHBhY2tldCBiZWdpbj0i77u/IiBpZD0iVzVNME1wQ2VoaUh6cmVTek5UY3prYzlkIj8+IDx4OnhtcG1ldGEgeG1sbnM6eD0iYWRvYmU6bnM6bWV0YS8iIHg6eG1wdGs9IkFkb2JlIFhNUCBDb3JlIDUuMC1jMDYwIDYxLjEzNDc3NywgMjAxMC8wMi8xMi0xNzozMjowMCAgICAgICAgIj4gPHJkZjpSREYgeG1sbnM6cmRmPSJodHRwOi8vd3d3LnczLm9yZy8xOTk5LzAyLzIyLXJkZi1zeW50YXgtbnMjIj4gPHJkZjpEZXNjcmlwdGlvbiByZGY6YWJvdXQ9IiIgeG1sbnM6eG1wTU09Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC9tbS8iIHhtbG5zOnN0UmVmPSJodHRwOi8vbnMuYWRvYmUuY29tL3hhcC8xLjAvc1R5cGUvUmVzb3VyY2VSZWYjIiB4bWxuczp4bXA9Imh0dHA6Ly9ucy5hZG9iZS5jb20veGFwLzEuMC8iIHhtcE1NOk9yaWdpbmFsRG9jdW1lbnRJRD0ieG1wLmRpZDo1NUIzQjc3MTI4N0RFMDExOUM4QzlBNkE2NUU3NDJFNCIgeG1wTU06RG9jdW1lbnRJRD0ieG1wLmRpZDpGN0Q1OEQyNjdEQzUxMUUwQThCNEE3MTU1NDU1NzY2OSIgeG1wTU06SW5zdGFuY2VJRD0ieG1wLmlpZDpGN0Q1OEQyNTdEQzUxMUUwQThCNEE3MTU1NDU1NzY2OSIgeG1wOkNyZWF0b3JUb29sPSJBZG9iZSBQaG90b3Nob3AgQ1M1IFdpbmRvd3MiPiA8eG1wTU06RGVyaXZlZEZyb20gc3RSZWY6aW5zdGFuY2VJRD0ieG1wLmlpZDo1NUIzQjc3MTI4N0RFMDExOUM4QzlBNkE2NUU3NDJFNCIgc3RSZWY6ZG9jdW1lbnRJRD0ieG1wLmRpZDo1NUIzQjc3MTI4N0RFMDExOUM4QzlBNkE2NUU3NDJFNCIvPiA8L3JkZjpEZXNjcmlwdGlvbj4gPC9yZGY6UkRGPiA8L3g6eG1wbWV0YT4gPD94cGFja2V0IGVuZD0iciI/Po6YcvQAAAQFSURBVHja7JzBSxRRHMdnp+gkiLdOgtshKGSljQVF8CK0biEErYfwFmT+BQpdA0MIBEFtTx2qSxESaAt5ioUQFDp5sjl06rbnumzfp7+VbZx5M+/Nb9wZ+f3g56wzO28//ua93/u9J/stdDodx2/P3o85llaFT8JvwlvwTfhf00a2Hv8IPO86PHYHvg//An8OfwRfg/9RfzvTZ7DBvoZXQq6p6D7MCuwT+N2I92zAB/sNO0yPO8quwxf7DasABmK+d0XTVVKHnYIvG96z1i9Ymw8ep/R2obAqNdkm41e2sFct71v1/f4BiXyOJpRpHKZ918s9527B5+FvLwJWDaoR3zmvZ/bZw2HPNyMeBOTeb/BfaXaDEuVMvx2G3QDQMkW21wZsUpkp7GbIeU9zz3TI+WXTVGYCW6XRbApb1lxbTwt2VVMltS1hVWRnuWFVqhoNudbW9NchHIpc+ToO7GDE49JFtRij/ZG4gy0O7CIVIjZWNuhiw0lhK1SA6GzI8ppxKouCjTNaOWC7qWzKFrYaNw/SQOKwNVtYk4KjyAQ7RpnHCHaeCg7ugZQon7sBj3RYM62mHdmTVAaGxbiRNVmqRM3/bUvgDQCX/CcLvZsceEOF1v82dgPTrkdVVp2iXU8Q4e9ob0IHu59gUecxdwdlMwBunusGAJ1NuPr0KLoFdYQ3GGBXAiMLWC9gBRDX2gTa9g3Wp7Rbk8TqaPfjWWRp9I0kaLARVCbiXMO/xLGwdfCd7Oa4eDGQdD0fYYcJ7z/bzXHpxbWEDRaddO1FF3aSobE6pazAawztX0H7465mXWVqB2hwqWdwFeFfGaM+Wlh4V/rkMO2fpmy3VWTf5AD0NzLLkYsfn53T7fUs21k2UPmw5jBs9qZgx/AH4Ns+VxvQwJg0rGXTMPUfnhYgj0MLmayb6+TIBFZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBVZgBTZzVrg3U+Nsz1iTo7m7c+GRFU2ONGBFkyMNWNHkSANWNDl0xqbJAZ+j1/nR5HBOv6zm/8JaPjQ5KKqiyRFVpORfk8PRf3NZq8lRrd3PhiaHc6pvcLk0ORDdfGlyAFg0OdKAPUlliG76mhyGUNaDLXOaHIjuJdXkoKVKXzU5wlJZZjU5AFyKKhErFkuVbjcoUo3Apcmhnu6Ebkcmc5oczd2dZlA3YNHkUAFwUtLkcJlWnm1a1ng94AvkbKnM1SxVTKwRMphYNDkAPNiFFU0OZuPV5NDMYiyaHOgKvJoc8CVftFk1ORRsi/FxvYR3yH9qZjYba+VGkwOTw5GCzZcmByzTmhyI6ra/kNkiz4wmByD/0+T4J8AAyDkZArebBxMAAAAASUVORK5CYII=) no-repeat 13px 15px;padding:12px 20px 10px 65px}#userscriptupdater:hover,#userscriptupdater:visited:hover{color:#55698c!important;background-position:13px -85px;border-color:#8f8d96}'); }
  113. }
  114. };
  115. update.check();
  116.  
  117. ie = !document.body.addEventListener && document.selection;
  118.  
  119. // M I S C HANDLE
  120. dom = {
  121. // a simple list iterator function for arrays, nodelists, etc
  122. aEach: function (list, cb, scope) { var i, j = list.length; for (i = 0; i < j; i++) { if (cb.call(scope, list[i], i, list) === false) { break; } } },
  123. // a simple object-type iterator | todo reverse cb order
  124. oEach: function (object, cb, scope) { var key; for (key in object) { if (object.hasOwnProperty(key)) { if (cb.call(scope, key, object[key], object) === false) { break; } } } },
  125. dom: function (name, attr, child, parent) {
  126. // dom element creator
  127. // attr is an object of attributes to apply
  128. // child to attach to this element
  129. // parent for this element
  130. var e = document.createElement(name);
  131. if (attr.txt) {
  132. e.appendChild(document.createTextNode(attr.txt));
  133. delete attr.txt;
  134. }
  135. dom.oEach(attr, function (key, data) {
  136. if (typeof data === 'object') {
  137. dom.oEach(data, function (name, value) {
  138. if (key === 'attr') {
  139. e.setAttribute(name, value);
  140. } else {
  141. e[key][name] = value;
  142. }
  143. });
  144. } else {
  145. e[key] = data;
  146. }
  147. });
  148. if (child) { e.appendChild(child); }
  149. if (parent) { parent.appendChild(e); }
  150. return e;
  151. },
  152. click: (function () {
  153. var e;
  154.  
  155. if (ie) {
  156. e = document.createEventObject();
  157. return function (el) { return el && el.fireEvent('onclick', e); };
  158. }
  159.  
  160. return function (el) {
  161. e = document.createEvent('MouseEvents');
  162. e.initMouseEvent('click', true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
  163. return el && !el.dispatchEvent(e);
  164. };
  165.  
  166. }()),
  167. evt: function (el, ev, cb, cap) {
  168. if (!ie) {
  169. return el.addEventListener(ev, cb, !!cap);
  170. }
  171. return el.attachEvent('on' + ev, function (e) {
  172. e.currentTarget = e.target = e.srcElement;
  173. e.preventDefault = function () { e.returnValue = false; };
  174. e.stopPropagation = function () { e.cancelBubble = true; };
  175. cb.call(el, e);
  176. e = null;
  177. });
  178. }
  179. };
  180.  
  181. /**
  182. * WhutBB Class
  183. * The principal class should not be used directly,
  184. * use WhutBB.create() instead
  185. *
  186. * Uses a textarea as a reference to attach elements and events
  187. *
  188. * @param textarea to use
  189. * @param id to place on the textarea
  190. */
  191. function WhutBB(textarea, id) {
  192. this.textarea = textarea;
  193. this.textarea.className += ' wbbarea';
  194. this.textarea.setAttribute('data-wbb', id);
  195. this.id = id;
  196. this.wrap = dom.dom('div', { className: 'wbbcode ' + WhutBB.$.data.getWrapClass() });
  197.  
  198. WhutBB.Panel.copyTo(this);
  199. this.buttonList = this.makeButtonList();
  200.  
  201. this.insert(WhutBB.config.beneath);
  202. this.events();
  203. }
  204.  
  205. window.WhutBB = WhutBB;
  206.  
  207. WhutBB.set = {};
  208.  
  209. /**
  210. * The factory gets all textareas on the page and creates new WhutBB instances
  211. * for textareas that are not read-only or disabled
  212. */
  213. WhutBB.factory = function () {
  214. dom.aEach(document.getElementsByTagName('textarea'), function (textarea) {
  215. if (!textarea.disabled && !textarea.readOnly) {
  216. WhutBB.create(textarea);
  217. }
  218. });
  219. };
  220.  
  221. /**
  222. * Creates a WhutBBCode? instance for a textarea
  223. * Ignores textareas that contain the class noWhutBB
  224. *
  225. * Stores reference in WhutBB.set
  226. *
  227. * @param textarea to use
  228. * @param force forces the creation of a new instance
  229. */
  230. WhutBB.create = function (textarea, force) {
  231. if (!WhutBB.$.data.ignore.test(textarea.getAttribute('class'))) {
  232. var id = WhutBB.id(textarea);
  233. if (!WhutBB.set[id] || force === true) {
  234. WhutBB.set[id] = new WhutBB(textarea, id);
  235. }
  236. return WhutBB.set[id];
  237. }
  238. };
  239.  
  240. /**
  241. * Locates or returns a unique ID
  242. * @param textarea to use
  243. */
  244. WhutBB.id = function (textarea) {
  245. var dat = textarea.getAttribute('data-wbb');
  246. if (dat && dat.length > 0) {
  247. return dat;
  248. }
  249. return Math.random().toString(32);
  250. };
  251.  
  252. /**
  253. * Inserts the buttons beneath or above a textarea
  254. */
  255. WhutBB.prototype.insert = function (beneath) {
  256. var node = beneath ? this.textarea.nextSibling : this.textarea;
  257. this.textarea.parentNode.insertBefore(this.wrap, node);
  258. };
  259.  
  260. // WhutBB.prototype.update = function (textarea) {
  261. // // update the textarea
  262. // this.textarea = textarea;
  263. // // update the wrap
  264. // this.insert(WhutBB.config.beneath);
  265. // // update the events if the previous elements are different
  266. // this.events();
  267. // };
  268.  
  269. /**
  270. * Attaches event handlers
  271. */
  272. WhutBB.prototype.events = function () {
  273. var type = (typeof document.documentElement.style.MozAppearance === 'string') ? 'keypress' : 'keydown';
  274. dom.evt(this.textarea, type, WhutBB.evt.key.register(this));
  275. dom.evt(this.wrap, 'click', WhutBB.evt.mouse.register(this));
  276. };
  277.  
  278. /**
  279. * Hides this instance's elements
  280. */
  281. WhutBB.prototype.hide = function () {
  282. this.wrap.className += ' wbbhide';
  283. };
  284.  
  285. /**
  286. * Shows this instance's elements
  287. */
  288. WhutBB.prototype.show = function () {
  289. this.wrap.className = this.wrap.className.replace(/(?: wbbhide)/g, '');
  290. };
  291.  
  292. /**
  293. * Returns a button (if any)
  294. * @param name of the button to get
  295. */
  296. WhutBB.prototype.getButton = function (name) {
  297. return this.buttonList[name];
  298. };
  299.  
  300. /**
  301. * Builds a list of DOM buttons for referencing
  302. * Used for keyboard shortcuts
  303. */
  304. WhutBB.prototype.makeButtonList = function () {
  305. var list = {};
  306. dom.aEach(this.panels.button.getElementsByTagName('button'), function (el) {
  307. list[el.name] = el;
  308. });
  309. return list;
  310. };
  311.  
  312. // WhutBB.$ is a collection of misc functions and storage
  313. WhutBB.$ = {
  314. data: {
  315. ignore: /(?:\b(?:noWhutBB)\b)/i, // Ignore textareas with a CSS class of "noWhutBB"
  316. web: [
  317. [':test', /^$|^localhost$/],
  318. ['what', /(?:what)\.cd/],
  319. ['waffles', /(?:waffles\.fm)/],
  320. ['indietorrents', /(?:indietorrents\.com)/],
  321. ['passthepopcorn', /(?:passthepopcorn\.me)/]
  322. // /(?:(last)(?:fm)?\.fm)/,
  323. ],
  324. wrapClasses: ['wbbimgless', 'wbbimg'], // Displays text or icons on buttons
  325. getWrapClass: function () {
  326. return this.wrapClasses[Number(WhutBB.user.settings.icon)];
  327. },
  328. glyph: { // http://twitter.github.io/bootstrap/2.3.2/assets/img/glyphicons-halflings.png http://twitter.github.io/bootstrap/assets/img/glyphicons-halflings.png
  329. black: 'data:image/png;base64,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'
  330. },
  331. css: function (o) { // bootstrap icons first
  332. o = '[class^=icon-],[class*=" icon-"]{display:inline-block;width:14px;height:14px;margin-right:.3em;line-height:14px;vertical-align:text-top;background-image:url(' + this.glyph.black + ');background-position:14px 14px;background-repeat:no-repeat;margin-top:1px}.icon-white{background-image:url("http://twitter.github.io/bootstrap/assets/img/glyphicons-halflings.png")}.icon-glass{background-position:0 0}.icon-music{background-position:-24px 0}.icon-search{background-position:-48px 0}.icon-envelope{background-position:-72px 0}.icon-heart{background-position:-96px 0}.icon-star{background-position:-120px 0}.icon-star-empty{background-position:-144px 0}.icon-user{background-position:-168px 0}.icon-film{background-position:-192px 0}.icon-th-large{background-position:-216px 0}.icon-th{background-position:-240px 0}.icon-th-list{background-position:-264px 0}.icon-ok{background-position:-288px 0}.icon-remove{background-position:-312px 0}.icon-zoom-in{background-position:-336px 0}.icon-zoom-out{background-position:-360px 0}.icon-off{background-position:-384px 0}.icon-signal{background-position:-408px 0}.icon-cog{background-position:-432px 0}.icon-trash{background-position:-456px 0}.icon-home{background-position:0 -24px}.icon-file{background-position:-24px -24px}.icon-time{background-position:-48px -24px}.icon-road{background-position:-72px -24px}.icon-download-alt{background-position:-96px -24px}.icon-download{background-position:-120px -24px}.icon-upload{background-position:-144px -24px}.icon-inbox{background-position:-168px -24px}.icon-play-circle{background-position:-192px -24px}.icon-repeat{background-position:-216px -24px}.icon-refresh{background-position:-240px -24px}.icon-list-alt{background-position:-264px -24px}.icon-lock{background-position:-287px -24px}.icon-flag{background-position:-312px -24px}.icon-headphones{background-position:-336px -24px}.icon-volume-off{background-position:-360px -24px}.icon-volume-down{background-position:-384px -24px}.icon-volume-up{background-position:-408px -24px}.icon-qrcode{background-position:-432px -24px}.icon-barcode{background-position:-456px -24px}.icon-tag{background-position:0 -48px}.icon-tags{background-position:-25px -48px}.icon-book{background-position:-48px -48px}.icon-bookmark{background-position:-72px -48px}.icon-print{background-position:-96px -48px}.icon-camera{background-position:-120px -48px}.icon-font{background-position:-144px -48px}.icon-bold{background-position:-167px -48px}.icon-italic{background-position:-192px -48px}.icon-text-height{background-position:-216px -48px}.icon-text-width{background-position:-240px -48px}.icon-align-left{background-position:-264px -48px}.icon-align-center{background-position:-288px -48px}.icon-align-right{background-position:-312px -48px}.icon-align-justify{background-position:-336px -48px}.icon-list{background-position:-360px -48px}.icon-indent-left{background-position:-384px -48px}.icon-indent-right{background-position:-408px -48px}.icon-facetime-video{background-position:-432px -48px}.icon-picture{background-position:-456px -48px}.icon-pencil{background-position:0 -72px}.icon-map-marker{background-position:-24px -72px}.icon-adjust{background-position:-48px -72px}.icon-tint{background-position:-72px -72px}.icon-edit{background-position:-96px -72px}.icon-share{background-position:-120px -72px}.icon-check{background-position:-144px -72px}.icon-move{background-position:-168px -72px}.icon-step-backward{background-position:-192px -72px}.icon-fast-backward{background-position:-216px -72px}.icon-backward{background-position:-240px -72px}.icon-play{background-position:-264px -72px}.icon-pause{background-position:-288px -72px}.icon-stop{background-position:-312px -72px}.icon-forward{background-position:-336px -72px}.icon-fast-forward{background-position:-360px -72px}.icon-step-forward{background-position:-384px -72px}.icon-eject{background-position:-408px -72px}.icon-chevron-left{background-position:-432px -72px}.icon-chevron-right{background-position:-456px -72px}.icon-plus-sign{background-position:0 -96px}.icon-minus-sign{background-position:-24px -96px}.icon-remove-sign{background-position:-48px -96px}.icon-ok-sign{background-position:-72px -96px}.icon-question-sign{background-position:-96px -96px}.icon-info-sign{background-position:-120px -96px}.icon-screenshot{background-position:-144px -96px}.icon-remove-circle{background-position:-168px -96px}.icon-ok-circle{background-position:-192px -96px}.icon-ban-circle{background-position:-216px -96px}.icon-arrow-left{background-position:-240px -96px}.icon-arrow-right{background-position:-264px -96px}.icon-arrow-up{background-position:-289px -96px}.icon-arrow-down{background-position:-312px -96px}.icon-share-alt{background-position:-336px -96px}.icon-resize-full{background-position:-360px -96px}.icon-resize-small{background-position:-384px -96px}.icon-plus{background-position:-408px -96px}.icon-minus{background-position:-433px -96px}.icon-asterisk{background-position:-456px -96px}.icon-exclamation-sign{background-position:0 -120px}.icon-gift{background-position:-24px -120px}.icon-leaf{background-position:-48px -120px}.icon-fire{background-position:-72px -120px}.icon-eye-open{background-position:-96px -120px}.icon-eye-close{background-position:-120px -120px}.icon-warning-sign{background-position:-144px -120px}.icon-plane{background-position:-168px -120px}.icon-calendar{background-position:-192px -120px}.icon-random{background-position:-216px -120px;width:16px}.icon-comment{background-position:-240px -120px}.icon-magnet{background-position:-264px -120px}.icon-chevron-up{background-position:-288px -120px}.icon-chevron-down{background-position:-313px -119px}.icon-retweet{background-position:-336px -120px}.icon-shopping-cart{background-position:-360px -120px}.icon-folder-close{background-position:-384px -120px}.icon-folder-open{background-position:-408px -120px;width:16px}.icon-resize-vertical{background-position:-432px -119px}.icon-resize-horizontal{background-position:-456px -118px}.icon-hdd{background-position:0 -144px}.icon-bullhorn{background-position:-24px -144px}.icon-bell{background-position:-48px -144px}.icon-certificate{background-position:-72px -144px}.icon-thumbs-up{background-position:-96px -144px}.icon-thumbs-down{background-position:-120px -144px}.icon-hand-right{background-position:-144px -144px}.icon-hand-left{background-position:-168px -144px}.icon-hand-up{background-position:-192px -144px}.icon-hand-down{background-position:-216px -144px}.icon-circle-arrow-right{background-position:-240px -144px}.icon-circle-arrow-left{background-position:-264px -144px}.icon-circle-arrow-up{background-position:-288px -144px}.icon-circle-arrow-down{background-position:-312px -144px}.icon-globe{background-position:-336px -144px}.icon-wrench{background-position:-360px -144px}.icon-tasks{background-position:-384px -144px}.icon-filter{background-position:-408px -144px}.icon-briefcase{background-position:-432px -144px}.icon-fullscreen{background-position:-456px -144px}';
  333. return o + ' .wbbcode button::-moz-focus-inner{border:0;padding:0}.wbbcode div,.wbbcode ul{margin:.2em;padding:.1em}.wbbset li{display:inline;margin:2px}.wbbset label input{vertical-align:text-bottom}.wbbset li label input{margin:0 3px 0 0}.wbbcode{width:' + WhutBB.config.width + 'px;text-align:center;font-size:11px;font-family:Tahoma, sans-serif;margin:auto;padding:3px}.sidebar .wbbcode {width: 100%;}.wbbcode.wbb_noimg button{background-image:none}.wbbcode.wbbimg button span{text-indent:-100px;overflow:hidden;margin:0}.wbbcode.wbbimgless button span{margin:0;background:none}.wbbcode button.whutbbutton{float:none;overflow:hidden;background:#eee;color:#555;font-size:11px;font-family:Arial, sans-serif;font-weight:400;cursor:pointer;width:22px;height:21px;text-shadow:#fff 1px 1px 1px;border-top:1px solid #fff;border-left:1px solid #fff;border-right:1px solid #ccc;border-bottom:1px solid #ccc;-moz-border-radius:2px;border-radius:2px;-moz-transition-duration:.2s;-webkit-transition-duration:.2s;-o-transition:none;transition-duration:.2s;vertical-align:middle;margin:0 1px 3px;padding:1px}.wbbcode button:hover{background-color:#fff;color:#555;border-top:1px solid #eee;border-left:1px solid #eee;border-right:1px solid #bbb;border-bottom:1px solid #bbb}.wbbcode button:active span{margin:3px 0 0 1px}.wbblink{padding:2px 0}.wbbemot,.wbbset{overflow:auto;margin:auto}.wbbemot{max-height:150px;box-shadow:0 0 3px #777;padding:3px}.wbbemot img,.wbbemot div{cursor:pointer}div.wbbcode button.wbbpressed{background-color:#ddd;border-top:1px solid #aaa;border-left:1px solid #aaa;border-right:1px solid #eee;border-bottom:1px solid #eee}.wbbcon{color:#d06620;display:block;padding:3px 0}textarea[id^=editbox]{max-height:400px;overflow:auto!important}.wbbarea{outline-color:#ADD8E6;max-height:500px!important;overflow:auto!important;display:block;margin:3px auto 6px}.wbbshortcut{overflow:hidden;text-align:center;color:#2F2F2F;margin:0;padding:0}.wbbshortcut li{background:#eee;border-top:1px solid #fff;border-left:1px solid #fff;border-right:1px solid #ccc;border-bottom:1px solid #ccc;border-radius:2px;display:inline-block;zoom:1;height:50px;vertical-align:top;width:58px;margin:3px;padding:2px 3px}* html .wbbshortcut li{display:inline}.wbbhide,.hidden.wbbarea{display:none !important}';
  334. }
  335. },
  336. findSite: function () {
  337. var website = ':generic';
  338. dom.aEach(this.data.web, function (site) {
  339. if (site[1].test(document.domain)) {
  340. website = site[0];
  341. return false;
  342. }
  343. });
  344. return website;
  345. }
  346. };
  347.  
  348. /**
  349. * The WhutBBCode? initializer
  350. *
  351. * @param config, see WhutBB.Settings
  352. *
  353. * example:
  354. *
  355. * WhutBB.init({
  356. * emoticonDir: 'https://ssl.what.cd/static/common/smileys/',
  357. * emoticons: WhutBB.db.emoticons.gz.slice(0, 4),
  358. * blueprint: [
  359. * ['b', 'i', 'u', 's'], ['code'],
  360. * ['color', 'size'], ['*'],
  361. * ['url', 'img'], ['quote'],
  362. * ['erase'], ['emoticon', 'shortcut', 'settings']
  363. * ]
  364. * });
  365. *
  366. */
  367. WhutBB.init = function (config) {
  368. WhutBB.config = new WhutBB.Settings(config || WhutBB.db.getSiteSettings(WhutBB.$.findSite()));
  369. // try {
  370. // console.info('WhutBBCode? mode ' + WhutBB.config.name);
  371. // console.log(WhutBB.config);
  372. // } catch (e) {}
  373. WhutBB.user.load();
  374. update.css(WhutBB.$.data.css());
  375. WhutBB.Panel.construct();
  376. if (document.getElementById('content')) {
  377. dom.evt(document.getElementById('content'), 'click', WhutBB.evt.delegate.edit);
  378. if (document.getElementById('messageform')) {
  379. dom.evt(document.getElementById('messageform'), 'click', WhutBB.evt.delegate.inbox);
  380. }
  381. }
  382. };
  383.  
  384. /**
  385. * Settings storage management
  386. * Uses localStorage to store a user's settings
  387. *
  388. * Sends an appropriate message when settings are saved or not
  389. *
  390. * All settings are stored in the options object. These are
  391. * also used in the Panel class, which generates check boxes per option.
  392. */
  393. WhutBB.user = {
  394. message: [
  395. 'Settings failed to save. D:',
  396. 'Settings saved. :D'
  397. ],
  398. options: {
  399. prompt: {
  400. txt: 'Prompts',
  401. title: 'Show browser prompts.',
  402. value: false
  403. },
  404. icon: {
  405. txt: 'Icons',
  406. title: 'Show icons.',
  407. value: false
  408. },
  409. link: {
  410. txt: 'WhutBBCode? Link',
  411. title: 'Show WhutBBCode? link',
  412. value: true
  413. }
  414. },
  415. load: function () {
  416. this.set(strg.grab('wbb3', this.defaults()));
  417. // console.log('load', this.settings);
  418. },
  419. set: function (settings) {
  420. this.settings = this.validate(settings);
  421. },
  422. save: function (settings) {
  423. WhutBB.Panel.message(this.message[Number(strg.save('wbb3', this.validate(settings)))]);
  424. return strg.on ? this.load() : this.set(settings);
  425. },
  426. validate: function (settings) { // returns only valid settings that exist in options
  427. var valid = {};
  428. dom.oEach(this.options, function (name) {
  429. valid[name] = !!settings[name];
  430. });
  431. return valid;
  432. },
  433. defaults: function () {
  434. var defaults = {};
  435. dom.oEach(this.options, function (name, options) {
  436. defaults[name] = options.value;
  437. });
  438. return defaults;
  439. },
  440. settings: {}
  441. };
  442.  
  443. /**
  444. * Psuedo-Database
  445. * Contains all sites, buttons, emoticons, shortcuts
  446. *
  447. * Shortcuts are sorted by modifier key (ctrl/alt/ctrl+alt)
  448. * Modifier properties (a single letter) should correspond to a keyboard key letter
  449. * and the value (text) to a button name (WhutBB.db.button[text])
  450. * Don't use CTRL with W, T, N, O (Chromium/IE Bugs)
  451. *
  452. * Special Note: Meta key (such as that on a Mac) is aliased to CTRL,
  453. * pressing either key returns the same result
  454. *
  455. */
  456. WhutBB.db = {
  457. sites: {
  458. ':default': function () {
  459. return {
  460. name: '',
  461. link: 'https://userscripts.org/scripts/show/89544',
  462. beneath: true,
  463. blueprint: [],
  464. width: 430,
  465. emoticonDir: '',
  466. emoticonMax: 39,
  467. emoticons: [['', '']], // null emoticon
  468. shortcuts: WhutBB.db.shortcuts
  469. };
  470. },
  471. ':generic': function () {
  472. return {
  473. emoticonDir: 'https://what.cd/static/common/smileys/',
  474. emoticons: WhutBB.db.emoticons.gz.slice(0, 4),
  475. blueprint: [
  476. ['b', 'i', 'u', 's'], ['code'],
  477. ['color', 'size'], ['*'],
  478. ['url', 'img'], ['quote'],
  479. ['erase'], ['emoticon', 'shortcut', 'settings']
  480. ]
  481. };
  482. },
  483. ':test': function () { // for tests
  484. return {
  485. emoticonDir: 'https://what.cd/static/common/smileys/',
  486. emoticons: WhutBB.db.emoticons.gz,
  487. blueprint: [
  488. ['b', 'i', 'u', 's'], ['important', 'heading', 'code'],
  489. ['color', 'size'], ['gz_left', 'gz_center', 'gz_right'],
  490. ['#', '*'], ['url', 'img'], ['quote', 'pre', 'gz_src'], ['hide', 'mature'],
  491. ['torrent', 'artist', 'user', 'wiki', 'gz_rule'], ['tex', 'plain'],
  492. ['erase'], ['emoticon', 'shortcut', 'settings']
  493. ]
  494. };
  495. },
  496. ':markdown': function () {
  497. return {};
  498. },
  499. what: function () {
  500. return {
  501. link: '/wiki.php?action=article&name=BBCode',
  502. emoticonDir: 'https://what.cd/static/common/smileys/',
  503. emoticons: 'gazelle',
  504. width: 430,
  505. blueprint: [
  506. ['b', 'i', 'u', 's'], ['important', 'heading', 'code'],
  507. ['color', 'size'], ['gz_left', 'gz_center', 'gz_right'],
  508. ['#', '*'], ['url', 'img'], ['quote', 'pre', 'gz_src'], ['hide', 'mature'],
  509. ['artist', 'torrent', 'user', 'wiki', 'gz_rule'], ['tex', 'plain'],
  510. [ 'erase'], ['emoticon', 'settings', 'shortcut']
  511. ]
  512. };
  513. },
  514. indietorrents: function () {
  515. return {
  516. link: '/wiki.php?action=article&id=3',
  517. emoticonDir: 'static/common/smileys/',
  518. emoticons: 'indie',
  519. width: 440,
  520. blueprint: [
  521. ['b', 'i', 'u', 's'], ['color', 'size'],
  522. ['gz_left', 'gz_center', 'gz_right'], ['*'], ['url', 'img', 'youtube'],
  523. ['quote', 'pre', 'gz_src', 'hide'], ['table', 'tr', 'th', 'td'],
  524. ['artist', 'user', 'wiki'], ['tex', 'plain'],
  525. ['erase'], ['emoticon', 'settings', 'shortcut']
  526. ]
  527. };
  528. },
  529. waffles: function () {
  530. WhutBB.db.buttons.raw = WhutBB.db.buttons.plain;
  531.  
  532. return {
  533. link: '/bbcode.php',
  534. emoticonDir: 'https://www.waffles.fm/pic/smilies/',
  535. emoticons: 'waffles',
  536. beneath: false,
  537. width: 540,
  538. blueprint: [
  539. ['b', 'i', 'u', 's'], ['size', 'color', 'font', 'spoiler'],
  540. ['*'], ['url', 'img', 'youtube'],
  541. ['center', 'quote', 'pre', 'raw'],
  542. ['artist', 'user', 'torrent', 'search'],
  543. ['erase'], ['emoticon', 'settings', 'shortcut']
  544. ]
  545. };
  546. }
  547. },
  548. buttons: {
  549. b: {title: 'Bold', icon: 'bold'},
  550. i: {title: 'Italic', icon: 'italic'},
  551. u: {title: 'Underline', icon: 'text-width'}, //underline
  552. s: {title: 'Strike', icon: 'minus'},
  553. code: {display: 'c', title: 'Inline Code', icon: 'leaf'},
  554. important: {display: '!', title: 'Important Text', icon: 'exclamation-sign'},
  555. color: {type: 1, display: '\u25ee', prompt: 'Enter a #hexadecimal or color name.', title: 'Color', val: '#', icon: 'tint'},
  556. size: {type: 1, display: '\u00b1', prompt: 'Enter a number.', title: 'Size', val: 3, icon: 'text-height'},
  557. align: {type: 1, display: '-', title: 'Align Text', icon: 'align-left'},
  558. left: {display: '<', title: 'Align Left', icon: 'align-left'},
  559. center: {display: '\u2013', title: 'Align Center', icon: 'align-center'},
  560. right: {display: '>', title: 'Align Right', icon: 'align-right'},
  561. '#': {type: 3, title: 'Ordered List Item', icon: 'list-alt'},
  562. '*': {type: 3, title: 'List Item', icon: 'list'},
  563. url: {type: 1, prompt: 'Enter a Link', title: 'Web Link', val: 'http://', icon: 'globe'},
  564. img: {title: 'Image', icon: 'picture'},
  565. quote: {type: 1, display: 'q', prompt: 'Enter an author or name.', title: 'Quote', placeholder: 'author', icon: 'comment'},
  566. pre: {title: 'Preformated text/Code block', icon: 'asterisk'},
  567. hide: {display: 'h', title: 'Hide content/Spoilers', icon: 'warning-sign'},
  568. spoiler: {display: '_', title: 'Spoilers!', icon: 'exclamation-sign'},
  569. mature: {display: 'm', title: 'Hide mature content', icon: 'eye-open'},
  570. artist: {display: 'a', title: 'Link to an artist/band on the site', icon: 'music'},
  571. user: {display: 'p', title: 'Link to a person on the site', icon: 'user'},
  572. wiki: {type: 4, tag: ['[[', ']]'], display: 'w', title: 'Link to a Wiki article', icon: 'share'},
  573. tex: {display: 't', title: 'LaTeX', icon: 'pencil'},
  574. plain: {display: 'x', title: 'Disable BB tags.', icon: 'ban-circle'},
  575. youtube: {type: 2, display: 'yt', title: 'YouTube video', icon: 'film'},
  576. font: {type: 1, display: 'f', prompt: 'Enter a font\'s name', title: 'Font', val: 'Arial', icon: 'font'},
  577. torrent: {display: 'id', title: 'Link to a torrent ID.', icon: 'download'},
  578. search: {type: 1, display: '@', prompt: 'Enter a search term', title: 'Link to a search term.', val: 'keywords', icon: 'search'},
  579. table: {display: 'tbl', title: 'Insert a table.', icon: 'th-large'},
  580. tr: {display: 'tr', title: 'Insert a table row.', icon: 'th-list'},
  581. th: {display: 'th', title: 'Insert a table heading.', icon: 'th'},
  582. td: {display: 'td', title: 'Insert a table cell.', icon: 'pencil'},
  583. heading: {type: 4, tag: '=', display: '=', title: 'Insert a heading', icon: 'arrow-right'},
  584. // Gazelle
  585. gz_left: {tag: 'align', val: 'left', type: 1, noPrompt: true, display: '<', title: 'Align left', icon: 'align-left'},
  586. gz_center: {tag: 'align', val: 'center', type: 1, noPrompt: true, display: '\u2013', title: 'Align center', icon: 'align-center'},
  587. gz_right: {tag: 'align', val: 'right', type: 1, noPrompt: true, display: '>', title: 'Align right', icon: 'align-right'},
  588. gz_src: {macro: ['quote', 'pre'], type: -3, display: '<>', title: 'Source code', icon: 'tasks'},
  589. gz_rule: {tag: 'rule', title: 'Link to a rule', icon: 'info-sign', display: 'r' },
  590. // Panels
  591. emoticon: {display: ':]', toggle: ';]', title: 'Emoticons', type: -1, icon: 'fire'},
  592. settings: {display: '%', title: 'Settings', type: -1, icon: 'wrench'},
  593. shortcut: {display: '?', title: 'Shortcuts', type: -1, icon: 'question-sign'},
  594. erase: {display: '-', title: 'Delete message', type: -2, icon: 'remove-sign'}
  595. },
  596. emoticons: {
  597. gazelle: [[":angry:", "angry.gif"], [":D", "biggrin.gif"], [":|", "blank.gif"], [":blush:", "blush.gif"], [":cool:", "cool.gif"], [":'(", "crying.gif"], [">.>", "eyesright.gif"], [":creepy:", "creepy.gif"], [":frown:", "frown.gif"], ["<3", "heart.gif"], [":unsure:", "hmm.gif"], [":whatlove:", "ilu.gif"], [":lol:", "laughing.gif"], [":loveflac:", "loveflac.gif"], [":ninja:", "ninja.gif"], [":no:", "no.gif"], [":nod:", "nod.gif"], [":ohno:", "ohnoes.gif"], [":omg:", "omg.gif"], [":o", "ohshit.gif"], [":paddle:", "paddle.gif"], [":(", "sad.gif"], [":shifty:", "shifty.gif"], [":sick:", "sick.gif"], [":)", "smile.gif"], [":-)", "smile.gif"], [":sorry:", "sorry.gif"], [":thanks:", "thanks.gif"], [":P", "tongue.gif"], [":wave:", "wave.gif"], [":wink:", "wink.gif"], [":worried:", "worried.gif"], [":wtf:", "wtf.gif"], [":wub:", "wub.gif"], [":qmarklove:", "ilqmark-what.gif"], [":ajaxlove:", "ilajax-what.gif"], [":athenalove:", "ilathena-what.gif"], [":alderaanlove:", "ilalderaan-what.gif"], [":anankelove:", "ilananke-what.gif"], [":bashmorelove:", "ilbashmore-what.gif"], [":brancusilove:", "ilbrancusi-what.gif"], [":brdlove:", "ilbrd-what.gif"], [":carllove:", "ilcarl-what.gif"], [":dumontlove:", "ildumont-what.gif"], [":entrapmentlove:", "ilentrapment-what.gif"], [":espressolove:", "ilespresso-what.gif"], [":gamehendgelove:", "ilgamehendge-what.gif"], [":hyperionlove:", "ilhyperion-what.gif"], [":iapetuslove:", "iliapetus-what.gif"], [":irimiaslove:", "ilirimias-what.gif"], [":irredentialove:", "ilirredentia-what.gif"], [":kitchenstafflove:", "ilkitchenstaff-what.gif"], [":kopitiamlove:", "ilkopitiam-what.gif"], [":kryptoslove:", "ilkryptos-what.gif"], [":lenreklove:", "illenrek-what.gif"], [":lesadieuxlove:", "illesadieux-what.gif"], [":lisbethlove:", "illisbeth-what.gif"], [":nandolove:", "ilnando-what.gif"], [":porkpielove:", "ilporkpie-what.gif"], [":sinetaxlove:", "ilsinetax-what.gif"], [":theseuslove:", "iltheseus-what.gif"], [":toruslove:", "iltorus-what.gif"], [":wtelove:", "ilwte-what.gif"], [":zettellove:", "ilzettel-what.gif"], [":a9love:", "ila9-what.gif"], [":bionicsockslove:", "ilbionicsocks-what.gif"], [":chailove:", "ilchai-what.gif"], [":changleslove:", "ilchangles-what.gif"], [":claptonlove:", "ilclapton-what.gif"], [":emmlove:", "ilemm-what.gif"], [":fzeroxlove:", "ilfzerox-what.gif"], [":hothlove:", "ilhoth-what.gif"], [":interstellarlove:", "ilinterstellar-what.gif"], [":jowalove:", "iljowa-what.gif"], [":kharonlove:", "ilkharon-what.gif"], [":lylaclove:", "illylac-what.gif"], [":marienbadlove:", "ilmarienbad-what.gif"], [":marigoldslove:", "ilmarigolds-what.gif"], [":mavericklove:", "ilmaverick-what.gif"], [":mnlove:", "ilmn-what.gif"], [":mre2melove:", "ilmre2me-what.gif"], [":mugglelove:", "ilmugglehump-what.gif"], [":nightoathlove:", "ilnightoath-what.gif"], [":oinkmeuplove:", "iloinkmeup-what.gif"], [":padutchlove:", "ilpadutch-what.gif"], [":paintrainlove:", "ilpaintrain-what.gif"], [":sdfflove:", "ilsdff-what.gif"], [":seraphiellove:", "ilseraphiel-what.gif"], [":sisterraylove:", "ilsisterray-what.gif"], [":snowflakelove:", "ilsnowflake-what.gif"], [":soamlove:", "ilsoam-what.gif"], [":spacireleilove:", "ilspacirelei-what.gif"], [":stwlove:", "ilstw-what.gif"], [":whatmanlove:", "ilwhatman-what.gif"], [":whynotmicelove:", "ilwhynotmice-what.gif"], [":xorianlove:", "ilxorian-what.gif"]],
  598. waffles: [[':waffleslove:', 'wubwaffles.gif'], [':opplove:', 'opplove.gif'], [':-)', 'smile1.gif'], [':smile:', 'smile2.gif'], [':-D', 'grin.gif'], [':lol:', 'laugh.gif'], [':w00t:', 'w00t.gif'], [':think:', 'think.gif'], [':-P', 'tongue.gif'], [';-)', 'wink.gif'], [':-|', 'noexpression.gif'], [':-/', 'confused.gif'], [':-(', 'sad.gif'], [':cry:', 'cry.gif'], [':crybaby:', 'crybaby.gif'], [':weep:', 'weep.gif'], [':-O', 'ohmy.gif'], [':o)', 'clown.gif'], ['8-)', 'cool1.gif'], ['|-)', 'sleeping.gif'], [':bite:', 'bite.gif'], [':innocent:', 'innocent.gif'], [':whistle:', 'whistle.gif'], [':unsure:', 'unsure.gif'], [':closedeyes:', 'closedeyes.gif'], [':cool:', 'cool2.gif'], [':fun:', 'fun.gif'], [':thumbsup:', 'thumbsup.gif'], [':thumbsdown:', 'thumbsdown.gif'], [':blush:', 'blush.gif'], [':yes:', 'yes.gif'], [':no:', 'no.gif'], [':love:', 'love.gif'], [':?:', 'question.gif'], [':!:', 'excl.gif'], [':idea:', 'idea.gif'], [':arrow:', 'arrow.gif'], [':arrow2:', 'arrow2.gif'], [':hmm:', 'hmm.gif'], [':hmmm:', 'hmmm.gif'], [':huh:', 'huh.gif'], [':geek:', 'geek.gif'], [':look:', 'look.gif'], [':rolleyes:', 'rolleyes.gif'], [':kiss:', 'kiss.gif'], [':shifty:', 'shifty.gif'], [':blink:', 'blink.gif'], [':smartass:', 'smartass.gif'], [':sick:', 'sick.gif'], [':crazy:', 'crazy.gif'], [':orly:', 'orly.gif'], [':wacko:', 'wacko.gif'], [':alien:', 'alien.gif'], [':wizard:', 'wizard.gif'], [':wave:', 'wave.gif'], [':wavecry:', 'wavecry.gif'], [':baby:', 'baby.gif'], [':angry:', 'angry.gif'], [':ras:', 'ras.gif'], [':sly:', 'sly.gif'], [':devil:', 'devil.gif'], [':evil:', 'evil.gif'], [':evilmad:', 'evilmad.gif'], [':sneaky:', 'sneaky.gif'], [':axe:', 'axe.gif'], [':slap:', 'slap.gif'], [':wall:', 'wall.gif'], [':rant:', 'rant.gif'], [':jump:', 'jump.gif'], [':yucky:', 'yucky.gif'], [':nugget:', 'nugget.gif'], [':smart:', 'smart.gif'], [':shutup:', 'shutup.gif'], [':shutup2:', 'shutup2.gif'], [':crockett:', 'crockett.gif'], [':zorro:', 'zorro.gif'], [':snap:', 'snap.gif'], [':beer:', 'beer.gif'], [':beer2:', 'beer2.gif'], [':drunk:', 'drunk.gif'], [':strongbench:', 'strongbench.gif'], [':weakbench:', 'weakbench.gif'], [':dumbells:', 'dumbells.gif'], [':music:', 'music.gif'], [':stupid:', 'stupid.gif'], [':dots:', 'dots.gif'], [':offtopic:', 'offtopic.gif'], [':spam:', 'spam.gif'], [':oops:', 'oops.gif'], [':lttd:', 'lttd.gif'], [':please:', 'please.gif'], [':sorry:', 'sorry.gif'], [':hi:', 'hi.gif'], [':yay:', 'yay.gif'], [':cake:', 'cake.gif'], [':hbd:', 'hbd.gif'], [':band:', 'band.gif'], [':punk:', 'punk.gif'], [':rofl:', 'rofl.gif'], [':bounce:', 'bounce.gif'], [':mbounce:', 'mbounce.gif'], [':thankyou:', 'thankyou.gif'], [':gathering:', 'gathering.gif'], [':hang:', 'hang.gif'], [':chop:', 'chop.gif'], [':rip:', 'rip.gif'], [':whip:', 'whip.gif'], [':judge:', 'judge.gif'], [':chair:', 'chair.gif'], [':tease:', 'tease.gif'], [':boxing:', 'boxing.gif'], [':guns:', 'guns.gif'], [':shoot:', 'shoot.gif'], [':shoot2:', 'shoot2.gif'], [':flowers:', 'flowers.gif'], [':wub:', 'wub.gif'], [':lovers:', 'lovers.gif'], [':kissing:', 'kissing.gif'], [':kissing2:', 'kissing2.gif'], [':console:', 'console.gif'], [':group:', 'group.gif'], [':hump:', 'hump.gif'], [':hooray:', 'hooray.gif'], [':happy2:', 'happy2.gif'], [':clap:', 'clap.gif'], [':clap2:', 'clap2.gif'], [':weirdo:', 'weirdo.gif'], [':yawn:', 'yawn.gif'], [':bow:', 'bow.gif'], [':dawgie:', 'dawgie.gif'], [':cylon:', 'cylon.gif'], [':book:', 'book.gif'], [':fish:', 'fish.gif'], [':mama:', 'mama.gif'], [':pepsi:', 'pepsi.gif'], [':medieval:', 'medieval.gif'], [':rambo:', 'rambo.gif'], [':ninja:', 'ninja.gif'], [':hannibal:', 'hannibal.gif'], [':party:', 'party.gif'], [':snorkle:', 'snorkle.gif'], [':evo:', 'evo.gif'], [':king:', 'king.gif'], [':chef:', 'chef.gif'], [':mario:', 'mario.gif'], [':pope:', 'pope.gif'], [':fez:', 'fez.gif'], [':cap:', 'cap.gif'], [':cowboy:', 'cowboy.gif'], [':pirate:', 'pirate.gif'], [':pirate2:', 'pirate2.gif'], [':rock:', 'rock.gif'], [':cigar:', 'cigar.gif'], [':icecream:', 'icecream.gif'], [':oldtimer:', 'oldtimer.gif'], [':trampoline:', 'trampoline.gif'], [':banana:', 'bananadance.gif'], [':smurf:', 'smurf.gif'], [':yikes:', 'yikes.gif'], [':osama:', 'osama.gif'], [':saddam:', 'saddam.gif'], [':santa:', 'santa.gif'], [':indian:', 'indian.gif'], [':pimp:', 'pimp.gif'], [':nuke:', 'nuke.gif'], [':jacko:', 'jacko.gif'], [':ike:', 'ike.gif'], [':greedy:', 'greedy.gif'], [':super:', 'super.gif'], [':wolverine:', 'wolverine.gif'], [':spidey:', 'spidey.gif'], [':spider:', 'spider.gif'], [':bandana:', 'bandana.gif'], [':construction:', 'construction.gif'], [':sheep:', 'sheep.gif'], [':police:', 'police.gif'], [':detective:', 'detective.gif'], [':bike:', 'bike.gif'], [':fishing:', 'fishing.gif'], [':clover:', 'clover.gif'], [':horse:', 'horse.gif'], [':shit:', 'shit.gif'], [':soldiers:', 'soldiers.gif'], [':search:', 'search.gif'], [':tinfoilhat:', 'tinfoilhat.gif'], [':moon1:', 'moon1.gif'], [':moon2:', 'moon2.gif'], [':user:', 'user.gif'], [':staff:', 'staff.gif'], [':eggo:', 'eggo.png']], /*, [':box:', 'box.gif']*/
  599. indie: [[':-)', 'smile.gif'], [';-)', 'wink.gif'], [':-D', 'biggrin.gif'], [':-P', 'tongue.gif'], [':-(', 'sad.gif'], ['>:-|', 'blank.gif'], [':-/', 'confused.gif'], [':-O', 'ohmy.gif'], [':o)', 'clown.gif'], ['8-)', 'cool1.gif'], ['|-)', 'sleeping.gif'], [':cupcake:', 'cupcake1.gif'], [':innocent:', 'innocent.gif'], [':whistle:', 'whistle.gif'], [':unsure:', 'hmm.gif'], [':closedeyes:', 'closedeyes.gif'], [':angry:', 'angry.gif'], [':smile:', 'smile2.gif'], [':lol:', 'laughing.gif'], [':cool:', 'cool.gif'], [':fun:', 'fun.gif'], [':thumbsup:', 'thumbsup.gif'], [':thumbsdown:', 'thumbsdown.gif'], [':blush:', 'blush.gif'], [':weep:', 'weep.gif'], [':yes:', 'yes.gif'], [':no:', 'no.gif'], [':love:', 'love.gif'], [':?:', 'question.gif'], [':!:', 'excl.gif'], [':idea:', 'idea.gif'], [':arrow:', 'arrow.gif'], [':hmm:', 'hmm.gif'], [':hmmm:', 'hmmm.gif'], [':huh:', 'huh.gif'], [':w00t:', 'w00t.gif'], [':geek:', 'geek.gif'], [':look:', 'look.gif'], [':rolleyes:', 'rolleyes.gif'], [':kiss:', 'kiss.gif'], [':shifty:', 'shifty.gif'], [':blink:', 'blink.gif'], [':smartass:', 'smartass.gif'], [':sick:', 'sick.gif'], [':crazy:', 'crazy.gif'], [':wacko:', 'wacko.gif'], [':alien:', 'alien.gif'], [':wizard:', 'wizard.gif'], [':wave:', 'wave.gif'], [':wavecry:', 'wavecry.gif'], [':baby:', 'baby.gif'], [':ras:', 'ras.gif'], [':sly:', 'sly.gif'], [':devil:', 'devil.gif'], [':evil:', 'evil.gif'], [':godisevil:', 'evil.gif'], [':evilmad:', 'evilmad.gif'], [':yucky:', 'yucky.gif'], [':nugget:', 'nugget.gif'], [':sneaky:', 'sneaky.gif'], [':smart:', 'smart.gif'], [':shutup:', 'shutup.gif'], [':shutup2:', 'shutup2.gif'], [':yikes:', 'yikes.gif'], [':flowers:', 'flowers.gif'], [':wub:', 'wub.gif'], [':osama:', 'osama.gif'], [':saddam:', 'saddam.gif'], [':santa:', 'santa.gif'], [':indian:', 'indian.gif'], [':guns:', 'guns.gif'], [':crockett:', 'crockett.gif'], [':zorro:', 'zorro.gif'], [':snap:', 'snap.gif'], [':beer:', 'beer.gif'], [':beer2:', 'beer2.gif'], [':drunk:', 'drunk.gif'], [':mama:', 'mama.gif'], [':pepsi:', 'pepsi.gif'], [':medieval:', 'medieval.gif'], [':rambo:', 'rambo.gif'], [':ninja:', 'ninja.gif'], [':hannibal:', 'hannibal.gif'], [':party:', 'party.gif'], [':snorkle:', 'snorkle.gif'], [':evo:', 'evo.gif'], [':king:', 'king.gif'], [':chef:', 'chef.gif'], [':mario:', 'mario.gif'], [':pope:', 'pope.gif'], [':fez:', 'fez.gif'], [':cap:', 'cap.gif'], [':cowboy:', 'cowboy.gif'], [':pirate:', 'pirate2.gif'], [':rock:', 'rock.gif'], [':cigar:', 'cigar.gif'], [':icecream:', 'icecream.gif'], [':oldtimer:', 'oldtimer.gif'], [':wolverine:', 'wolverine.gif'], [':strongbench:', 'strongbench.gif'], [':weakbench:', 'weakbench.gif'], [':bike:', 'bike.gif'], [':music:', 'music.gif'], [':book:', 'book.gif'], [':fish:', 'fish.gif'], [':stupid:', 'stupid.gif'], [':dots:', 'dots.gif'], [':kelso:', 'kelso.gif'], [':red:', 'red.gif'], [':dobbs:', 'bobdobbs.gif'], [':axe:', 'axe.gif'], [':hooray:', 'hooray.gif'], [':yay:', 'yay.gif'], [':cake:', 'cake.gif'], [':hbd:', 'hbd.gif'], [':hi:', 'hi.gif'], [':offtopic:', 'offtopic.gif'], [':band:', 'band.gif'], [':hump:', 'hump.gif'], [':punk:', 'punk.gif'], [':bounce:', 'bounce.gif'], [':mbounce:', 'mbounce.gif'], [':group:', 'group.gif'], [':console:', 'console.gif'], [':smurf:', 'smurf.gif'], [':soldiers:', 'soldiers.gif'], [':spidey:', 'spidey.gif'], [':rant:', 'rant.gif'], [':pimp:', 'pimp.gif'], [':nuke:', 'nuke.gif'], [':judge:', 'judge.gif'], [':jacko:', 'jacko.gif'], [':ike:', 'ike.gif'], [':greedy:', 'greedy.gif'], [':dumbells:', 'dumbells.gif'], [':clover:', 'clover.gif'], [':shit:', 'shit.gif'], [':thankyou:', 'thankyou.gif'], [':horse:', 'horse.gif'], [':box:', 'boxing.gif'], [':fight:', 'fighting05.gif'], [':gathering:', 'gathering.gif'], [':hang:', 'hang.gif'], [':chair:', 'chair.gif'], [':spam:', 'spam.gif'], [':bandana:', 'bandana.gif'], [':construction:', 'construction.gif'], [':oops:', 'oops.gif'], [':rip:', 'rip.gif'], [':sheep:', 'sheep.gif'], [':tease:', 'tease.gif'], [':spider:', 'spider.gif'], [':shoot:', 'shoot.gif'], [':shoot2:', 'shoot2.gif'], [':police:', 'police.gif'], [':lovers:', 'lovers.gif'], [':kissing:', 'kissing.gif'], [':kissing2:', 'kissing2.gif'], [':jump:', 'jump.gif'], [':happy2:', 'happy2.gif'], [':clap:', 'clap.gif'], [':clap2:', 'clap2.gif'], [':chop:', 'chop.gif'], [':lttd:', 'lttd.gif'], [':whip:', 'whip.gif'], [':yawn:', 'yawn.gif'], [':bow:', 'bow.gif'], [':slap:', 'slap.gif'], [':wall:', 'wall.gif'], [':please:', 'please.gif'], [':sorry:', 'sorry.gif'], [':finger:', 'finger.gif'], [':brown:', 'brownnoser.gif'], [':cloud9:', 'cloud9.gif'], [':pity:', 'mrt.gif'], [':mug:', 'mug.gif'], [':banned:', 'banned.gif'], [':tkfu:', 'ninja_hide.gif'], [':baldfresh:', 'baldy.png'], [':camera:', 'camera.gif'], [':loggeek:', 'log.jpg'], [':coleman83:', 'random'], [':locked:', 'lockd.gif'], [':tomjones1:', 'tomjones01.png'], [':tomjones2:', 'tomjones02.png'], [':D', 'biggrin.gif'], [':|', 'blank.gif'], [':\'(', 'crying.gif'], ['>.>', 'eyesright.gif'], [':frown:', 'frown.gif'], ['<3', 'heart.gif'], [':nod:', 'nod.gif'], [':ohno:', 'ohnoes.gif'], [':ohnoes:', 'ohnoes.gif'], [':omg:', 'omg.gif'], [':o', 'ohshit.gif'], [':O', 'ohshit.gif'], [':paddle:', 'paddle.gif'], [':(', 'sad.gif'], [':)', 'smile.gif'], [':thanks:', 'thanks.gif'], [':P', 'tongue.gif'], [':-p', 'tongue.gif'], [':wink:', 'wink.gif'], [':creepy:', 'creepy.gif'], [':worried:', 'worried.gif'], [':wtf:', 'wtf.gif'], [':lmgtfy:', 'lmgtfy.gif'], [':fart:', 'fart.gif'], [':hifi:', 'hifi.gif'], [':cheers:', 'cheers.gif'], [':jambox:', 'jambox.gif'], [':rimshot:', 'rimshot.gif'], [':rockout:', 'rockout.gif'], [':yourmom:', 'yourmom.gif'], [':bong:', 'bong.gif'], [':peace:', 'hippie.gif'], [':vinyl:', 'vinyl.gif'], ['\\m/', 'horns.gif']]
  600. },
  601. shortcuts: {
  602. alt: {
  603. c: 'gz_src'
  604. },
  605. ctrl: {
  606. b: 'b',
  607. i: 'i',
  608. u: 'u',
  609. s: 's',
  610. g: 'code',
  611. k: '#',
  612. l: '*',
  613. h: 'url',
  614. m: 'img',
  615. d: 'erase'
  616. },
  617. 'ctrl+alt': {
  618. i: 'important',
  619. e: 'emoticon',
  620. u: 'settings',
  621. x: 'shortcut'
  622. }
  623. },
  624. getShortcut: function (modifier, letter) {
  625. if (this.shortcuts[modifier] && this.shortcuts[modifier][letter]) {
  626. return this.shortcuts[modifier][letter];
  627. }
  628. },
  629. getSiteSettings: function (name) {
  630. if (WhutBB.db.sites[name]) {
  631. var settings = WhutBB.db.sites[name]();
  632. settings.name = name;
  633. return settings;
  634. }
  635. return {};
  636. },
  637. /**
  638. * Inserts or replaces buttons
  639. * Use this method before initializing the script (WhutBB.init)
  640. * @param buttons - object of objects
  641. */
  642. insertButtons: function (buttons) {
  643. dom.oEach(buttons, function (name, object) {
  644. WhutBB.db.buttons[name] = object;
  645. });
  646. },
  647. /**
  648. * Adds emoticons to (an exisiting) emoticons DB array
  649. * Use this method before initializing the script (WhutBB.init)
  650. *
  651. * @param name of array in the emoticons DB to use
  652. * if none exist, it will be created
  653. * @param emoticons array
  654. * make sure to use an array of arrays
  655. *
  656. * Example: add two emoticons to WhutBB.db.emoticons.gazelle
  657. * WhutBB.db.addEmoticons('gazelle', [[':new:', 'new.png'], [':pop:', 'pop.png']]);
  658. */
  659. addEmoticons: function (name, emoticons) {
  660. WhutBB.db.emoticons[name] = (WhutBB.db.emoticons[name] || []).concat(emoticons);
  661. }
  662. };
  663.  
  664. /**
  665. * Event manager
  666. * Aliases/references event data for easier use within various methods
  667. */
  668. WhutBB.e = {
  669. current: null, // alias for the current event
  670. target: null, // alias for the current event target element
  671. whut: null, // alias for the current event's WhutBB instance
  672. macro: false, // flag for events called through a macro
  673. set: function (event, target, wbb) {
  674. WhutBB.e.current = event;
  675. WhutBB.e.target = target;
  676. WhutBB.e.whut = wbb;
  677. },
  678. clean: function () {
  679. this.current = this.target = this.whut = null;
  680. }
  681. };
  682.  
  683. /**
  684. * Event Object
  685. *
  686. * Contains all possible events, divided into:
  687. * 1) mouse, 2) key, and 3) general button events
  688. *
  689. * Mouse and Key events trigger Button events, depending
  690. * on the button type
  691. *
  692. * As mentioned earlier, buttons with custom events should find
  693. * a method with that button's name within WhutBB.evt.button.custom
  694. *
  695. * WhutBB instances register themselves with the
  696. * register methods.
  697. *
  698. * The registers return an annonymous function that
  699. * is used for any subsequent click or key events.
  700. *
  701. */
  702. WhutBB.evt = {
  703. button: { // button events
  704. custom: { // Custom button events
  705. erase: function () { // erase button event
  706. WhutBB.e.whut.textarea.value = '';
  707. },
  708. emoticonLoader: function () { // removes "View all emoticons." div and loads remaining emoticons
  709. WhutBB.e.target.parentNode.removeChild(WhutBB.e.target);
  710. WhutBB.Panel.attach.emoticons(WhutBB.config.emoticonMax - 1,
  711. WhutBB.config.emoticons.length);
  712. }
  713. },
  714. macro: function (name, wbb) { // macro button events
  715. if (!WhutBB.e.macro) {
  716. WhutBB.e.macro = true;
  717. dom.aEach(WhutBB.db.buttons[name].macro || [], function (name) {
  718. // console.log(name);
  719. dom.click(wbb.getButton(name));
  720. });
  721. WhutBB.e.macro = false;
  722. }
  723. },
  724. bbcode: function () { // bbcode buttons
  725. WhutBB.Tag.get(WhutBB.e.target.name).insertTo(WhutBB.e.whut.textarea);
  726. },
  727. emoticon: function () { // emoticon buttons
  728. WhutBB.box.select(WhutBB.e.whut.textarea).insert([' ' + WhutBB.e.target.title, '']);
  729. },
  730. panel: { // panel buttons
  731. toggle: function (panel, el) { // el = WhutBB.e.target
  732. var visible = /(?:wbbpressed)/i.test(el.className); // panel's current visibility
  733. WhutBB.evt.button.panel.store(el);
  734. if (visible) {
  735. el.className = el.className.replace(' wbbpressed', '');
  736. panel.className += ' wbbhide';
  737. } else {
  738. WhutBB.e.whut.wrap.appendChild(WhutBB.Panel.global[el.name].element);
  739. el.className += ' wbbpressed';
  740. panel.className = panel.className.replace(' wbbhide', '');
  741. }
  742. WhutBB.evt.button.panel.toggleText(visible, el.firstChild);
  743. },
  744. toggleText: function (visible, span) {
  745. if (span.getAttribute('data-toggle') !== '') {
  746. span.firstChild.nodeValue = span.getAttribute(visible ? 'data-txt' : 'data-toggle');
  747. }
  748. },
  749. store: function (button) {
  750. // remove pressed (toggled) state of previous stored button
  751. if (WhutBB.evt.button.panel.store[button.name]) {
  752. WhutBB.evt.button.panel.store[button.name].className = 'whutbbutton';
  753. WhutBB.evt.button.panel.toggleText(true, WhutBB.evt.button.panel.store[button.name].firstChild);
  754. }
  755. WhutBB.evt.button.panel.store[button.name] = button;
  756. },
  757. stored: {}
  758. }
  759. },
  760. delegate: {
  761. button: function () { // TODO Polymorphism plz?
  762. var t = WhutBB.e.target;
  763. // console.log(t);
  764. WhutBB.e.current.stopPropagation();
  765. if (+t.getAttribute('data-type') === -3) {
  766. // console.log(-3);
  767. return WhutBB.evt.button.macro(t.name, WhutBB.e.whut);
  768. }
  769. if (+t.getAttribute('data-type') === -2) {
  770. // console.log(-2);
  771. return WhutBB.evt.button.custom[t.name]();
  772. }
  773. if (+t.getAttribute('data-type') === -1) {
  774. // console.log(-1);
  775. return WhutBB.evt.button.panel.toggle(WhutBB.Panel.global[t.name].element, t);
  776. }
  777. if (t.getAttribute('data-type') === 'emoticon') {
  778. // console.log(2);
  779. return WhutBB.evt.button.emoticon();
  780. }
  781. // console.log(1);
  782. return WhutBB.evt.button.bbcode();
  783. },
  784. edit: function (evt) { // RegExp.lastParen should contain an ID
  785. var el = evt.target,
  786. onclick = el.getAttribute('onclick') || '';
  787. if (onclick.match(/(?:Edit_Form\('(\d+))/)) {
  788. return window.setTimeout(function () {
  789. var txt = document.getElementById('editbox' + RegExp.lastParen);
  790. txt.setAttribute('data-wbb', RegExp.lastParen);
  791. WhutBB.create(txt, true);
  792. }, 500);
  793. }
  794. if (onclick.match(/(?:Preview_Edit\((\d+))/) || onclick.match(/(?:Save_Edit\((\d+))/)) {
  795. return WhutBB.set[RegExp.lastParen].hide();
  796. }
  797. if (onclick.match(/(?:Cancel_Preview\((\d+))/)) {
  798. return WhutBB.set[RegExp.lastParen].show();
  799. }
  800. },
  801. inbox: function (evt) { // todo inbox.php
  802. var el = evt.target;
  803. // console.log('inbox');
  804. if (/(?:preview)/i.test(el.value)) {
  805. document.getElementById('quickpost').className += ' wbbhide';
  806. document.getElementById('quickpost').nextSibling.className += ' wbbhide';
  807. }
  808. if (/(?:editor)/i.test(el.value)) {
  809. document.getElementById('quickpost').className = document.getElementById('quickpost').className.replace(/(?: wbbhide)/g, '');
  810. document.getElementById('quickpost').nextSibling.className = document.getElementById('quickpost').nextSibling.className.replace(/(?: wbbhide)/g, '');
  811. }
  812. },
  813. settings: { // settings events
  814. update: function () { // translates checks into settings to store
  815. var settings = {}, saved;
  816.  
  817. dom.aEach(WhutBB.Panel.global.settings.element.getElementsByTagName('input'), function (el) {
  818. settings[el.name] = el.checked;
  819. });
  820.  
  821. saved = WhutBB.user.save(settings);
  822.  
  823. // calls a sub function based on a setting's name
  824. // additional argument if the settings were saved
  825. if (this.fn[WhutBB.e.target.name]) {
  826. this.fn[WhutBB.e.target.name](saved);
  827. }
  828. },
  829. fn: {
  830. icon: function () { // toggles button icons
  831. var cls = 'wbbcode ' + WhutBB.$.data.getWrapClass();
  832. dom.oEach(WhutBB.set, function (id, wbb) {
  833. wbb.wrap.className = cls;
  834. });
  835. },
  836. link: function () { // toggles WhutBBCode? link
  837. var cls = 'wbblink ' + (WhutBB.user.settings.link ? '' : ' wbbhide');
  838. dom.oEach(WhutBB.set, function (id, wbb) {
  839. wbb.panels.link.className = cls;
  840. });
  841. }
  842. }
  843. }
  844. },
  845. mouse: {
  846. target: function (target) {
  847. // WebKit issue -- This returns an actual button, instead of the span.icon-* within it
  848. return (/(?:icon-)/).test(target.getAttribute('class')) ? target.parentNode : target;
  849. },
  850. down: function () {
  851. if (WhutBB.e.target.getAttribute('data-type')) {
  852. return WhutBB.evt.delegate.button();
  853. }
  854. if (WhutBB.e.target.getAttribute('data-setting')) {
  855. return WhutBB.evt.delegate.settings.update();
  856. }
  857. },
  858. register: function (wbb) {
  859. return function (evt) { // context for _this_ is the container div.wbbbuttons
  860. // console.log('mouse.register/anon');
  861. WhutBB.e.set(evt, WhutBB.evt.mouse.target(evt.target), wbb);
  862. WhutBB.evt.mouse.down();
  863. WhutBB.e.clean();
  864. };
  865. }
  866. },
  867. key: {
  868. down: function () {
  869. this.fire(this.button());
  870. },
  871. letter: function () {
  872. return String.fromCharCode(WhutBB.e.current.which || WhutBB.e.current.keyCode).toLowerCase();
  873. },
  874. modifier: function () {
  875. // meta key aliases to ctrl
  876. var cm = WhutBB.e.current.ctrlKey || WhutBB.e.current.metaKey;
  877. if (cm && WhutBB.e.current.altKey) { return 'ctrl+alt'; }
  878. if (cm) { return 'ctrl'; }
  879. if (WhutBB.e.current.altKey) { return 'alt'; }
  880. return '';
  881. },
  882. button: function () {
  883. return WhutBB.e.whut.getButton(WhutBB.db.getShortcut(this.modifier(), this.letter()));
  884. },
  885. fire: function (button) {
  886. if (button) {
  887. WhutBB.e.current.preventDefault();
  888. // dom.click(button);
  889. WhutBB.e.target = button;
  890. WhutBB.evt.mouse.down();
  891. }
  892. },
  893. register: function (wbb) {
  894. return function (evt) {
  895. // console.log('key.register/anon');
  896. WhutBB.e.set(evt, this, wbb); // _this_ is a textarea
  897. WhutBB.evt.key.down();
  898. WhutBB.e.clean();
  899. };
  900. },
  901. completeStop: function (e) {
  902. // prevents certain browsers (eg Firefox) from using their default actions
  903. e.preventDefault();
  904. e.stopPropagation();
  905. return false;
  906. }
  907. }
  908. };
  909.  
  910. /**
  911. * Box Object (aka textarea stuff)
  912. *
  913. * How it works:
  914. * WhutBB.box.select(textarea).insert(['{start}', '{end}']);
  915. *
  916. * An array is used because Tags parse to that data type.
  917. *
  918. * Result:
  919. * <textarea>{start}{end}</textarea>
  920. *
  921. * It's (more) magical when used in an event.
  922. */
  923. WhutBB.box = {
  924. select: function (textarea) {
  925. this.textarea = textarea;
  926. WhutBB.box.range.data = this.range.get();
  927. return this;
  928. },
  929. range: {
  930. get: function () {
  931. // Todo abstract ie and standard methods
  932. return ie ? this.ie() : this.standard();
  933. },
  934. rdata: function (start, end, selection) {
  935. return { start: start, end: end, selection: selection };
  936. },
  937. ie: function () {
  938. WhutBB.box.textarea.focus(); // important here
  939. var ieRange = document.selection.createRange(),
  940. dup = ieRange.duplicate(),
  941. start,
  942. end,
  943. selection;
  944. dup.moveToElementText(WhutBB.box.textarea);
  945. dup.setEndPoint('EndToEnd', ieRange);
  946.  
  947. start = dup.text.length - ieRange.text.length;
  948. end = start + ieRange.text.length;
  949. selection = ieRange.text.replace(/\r\n/g, '\n');
  950. if (end === 0 && start === 0) { // Push inserts to the end
  951. start = end = WhutBB.box.textarea.value.length;
  952. }
  953. // console.log('tx1 ' + ieRange.text, 'txs ' + dup.text, 'txt ' + ieRange.text.replace(/\r\n/g, '\n'), 'SST ' + start, 'SSE ' + end);
  954. ieRange = dup = null;
  955. return this.rdata(start, end, selection);
  956. },
  957. standard: function () {
  958. if (WhutBB.box.textarea.selectionStart < 0) { return; }
  959. if (WhutBB.box.textarea.selectionEnd > WhutBB.box.textarea.value.length) {
  960. WhutBB.box.textarea.selectionEnd = WhutBB.box.textarea.value.length;
  961. }
  962. var s = WhutBB.box.textarea.selectionStart || 0,
  963. e = WhutBB.box.textarea.selectionEnd || 0;
  964. return this.rdata(s, e, WhutBB.box.textarea.value.substring(s, e) || '');
  965. }
  966. },
  967. insert: function (tag) {
  968. var pre = WhutBB.box.textarea.value.substring(0, WhutBB.box.range.data.start) + tag[0],
  969. post = tag[1] + WhutBB.box.textarea.value.substring(WhutBB.box.range.data.end);
  970. WhutBB.box.textarea.value = [pre, WhutBB.box.range.data.selection, post].join('');
  971. WhutBB.box.selection(pre.length);
  972. },
  973. selection: function (start) {
  974. var r;
  975. WhutBB.box.textarea.focus();
  976. if (ie) {
  977. r = WhutBB.box.textarea.createTextRange();
  978. r.collapse(true);
  979. r.moveStart('character', start);
  980. r.moveEnd('character', WhutBB.box.range.data.selection.length);
  981. r.select();
  982. } else {
  983. WhutBB.box.textarea.setSelectionRange(start, start + WhutBB.box.range.data.selection.length);
  984. }
  985. }
  986. };
  987.  
  988. /**
  989. * WhutBBCode Settings Class
  990. * Intended to be a singleton used within WhutBB.init()
  991. *
  992. * This class is used to store site configurations for WhutBBCode?
  993. * Using these options, the script can create buttons, emoticons, etc.
  994. *
  995. * Effectively, without any settings, nothing really happens.
  996. *
  997. * The most important option is blueprint, which tells the script which
  998. * buttons to create.
  999. *
  1000. * The Panel class uses this blueprint to construct buttons, put them in the button
  1001. * panel, and attach them to WhutBB instances.
  1002. *
  1003. * All buttons that exist in WhutBB.db.buttons are stored as validButtons. The script
  1004. * uses validButtons to list available shortcuts to the user.
  1005. *
  1006. * To reiterate, options are the most important aspect of this class
  1007. *
  1008. * param @options object with the following (mostly optional) attributes
  1009. *
  1010. * if none is given, the script will try to find an appropriate match
  1011. * for the site.
  1012. *
  1013. * if no setting is found, the "generic" default options will be used
  1014. *
  1015. * name: (String) [ default: '' ]
  1016. * the website's name
  1017. *
  1018. * link:
  1019. * link to information about the site's BBCode or WhutBBCode? itself (default)
  1020. *
  1021. * beneath: (Boolean) [ default: true ]
  1022. * location to insert buttons, beneath or above the textarea
  1023. *
  1024. * blueprint: (Array) [ default: [] ]
  1025. * an array of arrays containing buttons to create
  1026. *
  1027. * group buttons together to create a set of similiar types
  1028. *
  1029. * example:
  1030. *
  1031. * blueprint: [
  1032. * ['b', 'i', 'u'], // a set of three buttons
  1033. * ['shortcut', 'settings'] // a set of two
  1034. * ]
  1035. *
  1036. * buttons are then placed in the DOM in the following order
  1037. * [b][i][u] [?][+]
  1038. *
  1039. * each set is separated by a space
  1040. *
  1041. * width: (Number) [ default: 430 ]
  1042. * a width (in pixels) to set for the WhutBB.wrap so that buttons fit well
  1043. *
  1044. * emoticonDir: [ default: '' ]
  1045. * absolute or relative (to the current location) location to where emoticons reside
  1046. * it should end in a slash (/)
  1047. *
  1048. * emoticonMax: (Number) [ default: 39 ]
  1049. * a limit of emoticons to create
  1050. *
  1051. * emoticons: (String|Array) [ default: [['', '']] ] (a null emoticon)
  1052. * - name of the object from WhutBB.db.emoticons[options.emoticons]
  1053. * three possible options: gazelle, waffles, indie
  1054. *
  1055. * - an array of arrays containing emoticons to create
  1056. *
  1057. * the sub-arrays are formed by the emoticon text and the name (and location) of the
  1058. * image to produce
  1059. *
  1060. * example:
  1061. * emoticons: [ [":)", "happy.png"], [":D", "grin.png"], [":(", "sad.png"] ]
  1062. *
  1063. * these create images based on the emoticon directory (emoticonDir)
  1064. * if the directory varies, it should be included
  1065. *
  1066. * example:
  1067. *
  1068. * [':D', 'some-other-dir/grin.png']
  1069. *
  1070. * absolute paths are not supported unless emoticonDir is an empty string
  1071. *
  1072. * To add emoticons to an existing object from WhutBB.db.emoticons, see
  1073. * WhutBB.db.addEmoticons().
  1074. *
  1075. * shortcuts: (Object) [ default: WhutBB.db.shortcuts ]
  1076. * an object of objects that account for shotcut mapping, see "Keyboard Shortcuts"
  1077. * part of the documentation
  1078. *
  1079. * example:
  1080. * shortcuts: {
  1081. * ctrl: {
  1082. * i: 'i'
  1083. * },
  1084. * 'alt+ctrl': {
  1085. * x: 'shotcuts'
  1086. * }
  1087. * }
  1088. *
  1089. */
  1090. WhutBB.Settings = function Settings(options) {
  1091. var def = WhutBB.db.sites[':default']();
  1092.  
  1093. try {
  1094. this.name = options.name || def.name;
  1095. this.link = options.link || def.link;
  1096.  
  1097. this.beneath = !!options.beneath;
  1098. this.blueprint = options.blueprint || def.blueprint;
  1099. this.width = options.width || def.width;
  1100.  
  1101. this.emoticonDir = options.emoticonDir || def.emoticonDir;
  1102. this.emoticonMax = options.emoticonMax || def.emoticonMax;
  1103. this.emoticons = (typeof options.emoticons === 'string') ? WhutBB.db.emoticons[options.emoticons] : (options.emoticons || def.emoticons); // null emoticon
  1104.  
  1105. this.shortcuts = options.shortcuts || WhutBB.db.shortcuts;
  1106. } catch (e) {
  1107. dom.oEach(def, function (name, setting) {
  1108. this[name] = setting;
  1109. }, this);
  1110. }
  1111. this.validButtons = {};
  1112. };
  1113.  
  1114. /**
  1115. * Button
  1116. *
  1117. * Generic button class that encapsulates data from
  1118. * WhutBB.db.buttons objects and creates a button element
  1119. *
  1120. * Do not use the constructor directly, use Button.create instead!
  1121. */
  1122. WhutBB.Button = (function () {
  1123.  
  1124. function Button(name) {
  1125. this.name = name;
  1126. this.data = WhutBB.db.buttons[name];
  1127. }
  1128.  
  1129. /**
  1130. * Button.create returns a Button or a Null button
  1131. * All possible buttons located at WhutBB.db.buttons
  1132. */
  1133. Button.create = function (button) {
  1134. if (WhutBB.db.buttons[button]) {
  1135. return new Button(button);
  1136. }
  1137. return Button.Null;
  1138. };
  1139.  
  1140. /**
  1141. * Creates a button element and also validates it
  1142. */
  1143. Button.prototype.make = function () {
  1144. this.validate();
  1145. return dom.dom('button', {
  1146. className: 'whutbbutton',
  1147. name: this.name,
  1148. title: this.data.title,
  1149. attr: {
  1150. type: 'button',
  1151. 'data-type': this.data.type || 'button'
  1152. }
  1153. }, dom.dom('span', {
  1154. className: 'icon-' + this.data.icon,
  1155. txt: this.data.display || this.name,
  1156. attr: {
  1157. 'data-txt': this.data.display || this.name,
  1158. 'data-toggle': this.data.toggle || ''
  1159. }
  1160. }));
  1161. };
  1162.  
  1163. /**
  1164. * Validates a button by adding it to WhutBB.config.validButtons
  1165. */
  1166. Button.prototype.validate = function () {
  1167. WhutBB.config.validButtons[this.name] = true;
  1168. return this;
  1169. };
  1170.  
  1171. /**
  1172. * Space creates a single-spaced text node.
  1173. *
  1174. * Both Space and Null objects are intended to mimic Buttons
  1175. * without using any real inheritance
  1176. */
  1177. Button.Space = {
  1178. make: function () {
  1179. return document.createTextNode(' ');
  1180. },
  1181. validate: function () {
  1182. return this;
  1183. },
  1184. data: {}
  1185. };
  1186.  
  1187. /**
  1188. * Null creates a simple text node.
  1189. * It's used when there is no real button in the db.
  1190. */
  1191. Button.Null = {
  1192. make: function () {
  1193. return document.createTextNode('');
  1194. },
  1195. validate: function () {
  1196. return this;
  1197. },
  1198. data: {}
  1199. };
  1200.  
  1201. Button.emoticon = function (emoticonData) {
  1202. return dom.dom('img', {
  1203. title: emoticonData[0],
  1204. alt: emoticonData[0],
  1205. src: WhutBB.config.emoticonDir + emoticonData[1],
  1206. attr: {
  1207. 'data-type': 'emoticon'
  1208. }
  1209. });
  1210. };
  1211.  
  1212. Button.emoticonLoader = function () {
  1213. return dom.dom('div', {
  1214. className: 'emoticonLoader',
  1215. name: 'emoticonLoader',
  1216. txt: 'View all emoticons.',
  1217. title: 'Loads all emoticons.',
  1218. attr: {
  1219. 'data-type': -2
  1220. }
  1221. });
  1222. };
  1223.  
  1224. return Button;
  1225.  
  1226. }());
  1227.  
  1228. /**
  1229. * Panel Class
  1230. * Generates all the panels used in the script.
  1231. *
  1232. * A panel is an element intended to be within a WhutBBInstance.wrap div.
  1233. *
  1234. * eg:
  1235. * { div (WhutBBInstance.wrap)
  1236. * [ wbb link panel ]
  1237. * [ buttons panel ]
  1238. * [ settings panel* ]
  1239. * [ shortcuts panel* ]
  1240. * }
  1241. *
  1242. * *Global panels
  1243. *
  1244. * Use Panel.factory, instead of new Panel().
  1245. *
  1246. * Global/public panels are static and part of the Panel object,
  1247. * not a WhutBB instance. They are typically transient, meaning that
  1248. * they appear in different WhutBB.wraps depending on the toggle state
  1249. *
  1250. * For example, emoticons are appended to WBB instace 1 when its
  1251. * emoticon button is clicked, but once WBB instace 2's emoticon button
  1252. * is clicked, the emoticon panel will be appended to WBB 2's wrap.
  1253. *
  1254. * This aliviates the need to generate each panel separately for every instance.
  1255. * This means that if there are 100s of emoticons, they will only be created once
  1256. * and moved around as needed, instead of creating 100s of emoticons per instance.
  1257. *
  1258. * Private panels are stored in the Panel.set object.
  1259. * Once panels are initially created within Panel.construct(),
  1260. * private panels can be copied to (copyTo) a WhutBB instance.
  1261. *
  1262. * The only two private panels are Button and Link, because they
  1263. * are not meant to be transient. Buttons are needed at every instance.
  1264. *
  1265. */
  1266. WhutBB.Panel = (function () {
  1267.  
  1268. /**
  1269. * An element is part of the instance
  1270. */
  1271. function Panel(element) {
  1272. this.element = element;
  1273. }
  1274.  
  1275. /**
  1276. * A set of private panels
  1277. */
  1278. Panel.set = {};
  1279.  
  1280. /**
  1281. * A set of global panels
  1282. */
  1283. Panel.global = {};
  1284.  
  1285. /**
  1286. * Panel.factory creates both global and private panels
  1287. *
  1288. * @param name for the panel
  1289. * @param element to encapsulate
  1290. * @param priv true for private panels, otherwise global
  1291. */
  1292. Panel.factory = function (name, element, priv) {
  1293. if (priv) {
  1294. if (!Panel.set[name]) {
  1295. Panel.set[name] = new Panel(element);
  1296. }
  1297. return Panel.set[name];
  1298. }
  1299. if (!Panel.global[name]) {
  1300. Panel.global[name] = new Panel(element);
  1301. }
  1302. return Panel.global[name];
  1303. };
  1304.  
  1305. /**
  1306. * Creates and initializes every necessary panel
  1307. */
  1308. Panel.construct = function () {
  1309. Panel.factory('link', dom.dom('div', {className: 'wbblink' + (WhutBB.user.settings.link ? '' : ' wbbhide') },
  1310. dom.dom('a', {href: WhutBB.config.link, title: 'Version 3.0', txt: 'WhutBBCode?'})), true);
  1311. Panel.factory('button', dom.dom('div', {className: 'wbbbuttons'}), true);
  1312.  
  1313. // Global Panels
  1314. Panel.factory('shortcut', dom.dom('ul', {className: 'wbbshortcut wbbhide'}));
  1315. Panel.factory('emoticon', dom.dom('div', {className: 'wbbemot wbbhide'}));
  1316. Panel.factory('settings', dom.dom('ul', {className: 'wbbset wbbhide'}, null, document.body));
  1317. Panel.factory('console', dom.dom('div', {className: 'wbbcon', txt: ''}));
  1318. Panel.attach.fill();
  1319. };
  1320.  
  1321. /**
  1322. * Copies private panels to a WhutBB Instance
  1323. */
  1324. Panel.copyTo = function (wbbInst) {
  1325. wbbInst.panels = {};
  1326. dom.oEach(Panel.set, function (name, panel) {
  1327. wbbInst.panels[name] = panel.element.cloneNode(true);
  1328. wbbInst.wrap.appendChild(wbbInst.panels[name]);
  1329. });
  1330. };
  1331.  
  1332. /**
  1333. * Prints a message to the console
  1334. */
  1335. Panel.message = function (text, time) {
  1336. var el = WhutBB.Panel.global.console.element;
  1337. el.innerHTML = text;
  1338. window.setTimeout(function () {
  1339. el.innerHTML = '';
  1340. }, isNaN(+time) ? 2500 : time);
  1341. };
  1342.  
  1343. Panel.attach = {
  1344. fill: function () {
  1345. // fills the panels appropriertly
  1346. this.buttons();
  1347. this.emoticons(-1, Math.min(WhutBB.config.emoticons.length,
  1348. WhutBB.config.emoticonMax));
  1349. this.settings();
  1350. this.shortcuts();
  1351. },
  1352. buttons: function () {
  1353. var f = document.createDocumentFragment();
  1354. dom.aEach(WhutBB.config.blueprint, function (set) {
  1355. dom.aEach(set, function (name) {
  1356. f.appendChild(WhutBB.Button.create(name).make());
  1357. });
  1358. f.appendChild(WhutBB.Button.Space.make());
  1359. });
  1360. Panel.set.button.element.appendChild(f);
  1361. f = null;
  1362. },
  1363. emoticons: function (i, max) {
  1364. var f = document.createDocumentFragment();
  1365. while (++i < max) {
  1366. f.appendChild(WhutBB.Button.emoticon(WhutBB.config.emoticons[i]));
  1367. }
  1368. // attach the div that loads all emoticons if required
  1369. if (max !== WhutBB.config.emoticons.length
  1370. && WhutBB.config.emoticons.length > WhutBB.config.emoticonMax) {
  1371. f.appendChild(WhutBB.Button.emoticonLoader());
  1372. }
  1373. Panel.global.emoticon.element.appendChild(f);
  1374. f = null;
  1375. },
  1376. settings: function () {
  1377. var list = [];
  1378. dom.oEach(WhutBB.user.options, function (name, data) {
  1379. list.push('<li><label title="' + data.title + '" ><input type="checkbox" data-setting="true" name="' + name + '" '
  1380. + (WhutBB.user.settings[name] ? 'checked="checked" ' : '') + '/>' + data.txt + '</label></li>');
  1381. });
  1382. Panel.global.settings.element.innerHTML = list.join('');
  1383. Panel.global.settings.element.appendChild(Panel.global.console.element);
  1384. },
  1385. shortcuts: function () {
  1386. dom.oEach(WhutBB.config.shortcuts, function (key, shortcuts) {
  1387. dom.oEach(shortcuts, function (letter) {
  1388. if (WhutBB.config.validButtons[shortcuts[letter]]) { // Checks if the site uses this button
  1389. Panel.global.shortcut.element.appendChild(dom.dom('li', {
  1390. innerHTML: [
  1391. key.toUpperCase(),
  1392. '+',
  1393. letter.toUpperCase(),
  1394. '<br>',
  1395. WhutBB.db.buttons[shortcuts[letter]].title
  1396. ].join('')
  1397. }));
  1398. }
  1399. });
  1400. });
  1401. }
  1402. };
  1403.  
  1404. return Panel;
  1405.  
  1406. }());
  1407.  
  1408. /**
  1409. * Tag Class
  1410. * Creates a tag of given name
  1411. *
  1412. * Use Tag.get(), not new Tag()!
  1413. * Tag.get() uses lazy loading, and stores all new
  1414. * tags within Tags.tags[]
  1415. *
  1416. * A tag's type generates the appropriate parsing
  1417. * All tags parse as a two-index array
  1418. *
  1419. * If a tag does not require an endpoint (matching tag),
  1420. * an empty string is required
  1421. *
  1422. * ['[tag]', '[/tag]']
  1423. * ['open', '']
  1424. * ['', 'close']
  1425. *
  1426. * Example, insert a tag directly into a textarea
  1427. * bTag = Tag.get('b');
  1428. * bTag.insertTo(someTextarea);
  1429. *
  1430. * PS: Note the use of WhutBB.box within insertTo().
  1431. */
  1432. WhutBB.Tag = (function () {
  1433.  
  1434. function Tag(text) {
  1435. Tag.tags[text] = this;
  1436. this.button = WhutBB.db.buttons[text];
  1437. this.tag = this.button.tag || text;
  1438. }
  1439.  
  1440. // Stores new Tags
  1441. Tag.tags = {};
  1442.  
  1443. /**
  1444. * Gets a tag by a name.
  1445. * Finds a tag in the tags object or creates a new tag.
  1446. * Returns an update()'d tag
  1447. */
  1448. Tag.get = function (name) {
  1449. if (WhutBB.db.buttons[name]) {
  1450. return (Tag.tags[name] || new Tag(name)).update();
  1451. }
  1452. };
  1453.  
  1454. /**
  1455. * Each button has a type which is used as the parsing method
  1456. */
  1457. Tag.types = {
  1458. 0: function () { // Basic tag [tag][/tag]
  1459. return ['[' + this.tag + ']', '[/' + this.tag + ']'];
  1460. },
  1461. 1: function () { // [tag=option][/tag]
  1462. return ['[' + this.tag + '=' + this.option + ']', '[/' + this.tag + ']'];
  1463. },
  1464. 2: function () { // [tag=]
  1465. return ['[' + this.tag + '=', ']'];
  1466. },
  1467. 3: function () { // List [*] or [#]
  1468. var j = [], li = WhutBB.box.range.data.selection.split('\n');
  1469.  
  1470. if (li.length > 1) {
  1471. dom.aEach(li, function (item) {
  1472. j.push('[' + this.tag + ']' + item);
  1473. }, this);
  1474. WhutBB.box.range.data.selection = j.join('\n');
  1475. return ['', ''];
  1476. }
  1477.  
  1478. return ['[' + this.tag + ']', ''];
  1479. },
  1480. 4: function () { // used for custom tags
  1481. if (typeof this.tag === 'string') {
  1482. return [this.tag, this.tag];
  1483. }
  1484. return [this.tag[0], this.tag[1]];
  1485. }
  1486. };
  1487.  
  1488. Tag.prototype.toString = function () {
  1489. return [this.tag, this.option, this.type].join(' ');
  1490. };
  1491.  
  1492. Tag.prototype.insertTo = function (textarea) {
  1493. WhutBB.box.select(textarea).insert(this.parse());
  1494. };
  1495.  
  1496. /**
  1497. * Parse uses some JavaScript magic to get the function
  1498. * based on the tag type, and call it with _this_ tag's
  1499. * instance
  1500. */
  1501. Tag.prototype.parse = function () {
  1502. return Tag.types[this.type].call(this);
  1503. };
  1504.  
  1505. Tag.prototype.findOption = function () {
  1506. // console.log('find option');
  1507. return this.button.type === 1 && this.optionText();
  1508. };
  1509.  
  1510. Tag.prototype.defaultText = function () {
  1511. return this.button.placeholder || this.button.val || '';
  1512. };
  1513.  
  1514. Tag.prototype.placeholderText = function () {
  1515. return this.button.placeholder ? ('\n(Default text [' + this.button.placeholder + '] will be removed automatically.)') : '';
  1516. };
  1517.  
  1518. Tag.prototype.optionText = function () {
  1519. if (!WhutBB.e.macro && WhutBB.user.settings.prompt && this.button.noPrompt !== true) {
  1520. this.option = window.prompt(this.button.prompt + this.placeholderText(), this.defaultText());
  1521. } else {
  1522. this.option = this.defaultText();
  1523. }
  1524. if (this.option === this.button.placeholder || this.option === '') {
  1525. this.option = false;
  1526. }
  1527. return true;
  1528. };
  1529.  
  1530. Tag.prototype.findType = function () {
  1531. return this.option === false ? 0 : this.button.type || 0;
  1532. };
  1533.  
  1534. Tag.prototype.update = function () {
  1535. this.findOption();
  1536. this.type = this.findType();
  1537. return this;
  1538. };
  1539.  
  1540. return Tag;
  1541.  
  1542. }());
  1543.  
  1544. WhutBB.init();
  1545. WhutBB.factory();
  1546. }());