script.js 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554
  1. var smf_formSubmitted = false;
  2. var lastKeepAliveCheck = new Date().getTime();
  3. var smf_editorArray = new Array();
  4. // Some very basic browser detection - from Mozilla's sniffer page.
  5. var ua = navigator.userAgent.toLowerCase();
  6. var is_opera = ua.indexOf('opera') != -1;
  7. var is_opera5 = ua.indexOf('opera/5') != -1 || ua.indexOf('opera 5') != -1;
  8. var is_opera6 = ua.indexOf('opera/6') != -1 || ua.indexOf('opera 6') != -1;
  9. var is_opera7 = ua.indexOf('opera/7') != -1 || ua.indexOf('opera 7') != -1;
  10. var is_opera8 = ua.indexOf('opera/8') != -1 || ua.indexOf('opera 8') != -1;
  11. var is_opera9 = ua.indexOf('opera/9') != -1 || ua.indexOf('opera 9') != -1;
  12. var is_opera95 = ua.indexOf('opera/9.5') != -1 || ua.indexOf('opera 9.5') != -1;
  13. var is_opera96 = ua.indexOf('opera/9.6') != -1 || ua.indexOf('opera 9.6') != -1;
  14. var is_opera10 = (ua.indexOf('opera/9.8') != -1 || ua.indexOf('opera 9.8') != -1 || ua.indexOf('opera/10.') != -1 || ua.indexOf('opera 10.') != -1) || ua.indexOf('version/10.') != -1;
  15. var is_opera95up = is_opera95 || is_opera96 || is_opera10;
  16. var is_ff = (ua.indexOf('firefox') != -1 || ua.indexOf('iceweasel') != -1 || ua.indexOf('icecat') != -1 || ua.indexOf('shiretoko') != -1 || ua.indexOf('minefield') != -1) && !is_opera;
  17. var is_gecko = ua.indexOf('gecko') != -1 && !is_opera;
  18. var is_chrome = ua.indexOf('chrome') != -1;
  19. var is_safari = ua.indexOf('applewebkit') != -1 && !is_chrome;
  20. var is_webkit = ua.indexOf('applewebkit') != -1;
  21. var is_ie = ua.indexOf('msie') != -1 && !is_opera;
  22. var is_ie4 = is_ie && ua.indexOf('msie 4') != -1;
  23. var is_ie5 = is_ie && ua.indexOf('msie 5') != -1;
  24. var is_ie50 = is_ie && ua.indexOf('msie 5.0') != -1;
  25. var is_ie55 = is_ie && ua.indexOf('msie 5.5') != -1;
  26. var is_ie5up = is_ie && !is_ie4;
  27. var is_ie6 = is_ie && ua.indexOf('msie 6') != -1;
  28. var is_ie6up = is_ie5up && !is_ie55 && !is_ie5;
  29. var is_ie6down = is_ie6 || is_ie5 || is_ie4;
  30. var is_ie7 = is_ie && ua.indexOf('msie 7') != -1;
  31. var is_ie7up = is_ie6up && !is_ie6;
  32. var is_ie7down = is_ie7 || is_ie6 || is_ie5 || is_ie4;
  33. var is_ie8 = is_ie && ua.indexOf('msie 8') != -1;
  34. var is_ie8up = is_ie8 && !is_ie7down;
  35. var is_iphone = ua.indexOf('iphone') != -1 || ua.indexOf('ipod') != -1;
  36. var is_android = ua.indexOf('android') != -1;
  37. var ajax_indicator_ele = null;
  38. // Define document.getElementById for Internet Explorer 4.
  39. if (!('getElementById' in document) && 'all' in document)
  40. document.getElementById = function (sId) {
  41. return document.all[sId];
  42. }
  43. // Define XMLHttpRequest for IE 5 and above. (don't bother for IE 4 :/.... works in Opera 7.6 and Safari 1.2!)
  44. else if (!('XMLHttpRequest' in window) && 'ActiveXObject' in window)
  45. window.XMLHttpRequest = function () {
  46. return new ActiveXObject(is_ie5 ? 'Microsoft.XMLHTTP' : 'MSXML2.XMLHTTP');
  47. };
  48. // Ensure the getElementsByTagName exists.
  49. if (!'getElementsByTagName' in document && 'all' in document)
  50. document.getElementsByTagName = function (sName) {
  51. return document.all.tags[sName];
  52. }
  53. // Some older versions of Mozilla don't have this, for some reason.
  54. if (!('forms' in document))
  55. document.forms = document.getElementsByTagName('form');
  56. // Load an XML document using XMLHttpRequest.
  57. function getXMLDocument(sUrl, funcCallback)
  58. {
  59. if (!window.XMLHttpRequest)
  60. return null;
  61. var oMyDoc = new XMLHttpRequest();
  62. var bAsync = typeof(funcCallback) != 'undefined';
  63. var oCaller = this;
  64. if (bAsync)
  65. {
  66. oMyDoc.onreadystatechange = function () {
  67. if (oMyDoc.readyState != 4)
  68. return;
  69. if (oMyDoc.responseXML != null && oMyDoc.status == 200)
  70. {
  71. if (funcCallback.call)
  72. {
  73. funcCallback.call(oCaller, oMyDoc.responseXML);
  74. }
  75. // A primitive substitute for the call method to support IE 5.0.
  76. else
  77. {
  78. oCaller.tmpMethod = funcCallback;
  79. oCaller.tmpMethod(oMyDoc.responseXML);
  80. delete oCaller.tmpMethod;
  81. }
  82. }
  83. };
  84. }
  85. oMyDoc.open('GET', sUrl, bAsync);
  86. oMyDoc.send(null);
  87. return oMyDoc;
  88. }
  89. // Send a post form to the server using XMLHttpRequest.
  90. function sendXMLDocument(sUrl, sContent, funcCallback)
  91. {
  92. if (!window.XMLHttpRequest)
  93. return false;
  94. var oSendDoc = new window.XMLHttpRequest();
  95. var oCaller = this;
  96. if (typeof(funcCallback) != 'undefined')
  97. {
  98. oSendDoc.onreadystatechange = function () {
  99. if (oSendDoc.readyState != 4)
  100. return;
  101. if (oSendDoc.responseXML != null && oSendDoc.status == 200)
  102. funcCallback.call(oCaller, oSendDoc.responseXML);
  103. else
  104. funcCallback.call(oCaller, false);
  105. };
  106. }
  107. oSendDoc.open('POST', sUrl, true);
  108. if ('setRequestHeader' in oSendDoc)
  109. oSendDoc.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
  110. oSendDoc.send(sContent);
  111. return true;
  112. }
  113. // A property we'll be needing for php_to8bit.
  114. String.prototype.oCharsetConversion = {
  115. from: '',
  116. to: ''
  117. };
  118. // Convert a string to an 8 bit representation (like in PHP).
  119. String.prototype.php_to8bit = function ()
  120. {
  121. if (smf_charset == 'UTF-8')
  122. {
  123. var n, sReturn = '';
  124. for (var i = 0, iTextLen = this.length; i < iTextLen; i++)
  125. {
  126. n = this.charCodeAt(i);
  127. if (n < 128)
  128. sReturn += String.fromCharCode(n)
  129. else if (n < 2048)
  130. sReturn += String.fromCharCode(192 | n >> 6) + String.fromCharCode(128 | n & 63);
  131. else if (n < 65536)
  132. sReturn += String.fromCharCode(224 | n >> 12) + String.fromCharCode(128 | n >> 6 & 63) + String.fromCharCode(128 | n & 63);
  133. else
  134. sReturn += String.fromCharCode(240 | n >> 18) + String.fromCharCode(128 | n >> 12 & 63) + String.fromCharCode(128 | n >> 6 & 63) + String.fromCharCode(128 | n & 63);
  135. }
  136. return sReturn;
  137. }
  138. else if (this.oCharsetConversion.from.length == 0)
  139. {
  140. switch (smf_charset)
  141. {
  142. case 'ISO-8859-1':
  143. this.oCharsetConversion = {
  144. from: '\xa0-\xff',
  145. to: '\xa0-\xff'
  146. };
  147. break;
  148. case 'ISO-8859-2':
  149. this.oCharsetConversion = {
  150. from: '\xa0\u0104\u02d8\u0141\xa4\u013d\u015a\xa7\xa8\u0160\u015e\u0164\u0179\xad\u017d\u017b\xb0\u0105\u02db\u0142\xb4\u013e\u015b\u02c7\xb8\u0161\u015f\u0165\u017a\u02dd\u017e\u017c\u0154\xc1\xc2\u0102\xc4\u0139\u0106\xc7\u010c\xc9\u0118\xcb\u011a\xcd\xce\u010e\u0110\u0143\u0147\xd3\xd4\u0150\xd6\xd7\u0158\u016e\xda\u0170\xdc\xdd\u0162\xdf\u0155\xe1\xe2\u0103\xe4\u013a\u0107\xe7\u010d\xe9\u0119\xeb\u011b\xed\xee\u010f\u0111\u0144\u0148\xf3\xf4\u0151\xf6\xf7\u0159\u016f\xfa\u0171\xfc\xfd\u0163\u02d9',
  151. to: '\xa0-\xff'
  152. };
  153. break;
  154. case 'ISO-8859-5':
  155. this.oCharsetConversion = {
  156. from: '\xa0\u0401-\u040c\xad\u040e-\u044f\u2116\u0451-\u045c\xa7\u045e\u045f',
  157. to: '\xa0-\xff'
  158. };
  159. break;
  160. case 'ISO-8859-9':
  161. this.oCharsetConversion = {
  162. from: '\xa0-\xcf\u011e\xd1-\xdc\u0130\u015e\xdf-\xef\u011f\xf1-\xfc\u0131\u015f\xff',
  163. to: '\xa0-\xff'
  164. };
  165. break;
  166. case 'ISO-8859-15':
  167. this.oCharsetConversion = {
  168. from: '\xa0-\xa3\u20ac\xa5\u0160\xa7\u0161\xa9-\xb3\u017d\xb5-\xb7\u017e\xb9-\xbb\u0152\u0153\u0178\xbf-\xff',
  169. to: '\xa0-\xff'
  170. };
  171. break;
  172. case 'tis-620':
  173. this.oCharsetConversion = {
  174. from: '\u20ac\u2026\u2018\u2019\u201c\u201d\u2022\u2013\u2014\xa0\u0e01-\u0e3a\u0e3f-\u0e5b',
  175. to: '\x80\x85\x91-\x97\xa0-\xda\xdf-\xfb'
  176. };
  177. break;
  178. case 'windows-1251':
  179. this.oCharsetConversion = {
  180. from: '\u0402\u0403\u201a\u0453\u201e\u2026\u2020\u2021\u20ac\u2030\u0409\u2039\u040a\u040c\u040b\u040f\u0452\u2018\u2019\u201c\u201d\u2022\u2013\u2014\u2122\u0459\u203a\u045a\u045c\u045b\u045f\xa0\u040e\u045e\u0408\xa4\u0490\xa6\xa7\u0401\xa9\u0404\xab-\xae\u0407\xb0\xb1\u0406\u0456\u0491\xb5-\xb7\u0451\u2116\u0454\xbb\u0458\u0405\u0455\u0457\u0410-\u044f',
  181. to: '\x80-\x97\x99-\xff'
  182. };
  183. break;
  184. case 'windows-1253':
  185. this.oCharsetConversion = {
  186. from: '\u20ac\u201a\u0192\u201e\u2026\u2020\u2021\u2030\u2039\u2018\u2019\u201c\u201d\u2022\u2013\u2014\u2122\u203a\xa0\u0385\u0386\xa3-\xa9\xab-\xae\u2015\xb0-\xb3\u0384\xb5-\xb7\u0388-\u038a\xbb\u038c\xbd\u038e-\u03a1\u03a3-\u03ce',
  187. to: '\x80\x82-\x87\x89\x8b\x91-\x97\x99\x9b\xa0-\xa9\xab-\xd1\xd3-\xfe'
  188. };
  189. break;
  190. case 'windows-1255':
  191. this.oCharsetConversion = {
  192. from: '\u20ac\u201a\u0192\u201e\u2026\u2020\u2021\u02c6\u2030\u2039\u2018\u2019\u201c\u201d\u2022\u2013\u2014\u02dc\u2122\u203a\xa0-\xa3\u20aa\xa5-\xa9\xd7\xab-\xb9\xf7\xbb-\xbf\u05b0-\u05b9\u05bb-\u05c3\u05f0-\u05f4\u05d0-\u05ea\u200e\u200f',
  193. to: '\x80\x82-\x89\x8b\x91-\x99\x9b\xa0-\xc9\xcb-\xd8\xe0-\xfa\xfd\xfe'
  194. };
  195. break;
  196. case 'windows-1256':
  197. this.oCharsetConversion = {
  198. from: '\u20ac\u067e\u201a\u0192\u201e\u2026\u2020\u2021\u02c6\u2030\u0679\u2039\u0152\u0686\u0698\u0688\u06af\u2018\u2019\u201c\u201d\u2022\u2013\u2014\u06a9\u2122\u0691\u203a\u0153\u200c\u200d\u06ba\xa0\u060c\xa2-\xa9\u06be\xab-\xb9\u061b\xbb-\xbe\u061f\u06c1\u0621-\u0636\xd7\u0637-\u063a\u0640-\u0643\xe0\u0644\xe2\u0645-\u0648\xe7-\xeb\u0649\u064a\xee\xef\u064b-\u064e\xf4\u064f\u0650\xf7\u0651\xf9\u0652\xfb\xfc\u200e\u200f\u06d2',
  199. to: '\x80-\xff'
  200. };
  201. break;
  202. default:
  203. this.oCharsetConversion = {
  204. from: '',
  205. to: ''
  206. };
  207. break;
  208. }
  209. var funcExpandString = function (sSearch) {
  210. var sInsert = '';
  211. for (var i = sSearch.charCodeAt(0), n = sSearch.charCodeAt(2); i <= n; i++)
  212. sInsert += String.fromCharCode(i);
  213. return sInsert;
  214. };
  215. this.oCharsetConversion.from = this.oCharsetConversion.from.replace(/.\-./g, funcExpandString);
  216. this.oCharsetConversion.to = this.oCharsetConversion.to.replace(/.\-./g, funcExpandString);
  217. }
  218. var sReturn = '', iOffsetFrom = 0;
  219. for (var i = 0, n = this.length; i < n; i++)
  220. {
  221. iOffsetFrom = this.oCharsetConversion.from.indexOf(this.charAt(i));
  222. sReturn += iOffsetFrom > -1 ? this.oCharsetConversion.to.charAt(iOffsetFrom) : (this.charCodeAt(i) > 127 ? '&#' + this.charCodeAt(i) + ';' : this.charAt(i));
  223. }
  224. return sReturn
  225. }
  226. // Character-level replacement function.
  227. String.prototype.php_strtr = function (sFrom, sTo)
  228. {
  229. return this.replace(new RegExp('[' + sFrom + ']', 'g'), function (sMatch) {
  230. return sTo.charAt(sFrom.indexOf(sMatch));
  231. });
  232. }
  233. // Simulate PHP's strtolower (in SOME cases PHP uses ISO-8859-1 case folding).
  234. String.prototype.php_strtolower = function ()
  235. {
  236. return typeof(smf_iso_case_folding) == 'boolean' && smf_iso_case_folding == true ? this.php_strtr(
  237. 'ABCDEFGHIJKLMNOPQRSTUVWXYZ\x8a\x8c\x8e\x9f\xc0\xc1\xc2\xc3\xc4\xc5\xc6\xc7\xc8\xc9\xca\xcb\xcc\xcd\xce\xcf\xd0\xd1\xd2\xd3\xd4\xd5\xd6\xd7\xd8\xd9\xda\xdb\xdc\xdd\xde',
  238. 'abcdefghijklmnopqrstuvwxyz\x9a\x9c\x9e\xff\xe0\xe1\xe2\xe3\xe4\xe5\xe6\xe7\xe8\xe9\xea\xeb\xec\xed\xee\xef\xf0\xf1\xf2\xf3\xf4\xf5\xf6\xf7\xf8\xf9\xfa\xfb\xfc\xfd\xfe'
  239. ) : this.php_strtr('ABCDEFGHIJKLMNOPQRSTUVWXYZ', 'abcdefghijklmnopqrstuvwxyz');
  240. }
  241. String.prototype.php_urlencode = function()
  242. {
  243. return escape(this).replace(/\+/g, '%2b').replace('*', '%2a').replace('/', '%2f').replace('@', '%40');
  244. }
  245. String.prototype.php_htmlspecialchars = function()
  246. {
  247. return this.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
  248. }
  249. String.prototype.php_unhtmlspecialchars = function()
  250. {
  251. return this.replace(/&quot;/g, '"').replace(/&gt;/g, '>').replace(/&lt;/g, '<').replace(/&amp;/g, '&');
  252. }
  253. String.prototype.php_addslashes = function()
  254. {
  255. return this.replace(/\\/g, '\\\\').replace(/'/g, '\\\'');
  256. }
  257. String.prototype._replaceEntities = function(sInput, sDummy, sNum)
  258. {
  259. return String.fromCharCode(parseInt(sNum));
  260. }
  261. String.prototype.removeEntities = function()
  262. {
  263. return this.replace(/&(amp;)?#(\d+);/g, this._replaceEntities);
  264. }
  265. String.prototype.easyReplace = function (oReplacements)
  266. {
  267. var sResult = this;
  268. for (var sSearch in oReplacements)
  269. sResult = sResult.replace(new RegExp('%' + sSearch + '%', 'g'), oReplacements[sSearch]);
  270. return sResult;
  271. }
  272. // Open a new window.
  273. function reqWin(desktopURL, alternateWidth, alternateHeight, noScrollbars)
  274. {
  275. if ((alternateWidth && self.screen.availWidth * 0.8 < alternateWidth) || (alternateHeight && self.screen.availHeight * 0.8 < alternateHeight))
  276. {
  277. noScrollbars = false;
  278. alternateWidth = Math.min(alternateWidth, self.screen.availWidth * 0.8);
  279. alternateHeight = Math.min(alternateHeight, self.screen.availHeight * 0.8);
  280. }
  281. else
  282. noScrollbars = typeof(noScrollbars) == 'boolean' && noScrollbars == true;
  283. window.open(desktopURL, 'requested_popup', 'toolbar=no,location=no,status=no,menubar=no,scrollbars=' + (noScrollbars ? 'no' : 'yes') + ',width=' + (alternateWidth ? alternateWidth : 480) + ',height=' + (alternateHeight ? alternateHeight : 220) + ',resizable=no');
  284. // Return false so the click won't follow the link ;).
  285. return false;
  286. }
  287. // Remember the current position.
  288. function storeCaret(oTextHandle)
  289. {
  290. // Only bother if it will be useful.
  291. if ('createTextRange' in oTextHandle)
  292. oTextHandle.caretPos = document.selection.createRange().duplicate();
  293. }
  294. // Replaces the currently selected text with the passed text.
  295. function replaceText(text, oTextHandle)
  296. {
  297. // Attempt to create a text range (IE).
  298. if ('caretPos' in oTextHandle && 'createTextRange' in oTextHandle)
  299. {
  300. var caretPos = oTextHandle.caretPos;
  301. caretPos.text = caretPos.text.charAt(caretPos.text.length - 1) == ' ' ? text + ' ' : text;
  302. caretPos.select();
  303. }
  304. // Mozilla text range replace.
  305. else if ('selectionStart' in oTextHandle)
  306. {
  307. var begin = oTextHandle.value.substr(0, oTextHandle.selectionStart);
  308. var end = oTextHandle.value.substr(oTextHandle.selectionEnd);
  309. var scrollPos = oTextHandle.scrollTop;
  310. oTextHandle.value = begin + text + end;
  311. if (oTextHandle.setSelectionRange)
  312. {
  313. oTextHandle.focus();
  314. var goForward = is_opera ? text.match(/\n/g).length : 0;
  315. oTextHandle.setSelectionRange(begin.length + text.length + goForward, begin.length + text.length + goForward);
  316. }
  317. oTextHandle.scrollTop = scrollPos;
  318. }
  319. // Just put it on the end.
  320. else
  321. {
  322. oTextHandle.value += text;
  323. oTextHandle.focus(oTextHandle.value.length - 1);
  324. }
  325. }
  326. // Surrounds the selected text with text1 and text2.
  327. function surroundText(text1, text2, oTextHandle)
  328. {
  329. // Can a text range be created?
  330. if ('caretPos' in oTextHandle && 'createTextRange' in oTextHandle)
  331. {
  332. var caretPos = oTextHandle.caretPos, temp_length = caretPos.text.length;
  333. caretPos.text = caretPos.text.charAt(caretPos.text.length - 1) == ' ' ? text1 + caretPos.text + text2 + ' ' : text1 + caretPos.text + text2;
  334. if (temp_length == 0)
  335. {
  336. caretPos.moveStart('character', -text2.length);
  337. caretPos.moveEnd('character', -text2.length);
  338. caretPos.select();
  339. }
  340. else
  341. oTextHandle.focus(caretPos);
  342. }
  343. // Mozilla text range wrap.
  344. else if ('selectionStart' in oTextHandle)
  345. {
  346. var begin = oTextHandle.value.substr(0, oTextHandle.selectionStart);
  347. var selection = oTextHandle.value.substr(oTextHandle.selectionStart, oTextHandle.selectionEnd - oTextHandle.selectionStart);
  348. var end = oTextHandle.value.substr(oTextHandle.selectionEnd);
  349. var newCursorPos = oTextHandle.selectionStart;
  350. var scrollPos = oTextHandle.scrollTop;
  351. oTextHandle.value = begin + text1 + selection + text2 + end;
  352. if (oTextHandle.setSelectionRange)
  353. {
  354. var goForward = is_opera ? text1.match(/\n/g).length : 0, goForwardAll = is_opera ? (text1 + text2).match(/\n/g).length : 0;
  355. if (selection.length == 0)
  356. oTextHandle.setSelectionRange(newCursorPos + text1.length + goForward, newCursorPos + text1.length + goForward);
  357. else
  358. oTextHandle.setSelectionRange(newCursorPos, newCursorPos + text1.length + selection.length + text2.length + goForwardAll);
  359. oTextHandle.focus();
  360. }
  361. oTextHandle.scrollTop = scrollPos;
  362. }
  363. // Just put them on the end, then.
  364. else
  365. {
  366. oTextHandle.value += text1 + text2;
  367. oTextHandle.focus(oTextHandle.value.length - 1);
  368. }
  369. }
  370. // Checks if the passed input's value is nothing.
  371. function isEmptyText(theField)
  372. {
  373. // Copy the value so changes can be made..
  374. var theValue = theField.value;
  375. // Strip whitespace off the left side.
  376. while (theValue.length > 0 && (theValue.charAt(0) == ' ' || theValue.charAt(0) == '\t'))
  377. theValue = theValue.substring(1, theValue.length);
  378. // Strip whitespace off the right side.
  379. while (theValue.length > 0 && (theValue.charAt(theValue.length - 1) == ' ' || theValue.charAt(theValue.length - 1) == '\t'))
  380. theValue = theValue.substring(0, theValue.length - 1);
  381. if (theValue == '')
  382. return true;
  383. else
  384. return false;
  385. }
  386. // Only allow form submission ONCE.
  387. function submitonce(theform)
  388. {
  389. smf_formSubmitted = true;
  390. // If there are any editors warn them submit is coming!
  391. for (var i = 0; i < smf_editorArray.length; i++)
  392. smf_editorArray[i].doSubmit();
  393. }
  394. function submitThisOnce(oControl)
  395. {
  396. // Hateful, hateful fix for Safari 1.3 beta.
  397. if (is_safari)
  398. return !smf_formSubmitted;
  399. // oControl might also be a form.
  400. var oForm = 'form' in oControl ? oControl.form : oControl;
  401. var aTextareas = oForm.getElementsByTagName('textarea');
  402. for (var i = 0, n = aTextareas.length; i < n; i++)
  403. aTextareas[i].readOnly = true;
  404. return !smf_formSubmitted;
  405. }
  406. // Deprecated, as innerHTML is supported everywhere.
  407. function setInnerHTML(oElement, sToValue)
  408. {
  409. oElement.innerHTML = sToValue;
  410. }
  411. function getInnerHTML(oElement)
  412. {
  413. return oElement.innerHTML;
  414. }
  415. // Set the "outer" HTML of an element.
  416. function setOuterHTML(oElement, sToValue)
  417. {
  418. if ('outerHTML' in oElement)
  419. oElement.outerHTML = sToValue;
  420. else
  421. {
  422. var range = document.createRange();
  423. range.setStartBefore(oElement);
  424. oElement.parentNode.replaceChild(range.createContextualFragment(sToValue), oElement);
  425. }
  426. }
  427. // Checks for variable in theArray.
  428. function in_array(variable, theArray)
  429. {
  430. for (var i in theArray)
  431. if (theArray[i] == variable)
  432. return true;
  433. return false;
  434. }
  435. // Checks for variable in theArray.
  436. function array_search(variable, theArray)
  437. {
  438. for (var i in theArray)
  439. if (theArray[i] == variable)
  440. return i;
  441. return null;
  442. }
  443. // Find a specific radio button in its group and select it.
  444. function selectRadioByName(oRadioGroup, sName)
  445. {
  446. if (!('length' in oRadioGroup))
  447. return oRadioGroup.checked = true;
  448. for (var i = 0, n = oRadioGroup.length; i < n; i++)
  449. if (oRadioGroup[i].value == sName)
  450. return oRadioGroup[i].checked = true;
  451. return false;
  452. }
  453. // Invert all checkboxes at once by clicking a single checkbox.
  454. function invertAll(oInvertCheckbox, oForm, sMask, bIgnoreDisabled)
  455. {
  456. for (var i = 0; i < oForm.length; i++)
  457. {
  458. if (!('name' in oForm[i]) || (typeof(sMask) == 'string' && oForm[i].name.substr(0, sMask.length) != sMask && oForm[i].id.substr(0, sMask.length) != sMask))
  459. continue;
  460. if (!oForm[i].disabled || (typeof(bIgnoreDisabled) == 'boolean' && bIgnoreDisabled))
  461. oForm[i].checked = oInvertCheckbox.checked;
  462. }
  463. }
  464. // Keep the session alive - always!
  465. var lastKeepAliveCheck = new Date().getTime();
  466. function smf_sessionKeepAlive()
  467. {
  468. var curTime = new Date().getTime();
  469. // Prevent a Firefox bug from hammering the server.
  470. if (smf_scripturl && curTime - lastKeepAliveCheck > 900000)
  471. {
  472. var tempImage = new Image();
  473. tempImage.src = smf_prepareScriptUrl(smf_scripturl) + 'action=keepalive;time=' + curTime;
  474. lastKeepAliveCheck = curTime;
  475. }
  476. window.setTimeout('smf_sessionKeepAlive();', 1200000);
  477. }
  478. window.setTimeout('smf_sessionKeepAlive();', 1200000);
  479. // Set a theme option through javascript.
  480. function smf_setThemeOption(option, value, theme, cur_session_id, cur_session_var, additional_vars)
  481. {
  482. // Compatibility.
  483. if (cur_session_id == null)
  484. cur_session_id = smf_session_id;
  485. if (typeof(cur_session_var) == 'undefined')
  486. cur_session_var = 'sesc';
  487. if (additional_vars == null)
  488. additional_vars = '';
  489. var tempImage = new Image();
  490. tempImage.src = smf_prepareScriptUrl(smf_scripturl) + 'action=jsoption;var=' + option + ';val=' + value + ';' + cur_session_var + '=' + cur_session_id + additional_vars + (theme == null ? '' : '&th=' + theme) + ';time=' + (new Date().getTime());
  491. }
  492. function smf_avatarResize()
  493. {
  494. var possibleAvatars = document.getElementsByTagName('img');
  495. for (var i = 0; i < possibleAvatars.length; i++)
  496. {
  497. var tempAvatars = []; j = 0;
  498. if (possibleAvatars[i].className != 'avatar')
  499. continue;
  500. // Image.prototype.avatar = possibleAvatars[i];
  501. tempAvatars[j] = new Image();
  502. tempAvatars[j].avatar = possibleAvatars[i];
  503. tempAvatars[j].onload = function()
  504. {
  505. this.avatar.width = this.width;
  506. this.avatar.height = this.height;
  507. if (smf_avatarMaxWidth != 0 && this.width > smf_avatarMaxWidth)
  508. {
  509. this.avatar.height = (smf_avatarMaxWidth * this.height) / this.width;
  510. this.avatar.width = smf_avatarMaxWidth;
  511. }
  512. if (smf_avatarMaxHeight != 0 && this.avatar.height > smf_avatarMaxHeight)
  513. {
  514. this.avatar.width = (smf_avatarMaxHeight * this.avatar.width) / this.avatar.height;
  515. this.avatar.height = smf_avatarMaxHeight;
  516. }
  517. }
  518. tempAvatars[j].src = possibleAvatars[i].src;
  519. j++;
  520. }
  521. if (typeof(window_oldAvatarOnload) != 'undefined' && window_oldAvatarOnload)
  522. {
  523. window_oldAvatarOnload();
  524. window_oldAvatarOnload = null;
  525. }
  526. }
  527. function hashLoginPassword(doForm, cur_session_id)
  528. {
  529. // Compatibility.
  530. if (cur_session_id == null)
  531. cur_session_id = smf_session_id;
  532. if (typeof(hex_sha1) == 'undefined')
  533. return;
  534. // Are they using an email address?
  535. if (doForm.user.value.indexOf('@') != -1)
  536. return;
  537. // Unless the browser is Opera, the password will not save properly.
  538. if (!('opera' in window))
  539. doForm.passwrd.autocomplete = 'off';
  540. doForm.hash_passwrd.value = hex_sha1(hex_sha1(doForm.user.value.php_to8bit().php_strtolower() + doForm.passwrd.value.php_to8bit()) + cur_session_id);
  541. // It looks nicer to fill it with asterisks, but Firefox will try to save that.
  542. if (is_ff != -1)
  543. doForm.passwrd.value = '';
  544. else
  545. doForm.passwrd.value = doForm.passwrd.value.replace(/./g, '*');
  546. }
  547. function hashAdminPassword(doForm, username, cur_session_id)
  548. {
  549. // Compatibility.
  550. if (cur_session_id == null)
  551. cur_session_id = smf_session_id;
  552. if (typeof(hex_sha1) == 'undefined')
  553. return;
  554. doForm.admin_hash_pass.value = hex_sha1(hex_sha1(username.php_to8bit().php_strtolower() + doForm.admin_pass.value.php_to8bit()) + cur_session_id);
  555. doForm.admin_pass.value = doForm.admin_pass.value.replace(/./g, '*');
  556. }
  557. function hashModeratePassword(doForm, username, cur_session_id)
  558. {
  559. if (typeof(hex_sha1) == 'undefined')
  560. return;
  561. doForm.moderate_hash_pass.value = hex_sha1(hex_sha1(username.php_to8bit().php_strtolower() + doForm.moderate_pass.value.php_to8bit()) + cur_session_id);
  562. doForm.moderate_pass.value = doForm.moderate_pass.value.replace(/./g, '*');
  563. }
  564. // Shows the page numbers by clicking the dots (in compact view).
  565. function expandPages(spanNode, baseURL, firstPage, lastPage, perPage)
  566. {
  567. var replacement = '', i, oldLastPage = 0;
  568. var perPageLimit = 50;
  569. // The dots were bold, the page numbers are not (in most cases).
  570. spanNode.style.fontWeight = 'normal';
  571. spanNode.onclick = '';
  572. // Prevent too many pages to be loaded at once.
  573. if ((lastPage - firstPage) / perPage > perPageLimit)
  574. {
  575. oldLastPage = lastPage;
  576. lastPage = firstPage + perPageLimit * perPage;
  577. }
  578. // Calculate the new pages.
  579. for (i = firstPage; i < lastPage; i += perPage)
  580. replacement += '<a class="navPages" href="' + baseURL.replace(/%1\$d/, i).replace(/%%/g, '%') + '">' + (1 + i / perPage) + '</a> ';
  581. if (oldLastPage > 0)
  582. replacement += '<span style="font-weight: bold; cursor: ' + (is_ie && !is_ie6up ? 'hand' : 'pointer') + ';" onclick="expandPages(this, \'' + baseURL + '\', ' + lastPage + ', ' + oldLastPage + ', ' + perPage + ');"> ... </span> ';
  583. // Replace the dots by the new page links.
  584. setInnerHTML(spanNode, replacement);
  585. }
  586. function smc_preCacheImage(sSrc)
  587. {
  588. if (!('smc_aCachedImages' in window))
  589. window.smc_aCachedImages = [];
  590. if (!in_array(sSrc, window.smc_aCachedImages))
  591. {
  592. var oImage = new Image();
  593. oImage.src = sSrc;
  594. }
  595. }
  596. // *** smc_Cookie class.
  597. function smc_Cookie(oOptions)
  598. {
  599. this.opt = oOptions;
  600. this.oCookies = {};
  601. this.init();
  602. }
  603. smc_Cookie.prototype.init = function()
  604. {
  605. if ('cookie' in document && document.cookie != '')
  606. {
  607. var aCookieList = document.cookie.split(';');
  608. for (var i = 0, n = aCookieList.length; i < n; i++)
  609. {
  610. var aNameValuePair = aCookieList[i].split('=');
  611. this.oCookies[aNameValuePair[0].replace(/^\s+|\s+$/g, '')] = decodeURIComponent(aNameValuePair[1]);
  612. }
  613. }
  614. }
  615. smc_Cookie.prototype.get = function(sKey)
  616. {
  617. return sKey in this.oCookies ? this.oCookies[sKey] : null;
  618. }
  619. smc_Cookie.prototype.set = function(sKey, sValue)
  620. {
  621. document.cookie = sKey + '=' + encodeURIComponent(sValue);
  622. }
  623. // *** smc_Toggle class.
  624. function smc_Toggle(oOptions)
  625. {
  626. this.opt = oOptions;
  627. this.bCollapsed = false;
  628. this.oCookie = null;
  629. this.init();
  630. }
  631. smc_Toggle.prototype.init = function ()
  632. {
  633. // The master switch can disable this toggle fully.
  634. if ('bToggleEnabled' in this.opt && !this.opt.bToggleEnabled)
  635. return;
  636. // If cookies are enabled and they were set, override the initial state.
  637. if ('oCookieOptions' in this.opt && this.opt.oCookieOptions.bUseCookie)
  638. {
  639. // Initialize the cookie handler.
  640. this.oCookie = new smc_Cookie({});
  641. // Check if the cookie is set.
  642. var cookieValue = this.oCookie.get(this.opt.oCookieOptions.sCookieName)
  643. if (cookieValue != null)
  644. this.opt.bCurrentlyCollapsed = cookieValue == '1';
  645. }
  646. // If the init state is set to be collapsed, collapse it.
  647. if (this.opt.bCurrentlyCollapsed)
  648. this.changeState(true, true);
  649. // Initialize the images to be clickable.
  650. if ('aSwapImages' in this.opt)
  651. {
  652. for (var i = 0, n = this.opt.aSwapImages.length; i < n; i++)
  653. {
  654. var oImage = document.getElementById(this.opt.aSwapImages[i].sId);
  655. if (typeof(oImage) == 'object' && oImage != null)
  656. {
  657. // Display the image in case it was hidden.
  658. if (oImage.style.display == 'none')
  659. oImage.style.display = '';
  660. oImage.instanceRef = this;
  661. oImage.onclick = function () {
  662. this.instanceRef.toggle();
  663. this.blur();
  664. }
  665. oImage.style.cursor = 'pointer';
  666. // Preload the collapsed image.
  667. smc_preCacheImage(this.opt.aSwapImages[i].srcCollapsed);
  668. }
  669. }
  670. }
  671. // Initialize links.
  672. if ('aSwapLinks' in this.opt)
  673. {
  674. for (var i = 0, n = this.opt.aSwapLinks.length; i < n; i++)
  675. {
  676. var oLink = document.getElementById(this.opt.aSwapLinks[i].sId);
  677. if (typeof(oLink) == 'object' && oLink != null)
  678. {
  679. // Display the link in case it was hidden.
  680. if (oLink.style.display == 'none')
  681. oLink.style.display = '';
  682. oLink.instanceRef = this;
  683. oLink.onclick = function () {
  684. this.instanceRef.toggle();
  685. this.blur();
  686. return false;
  687. }
  688. }
  689. }
  690. }
  691. }
  692. // Collapse or expand the section.
  693. smc_Toggle.prototype.changeState = function(bCollapse, bInit)
  694. {
  695. // Default bInit to false.
  696. bInit = typeof(bInit) == 'undefined' ? false : true;
  697. // Handle custom function hook before collapse.
  698. if (!bInit && bCollapse && 'funcOnBeforeCollapse' in this.opt)
  699. {
  700. this.tmpMethod = this.opt.funcOnBeforeCollapse;
  701. this.tmpMethod();
  702. delete this.tmpMethod;
  703. }
  704. // Handle custom function hook before expand.
  705. else if (!bInit && !bCollapse && 'funcOnBeforeExpand' in this.opt)
  706. {
  707. this.tmpMethod = this.opt.funcOnBeforeExpand;
  708. this.tmpMethod();
  709. delete this.tmpMethod;
  710. }
  711. // Loop through all the images that need to be toggled.
  712. if ('aSwapImages' in this.opt)
  713. {
  714. for (var i = 0, n = this.opt.aSwapImages.length; i < n; i++)
  715. {
  716. var oImage = document.getElementById(this.opt.aSwapImages[i].sId);
  717. if (typeof(oImage) == 'object' && oImage != null)
  718. {
  719. // Only (re)load the image if it's changed.
  720. var sTargetSource = bCollapse ? this.opt.aSwapImages[i].srcCollapsed : this.opt.aSwapImages[i].srcExpanded;
  721. if (oImage.src != sTargetSource)
  722. oImage.src = sTargetSource;
  723. oImage.alt = oImage.title = bCollapse ? this.opt.aSwapImages[i].altCollapsed : this.opt.aSwapImages[i].altExpanded;
  724. }
  725. }
  726. }
  727. // Loop through all the links that need to be toggled.
  728. if ('aSwapLinks' in this.opt)
  729. {
  730. for (var i = 0, n = this.opt.aSwapLinks.length; i < n; i++)
  731. {
  732. var oLink = document.getElementById(this.opt.aSwapLinks[i].sId);
  733. if (typeof(oLink) == 'object' && oLink != null)
  734. setInnerHTML(oLink, bCollapse ? this.opt.aSwapLinks[i].msgCollapsed : this.opt.aSwapLinks[i].msgExpanded);
  735. }
  736. }
  737. // Now go through all the sections to be collapsed.
  738. for (var i = 0, n = this.opt.aSwappableContainers.length; i < n; i++)
  739. {
  740. if (this.opt.aSwappableContainers[i] == null)
  741. continue;
  742. var oContainer = document.getElementById(this.opt.aSwappableContainers[i]);
  743. if (typeof(oContainer) == 'object' && oContainer != null)
  744. oContainer.style.display = bCollapse ? 'none' : '';
  745. }
  746. // Update the new state.
  747. this.bCollapsed = bCollapse;
  748. // Update the cookie, if desired.
  749. if ('oCookieOptions' in this.opt && this.opt.oCookieOptions.bUseCookie)
  750. this.oCookie.set(this.opt.oCookieOptions.sCookieName, this.bCollapsed ? '1' : '0');
  751. if ('oThemeOptions' in this.opt && this.opt.oThemeOptions.bUseThemeSettings)
  752. smf_setThemeOption(this.opt.oThemeOptions.sOptionName, this.bCollapsed ? '1' : '0', 'sThemeId' in this.opt.oThemeOptions ? this.opt.oThemeOptions.sThemeId : null, smf_session_id, smf_session_var, 'sAdditionalVars' in this.opt.oThemeOptions ? this.opt.oThemeOptions.sAdditionalVars : null);
  753. }
  754. smc_Toggle.prototype.toggle = function()
  755. {
  756. // Change the state by reversing the current state.
  757. this.changeState(!this.bCollapsed);
  758. }
  759. function ajax_indicator(turn_on)
  760. {
  761. if (ajax_indicator_ele == null)
  762. {
  763. ajax_indicator_ele = document.getElementById('ajax_in_progress');
  764. if (ajax_indicator_ele == null && typeof(ajax_notification_text) != null)
  765. {
  766. create_ajax_indicator_ele();
  767. }
  768. }
  769. if (ajax_indicator_ele != null)
  770. {
  771. if (navigator.appName == 'Microsoft Internet Explorer' && !is_ie7up)
  772. {
  773. ajax_indicator_ele.style.position = 'absolute';
  774. ajax_indicator_ele.style.top = document.documentElement.scrollTop;
  775. }
  776. ajax_indicator_ele.style.display = turn_on ? 'block' : 'none';
  777. }
  778. }
  779. function create_ajax_indicator_ele()
  780. {
  781. // Create the div for the indicator.
  782. ajax_indicator_ele = document.createElement('div');
  783. // Set the id so it'll load the style properly.
  784. ajax_indicator_ele.id = 'ajax_in_progress';
  785. // Add the image in and link to turn it off.
  786. var cancel_link = document.createElement('a');
  787. cancel_link.href = 'javascript:ajax_indicator(false)';
  788. var cancel_img = document.createElement('img');
  789. cancel_img.src = smf_images_url + '/icons/quick_remove.gif';
  790. if (typeof(ajax_notification_cancel_text) != 'undefined')
  791. {
  792. cancel_img.alt = ajax_notification_cancel_text;
  793. cancel_img.title = ajax_notification_cancel_text;
  794. }
  795. // Add the cancel link and image to the indicator.
  796. cancel_link.appendChild(cancel_img);
  797. ajax_indicator_ele.appendChild(cancel_link);
  798. // Set the text. (Note: You MUST append here and not overwrite.)
  799. ajax_indicator_ele.innerHTML += ajax_notification_text;
  800. // Finally attach the element to the body.
  801. document.body.appendChild(ajax_indicator_ele);
  802. }
  803. function createEventListener(oTarget)
  804. {
  805. if (!('addEventListener' in oTarget))
  806. {
  807. if (oTarget.attachEvent)
  808. {
  809. oTarget.addEventListener = function (sEvent, funcHandler, bCapture) {
  810. oTarget.attachEvent('on' + sEvent, funcHandler);
  811. }
  812. oTarget.removeEventListener = function (sEvent, funcHandler, bCapture) {
  813. oTarget.detachEvent('on' + sEvent, funcHandler);
  814. }
  815. }
  816. else
  817. {
  818. oTarget.addEventListener = function (sEvent, funcHandler, bCapture) {
  819. oTarget['on' + sEvent] = funcHandler;
  820. }
  821. oTarget.removeEventListener = function (sEvent, funcHandler, bCapture) {
  822. oTarget['on' + sEvent] = null;
  823. }
  824. }
  825. }
  826. }
  827. // This function will retrieve the contents needed for the jump to boxes.
  828. function grabJumpToContent()
  829. {
  830. var oXMLDoc = getXMLDocument(smf_prepareScriptUrl(smf_scripturl) + 'action=xmlhttp;sa=jumpto;xml');
  831. var aBoardsAndCategories = new Array();
  832. ajax_indicator(true);
  833. if (oXMLDoc.responseXML)
  834. {
  835. var items = oXMLDoc.responseXML.getElementsByTagName('smf')[0].getElementsByTagName('item');
  836. for (var i = 0, n = items.length; i < n; i++)
  837. {
  838. aBoardsAndCategories[aBoardsAndCategories.length] = {
  839. id: parseInt(items[i].getAttribute('id')),
  840. isCategory: items[i].getAttribute('type') == 'category',
  841. name: items[i].firstChild.nodeValue.removeEntities(),
  842. is_current: false,
  843. childLevel: parseInt(items[i].getAttribute('childlevel'))
  844. }
  845. }
  846. }
  847. ajax_indicator(false);
  848. for (var i = 0, n = aJumpTo.length; i < n; i++)
  849. aJumpTo[i].fillSelect(aBoardsAndCategories);
  850. }
  851. // This'll contain all JumpTo objects on the page.
  852. var aJumpTo = new Array();
  853. // *** JumpTo class.
  854. function JumpTo(oJumpToOptions)
  855. {
  856. this.opt = oJumpToOptions;
  857. this.dropdownList = null;
  858. this.showSelect();
  859. }
  860. // Show the initial select box (onload). Method of the JumpTo class.
  861. JumpTo.prototype.showSelect = function ()
  862. {
  863. var sChildLevelPrefix = '';
  864. for (var i = this.opt.iCurBoardChildLevel; i > 0; i--)
  865. sChildLevelPrefix += this.opt.sBoardChildLevelIndicator;
  866. setInnerHTML(document.getElementById(this.opt.sContainerId), this.opt.sJumpToTemplate.replace(/%select_id%/, this.opt.sContainerId + '_select').replace(/%dropdown_list%/, '<select name="' + this.opt.sContainerId + '_select" id="' + this.opt.sContainerId + '_select" ' + ('implementation' in document ? '' : 'onmouseover="grabJumpToContent();" ') + ('onbeforeactivate' in document ? 'onbeforeactivate' : 'onfocus') + '="grabJumpToContent();"><option value="?board=' + this.opt.iCurBoardId + '.0">' + sChildLevelPrefix + this.opt.sBoardPrefix + this.opt.sCurBoardName.removeEntities() + '</option></select>&nbsp;<input type="button" value="' + this.opt.sGoButtonLabel + '" onclick="window.location.href = \'' + smf_prepareScriptUrl(smf_scripturl) + 'board=' + this.opt.iCurBoardId + '.0\';" />'));
  867. this.dropdownList = document.getElementById(this.opt.sContainerId + '_select');
  868. }
  869. // Fill the jump to box with entries. Method of the JumpTo class.
  870. JumpTo.prototype.fillSelect = function (aBoardsAndCategories)
  871. {
  872. var bIE5x = !('implementation' in document);
  873. var iIndexPointer = 0;
  874. // Create an option that'll be above and below the category.
  875. var oDashOption = document.createElement('option');
  876. oDashOption.appendChild(document.createTextNode(this.opt.sCatSeparator));
  877. oDashOption.disabled = 'disabled';
  878. oDashOption.value = '';
  879. // Reset the events and clear the list (IE5.x only).
  880. if (bIE5x)
  881. {
  882. this.dropdownList.onmouseover = null;
  883. this.dropdownList.remove(0);
  884. }
  885. if ('onbeforeactivate' in document)
  886. this.dropdownList.onbeforeactivate = null;
  887. else
  888. this.dropdownList.onfocus = null;
  889. // Create a document fragment that'll allowing inserting big parts at once.
  890. var oListFragment = bIE5x ? this.dropdownList : document.createDocumentFragment();
  891. // Loop through all items to be added.
  892. for (var i = 0, n = aBoardsAndCategories.length; i < n; i++)
  893. {
  894. var j, sChildLevelPrefix, oOption;
  895. // If we've reached the currently selected board add all items so far.
  896. if (!aBoardsAndCategories[i].isCategory && aBoardsAndCategories[i].id == this.opt.iCurBoardId)
  897. {
  898. if (bIE5x)
  899. iIndexPointer = this.dropdownList.options.length;
  900. else
  901. {
  902. this.dropdownList.insertBefore(oListFragment, this.dropdownList.options[0]);
  903. oListFragment = document.createDocumentFragment();
  904. continue;
  905. }
  906. }
  907. if (aBoardsAndCategories[i].isCategory)
  908. oListFragment.appendChild(oDashOption.cloneNode(true));
  909. else
  910. for (j = aBoardsAndCategories[i].childLevel, sChildLevelPrefix = ''; j > 0; j--)
  911. sChildLevelPrefix += this.opt.sBoardChildLevelIndicator;
  912. oOption = document.createElement('option');
  913. oOption.appendChild(document.createTextNode((aBoardsAndCategories[i].isCategory ? this.opt.sCatPrefix : sChildLevelPrefix + this.opt.sBoardPrefix) + aBoardsAndCategories[i].name));
  914. oOption.value = aBoardsAndCategories[i].isCategory ? '#c' + aBoardsAndCategories[i].id : '?board=' + aBoardsAndCategories[i].id + '.0';
  915. oListFragment.appendChild(oOption);
  916. if (aBoardsAndCategories[i].isCategory)
  917. oListFragment.appendChild(oDashOption.cloneNode(true));
  918. }
  919. // Add the remaining items after the currently selected item.
  920. this.dropdownList.appendChild(oListFragment);
  921. if (bIE5x)
  922. this.dropdownList.options[iIndexPointer].selected = true;
  923. // Internet Explorer needs this to keep the box dropped down.
  924. this.dropdownList.style.width = 'auto';
  925. this.dropdownList.focus();
  926. // Add an onchange action
  927. this.dropdownList.onchange = function() {
  928. if (this.selectedIndex > 0 && this.options[this.selectedIndex].value)
  929. window.location.href = smf_scripturl + this.options[this.selectedIndex].value.substr(smf_scripturl.indexOf('?') == -1 || this.options[this.selectedIndex].value.substr(0, 1) != '?' ? 0 : 1);
  930. }
  931. }
  932. // A global array containing all IconList objects.
  933. var aIconLists = new Array();
  934. // *** IconList object.
  935. function IconList(oOptions)
  936. {
  937. if (!window.XMLHttpRequest)
  938. return;
  939. this.opt = oOptions;
  940. this.bListLoaded = false;
  941. this.oContainerDiv = null;
  942. this.funcMousedownHandler = null;
  943. this.funcParent = this;
  944. this.iCurMessageId = 0;
  945. this.iCurTimeout = 0;
  946. // Add backwards compatibility with old themes.
  947. if (!('sSessionVar' in this.opt))
  948. this.opt.sSessionVar = 'sesc';
  949. this.initIcons();
  950. }
  951. // Replace all message icons by icons with hoverable and clickable div's.
  952. IconList.prototype.initIcons = function ()
  953. {
  954. for (var i = document.images.length - 1, iPrefixLength = this.opt.sIconIdPrefix.length; i >= 0; i--)
  955. if (document.images[i].id.substr(0, iPrefixLength) == this.opt.sIconIdPrefix)
  956. setOuterHTML(document.images[i], '<div title="' + this.opt.sLabelIconList + '" onclick="' + this.opt.sBackReference + '.openPopup(this, ' + document.images[i].id.substr(iPrefixLength) + ')" onmouseover="' + this.opt.sBackReference + '.onBoxHover(this, true)" onmouseout="' + this.opt.sBackReference + '.onBoxHover(this, false)" style="background: ' + this.opt.sBoxBackground + '; cursor: ' + (is_ie && !is_ie6up ? 'hand' : 'pointer') + '; padding: 3px; text-align: center;"><img src="' + document.images[i].src + '" alt="' + document.images[i].alt + '" id="' + document.images[i].id + '" style="margin: 0px; padding: ' + (is_ie ? '3px' : '3px 0px 3px 0px') + ';" /></div>');
  957. }
  958. // Event for the mouse hovering over the original icon.
  959. IconList.prototype.onBoxHover = function (oDiv, bMouseOver)
  960. {
  961. oDiv.style.border = bMouseOver ? this.opt.iBoxBorderWidthHover + 'px solid ' + this.opt.sBoxBorderColorHover : '';
  962. oDiv.style.background = bMouseOver ? this.opt.sBoxBackgroundHover : this.opt.sBoxBackground;
  963. oDiv.style.padding = bMouseOver ? (3 - this.opt.iBoxBorderWidthHover) + 'px' : '3px'
  964. }
  965. // Show the list of icons after the user clicked the original icon.
  966. IconList.prototype.openPopup = function (oDiv, iMessageId)
  967. {
  968. this.iCurMessageId = iMessageId;
  969. if (!this.bListLoaded && this.oContainerDiv == null)
  970. {
  971. // Create a container div.
  972. this.oContainerDiv = document.createElement('div');
  973. this.oContainerDiv.id = 'iconList';
  974. this.oContainerDiv.style.display = 'none';
  975. this.oContainerDiv.style.cursor = is_ie && !is_ie6up ? 'hand' : 'pointer';
  976. this.oContainerDiv.style.position = 'absolute';
  977. this.oContainerDiv.style.width = oDiv.offsetWidth + 'px';
  978. this.oContainerDiv.style.background = this.opt.sContainerBackground;
  979. this.oContainerDiv.style.border = this.opt.sContainerBorder;
  980. this.oContainerDiv.style.padding = '1px';
  981. this.oContainerDiv.style.textAlign = 'center';
  982. document.body.appendChild(this.oContainerDiv);
  983. // Start to fetch its contents.
  984. ajax_indicator(true);
  985. this.tmpMethod = getXMLDocument;
  986. this.tmpMethod(smf_prepareScriptUrl(this.opt.sScriptUrl) + 'action=xmlhttp;sa=messageicons;board=' + this.opt.iBoardId + ';xml', this.onIconsReceived);
  987. delete this.tmpMethod;
  988. createEventListener(document.body);
  989. }
  990. // Set the position of the container.
  991. var aPos = smf_itemPos(oDiv);
  992. if (is_ie50)
  993. aPos[1] += 4;
  994. this.oContainerDiv.style.top = (aPos[1] + oDiv.offsetHeight) + 'px';
  995. this.oContainerDiv.style.left = (aPos[0] - 1) + 'px';
  996. this.oClickedIcon = oDiv;
  997. if (this.bListLoaded)
  998. this.oContainerDiv.style.display = 'block';
  999. document.body.addEventListener('mousedown', this.onWindowMouseDown, false);
  1000. }
  1001. // Setup the list of icons once it is received through xmlHTTP.
  1002. IconList.prototype.onIconsReceived = function (oXMLDoc)
  1003. {
  1004. var icons = oXMLDoc.getElementsByTagName('smf')[0].getElementsByTagName('icon');
  1005. var sItems = '';
  1006. for (var i = 0, n = icons.length; i < n; i++)
  1007. sItems += '<div onmouseover="' + this.opt.sBackReference + '.onItemHover(this, true)" onmouseout="' + this.opt.sBackReference + '.onItemHover(this, false);" onmousedown="' + this.opt.sBackReference + '.onItemMouseDown(this, \'' + icons[i].getAttribute('value') + '\');" style="padding: 3px 0px 3px 0px; margin-left: auto; margin-right: auto; border: ' + this.opt.sItemBorder + '; background: ' + this.opt.sItemBackground + '"><img src="' + icons[i].getAttribute('url') + '" alt="' + icons[i].getAttribute('name') + '" title="' + icons[i].firstChild.nodeValue + '" /></div>';
  1008. setInnerHTML(this.oContainerDiv, sItems);
  1009. this.oContainerDiv.style.display = 'block';
  1010. this.bListLoaded = true;
  1011. if (is_ie)
  1012. this.oContainerDiv.style.width = this.oContainerDiv.clientWidth + 'px';
  1013. ajax_indicator(false);
  1014. }
  1015. // Event handler for hovering over the icons.
  1016. IconList.prototype.onItemHover = function (oDiv, bMouseOver)
  1017. {
  1018. oDiv.style.background = bMouseOver ? this.opt.sItemBackgroundHover : this.opt.sItemBackground;
  1019. oDiv.style.border = bMouseOver ? this.opt.sItemBorderHover : this.opt.sItemBorder;
  1020. if (this.iCurTimeout != 0)
  1021. window.clearTimeout(this.iCurTimeout);
  1022. if (bMouseOver)
  1023. this.onBoxHover(this.oClickedIcon, true);
  1024. else
  1025. this.iCurTimeout = window.setTimeout(this.opt.sBackReference + '.collapseList();', 500);
  1026. }
  1027. // Event handler for clicking on one of the icons.
  1028. IconList.prototype.onItemMouseDown = function (oDiv, sNewIcon)
  1029. {
  1030. if (this.iCurMessageId != 0)
  1031. {
  1032. ajax_indicator(true);
  1033. this.tmpMethod = getXMLDocument;
  1034. var oXMLDoc = this.tmpMethod(smf_prepareScriptUrl(this.opt.sScriptUrl) + 'action=jsmodify;topic=' + this.opt.iTopicId + ';msg=' + this.iCurMessageId + ';' + smf_session_var + '=' + smf_session_id + ';icon=' + sNewIcon + ';xml');
  1035. delete this.tmpMethod;
  1036. ajax_indicator(false);
  1037. var oMessage = oXMLDoc.responseXML.getElementsByTagName('smf')[0].getElementsByTagName('message')[0];
  1038. if (oMessage.getElementsByTagName('error').length == 0)
  1039. {
  1040. if (this.opt.bShowModify && oMessage.getElementsByTagName('modified').length != 0)
  1041. setInnerHTML(document.getElementById('modified_' + this.iCurMessageId), oMessage.getElementsByTagName('modified')[0].childNodes[0].nodeValue);
  1042. this.oClickedIcon.getElementsByTagName('img')[0].src = oDiv.getElementsByTagName('img')[0].src;
  1043. }
  1044. }
  1045. }
  1046. // Event handler for clicking outside the list (will make the list disappear).
  1047. IconList.prototype.onWindowMouseDown = function ()
  1048. {
  1049. for (var i = aIconLists.length - 1; i >= 0; i--)
  1050. {
  1051. aIconLists[i].funcParent.tmpMethod = aIconLists[i].collapseList;
  1052. aIconLists[i].funcParent.tmpMethod();
  1053. delete aIconLists[i].funcParent.tmpMethod;
  1054. }
  1055. }
  1056. // Collapse the list of icons.
  1057. IconList.prototype.collapseList = function()
  1058. {
  1059. this.onBoxHover(this.oClickedIcon, false);
  1060. this.oContainerDiv.style.display = 'none';
  1061. this.iCurMessageId = 0;
  1062. document.body.removeEventListener('mousedown', this.onWindowMouseDown, false);
  1063. }
  1064. // Handy shortcuts for getting the mouse position on the screen - only used for IE at the moment.
  1065. function smf_mousePose(oEvent)
  1066. {
  1067. var x = 0;
  1068. var y = 0;
  1069. if (oEvent.pageX)
  1070. {
  1071. y = oEvent.pageY;
  1072. x = oEvent.pageX;
  1073. }
  1074. else if (oEvent.clientX)
  1075. {
  1076. x = oEvent.clientX + (document.documentElement.scrollLeft ? document.documentElement.scrollLeft : document.body.scrollLeft);
  1077. y = oEvent.clientY + (document.documentElement.scrollTop ? document.documentElement.scrollTop : document.body.scrollTop);
  1078. }
  1079. return [x, y];
  1080. }
  1081. // Short function for finding the actual position of an item.
  1082. function smf_itemPos(itemHandle)
  1083. {
  1084. var itemX = 0;
  1085. var itemY = 0;
  1086. if ('offsetParent' in itemHandle)
  1087. {
  1088. itemX = itemHandle.offsetLeft;
  1089. itemY = itemHandle.offsetTop;
  1090. while (itemHandle.offsetParent && typeof(itemHandle.offsetParent) == 'object')
  1091. {
  1092. itemHandle = itemHandle.offsetParent;
  1093. itemX += itemHandle.offsetLeft;
  1094. itemY += itemHandle.offsetTop;
  1095. }
  1096. }
  1097. else if ('x' in itemHandle)
  1098. {
  1099. itemX = itemHandle.x;
  1100. itemY = itemHandle.y;
  1101. }
  1102. return [itemX, itemY];
  1103. }
  1104. // This function takes the script URL and prepares it to allow the query string to be appended to it.
  1105. function smf_prepareScriptUrl(sUrl)
  1106. {
  1107. return sUrl.indexOf('?') == -1 ? sUrl + '?' : sUrl + (sUrl.charAt(sUrl.length - 1) == '?' || sUrl.charAt(sUrl.length - 1) == '&' || sUrl.charAt(sUrl.length - 1) == ';' ? '' : ';');
  1108. }
  1109. var aOnloadEvents = new Array();
  1110. function addLoadEvent(fNewOnload)
  1111. {
  1112. // If there's no event set, just set this one
  1113. if (typeof(fNewOnload) == 'function' && (!('onload' in window) || typeof(window.onload) != 'function'))
  1114. window.onload = fNewOnload;
  1115. // If there's just one event, setup the array.
  1116. else if (aOnloadEvents.length == 0)
  1117. {
  1118. aOnloadEvents[0] = window.onload;
  1119. aOnloadEvents[1] = fNewOnload;
  1120. window.onload = function() {
  1121. for (var i = 0, n = aOnloadEvents.length; i < n; i++)
  1122. {
  1123. if (typeof(aOnloadEvents[i]) == 'function')
  1124. aOnloadEvents[i]();
  1125. else if (typeof(aOnloadEvents[i]) == 'string')
  1126. eval(aOnloadEvents[i]);
  1127. }
  1128. }
  1129. }
  1130. // This isn't the first event function, add it to the list.
  1131. else
  1132. aOnloadEvents[aOnloadEvents.length] = fNewOnload;
  1133. }
  1134. function smfFooterHighlight(element, value)
  1135. {
  1136. element.src = smf_images_url + '/' + (value ? 'h_' : '') + element.id + '.gif';
  1137. }
  1138. // Get the text in a code tag.
  1139. function smfSelectText(oCurElement, bActOnElement)
  1140. {
  1141. // The place we're looking for is one div up, and next door - if it's auto detect.
  1142. if (typeof(bActOnElement) == 'boolean' && bActOnElement)
  1143. var oCodeArea = document.getElementById(oCurElement);
  1144. else
  1145. var oCodeArea = oCurElement.parentNode.nextSibling;
  1146. if (typeof(oCodeArea) != 'object' || oCodeArea == null)
  1147. return false;
  1148. // Start off with my favourite, internet explorer.
  1149. if ('createTextRange' in document.body)
  1150. {
  1151. var oCurRange = document.body.createTextRange();
  1152. oCurRange.moveToElementText(oCodeArea);
  1153. oCurRange.select();
  1154. }
  1155. // Firefox at el.
  1156. else if (window.getSelection)
  1157. {
  1158. var oCurSelection = window.getSelection();
  1159. // Safari is special!
  1160. if (oCurSelection.setBaseAndExtent)
  1161. {
  1162. var oLastChild = oCodeArea.lastChild;
  1163. oCurSelection.setBaseAndExtent(oCodeArea, 0, oLastChild, 'innerText' in oLastChild ? oLastChild.innerText.length : oLastChild.textContent.length);
  1164. }
  1165. else
  1166. {
  1167. var curRange = document.createRange();
  1168. curRange.selectNodeContents(oCodeArea);
  1169. oCurSelection.removeAllRanges();
  1170. oCurSelection.addRange(curRange);
  1171. }
  1172. }
  1173. return false;
  1174. }
  1175. // A function needed to discern HTML entities from non-western characters.
  1176. function smc_saveEntities(sFormName, aElementNames, sMask)
  1177. {
  1178. if (typeof(sMask) == 'string')
  1179. {
  1180. for (var i = 0, n = document.forms[sFormName].elements.length; i < n; i++)
  1181. if (document.forms[sFormName].elements[i].id.substr(0, sMask.length) == sMask)
  1182. aElementNames[aElementNames.length] = document.forms[sFormName].elements[i].name;
  1183. }
  1184. for (var i = 0, n = aElementNames.length; i < n; i++)
  1185. {
  1186. if (aElementNames[i] in document.forms[sFormName])
  1187. document.forms[sFormName][aElementNames[i]].value = document.forms[sFormName][aElementNames[i]].value.replace(/&#/g, '&#38;#');
  1188. }
  1189. }
  1190. // A function used to clean the attachments on post page
  1191. function cleanFileInput(idElement)
  1192. {
  1193. // Simpler solutions work in Opera, IE, Safari and Chrome.
  1194. if (is_opera || is_ie || is_safari || is_chrome)
  1195. {
  1196. document.getElementById(idElement).outerHTML = document.getElementById(idElement).outerHTML;
  1197. }
  1198. // What else can we do? By the way, this doesn't work in Chrome and Mac's Safari.
  1199. else
  1200. {
  1201. document.getElementById(idElement).type = 'input';
  1202. document.getElementById(idElement).type = 'file';
  1203. }
  1204. }
  1205. function applyWindowClasses(oList)
  1206. {
  1207. var bAlternate = false;
  1208. oListItems = oList.getElementsByTagName("LI");
  1209. for (i = 0; i < oListItems.length; i++)
  1210. {
  1211. // Skip dummies.
  1212. if (oListItems[i].id == "")
  1213. continue;
  1214. oListItems[i].className = "windowbg" + (bAlternate ? "2" : "");
  1215. bAlternate = !bAlternate;
  1216. }
  1217. }
  1218. function reActivate()
  1219. {
  1220. document.forms.postmodify.message.readOnly = false;
  1221. }
  1222. // The actual message icon selector.
  1223. function showimage()
  1224. {
  1225. document.images.icons.src = icon_urls[document.forms.postmodify.icon.options[document.forms.postmodify.icon.selectedIndex].value];
  1226. }
  1227. function pollOptions()
  1228. {
  1229. var expire_time = document.getElementById('poll_expire');
  1230. if (isEmptyText(expire_time) || expire_time.value == 0)
  1231. {
  1232. document.forms.postmodify.poll_hide[2].disabled = true;
  1233. if (document.forms.postmodify.poll_hide[2].checked)
  1234. document.forms.postmodify.poll_hide[1].checked = true;
  1235. }
  1236. else
  1237. document.forms.postmodify.poll_hide[2].disabled = false;
  1238. }
  1239. function generateDays(offset)
  1240. {
  1241. // Work around JavaScript's lack of support for default values...
  1242. offset = typeof(offset) != 'undefined' ? offset : 0;
  1243. var days = 0, selected = 0;
  1244. var dayElement = document.getElementById("day" + offset), yearElement = document.getElementById("year" + offset), monthElement = document.getElementById("month" + offset);
  1245. monthLength[1] = 28;
  1246. if (yearElement.options[yearElement.selectedIndex].value % 4 == 0)
  1247. monthLength[1] = 29;
  1248. selected = dayElement.selectedIndex;
  1249. while (dayElement.options.length)
  1250. dayElement.options[0] = null;
  1251. days = monthLength[monthElement.value - 1];
  1252. for (i = 1; i <= days; i++)
  1253. dayElement.options[dayElement.length] = new Option(i, i);
  1254. if (selected < days)
  1255. dayElement.selectedIndex = selected;
  1256. }
  1257. function toggleLinked(form)
  1258. {
  1259. form.board.disabled = !form.link_to_board.checked;
  1260. }
  1261. function initSearch()
  1262. {
  1263. if (document.forms.searchform.search.value.indexOf("%u") != -1)
  1264. document.forms.searchform.search.value = unescape(document.forms.searchform.search.value);
  1265. }
  1266. function selectBoards(ids)
  1267. {
  1268. var toggle = true;
  1269. for (i = 0; i < ids.length; i++)
  1270. toggle = toggle & document.forms.searchform["brd" + ids[i]].checked;
  1271. for (i = 0; i < ids.length; i++)
  1272. document.forms.searchform["brd" + ids[i]].checked = !toggle;
  1273. }
  1274. function expandCollapseBoards()
  1275. {
  1276. var current = document.getElementById("searchBoardsExpand").style.display != "none";
  1277. document.getElementById("searchBoardsExpand").style.display = current ? "none" : "";
  1278. document.getElementById("expandBoardsIcon").src = smf_images_url + (current ? "/expand.gif" : "/collapse.gif");
  1279. }
  1280. function expandCollapseLabels()
  1281. {
  1282. var current = document.getElementById("searchLabelsExpand").style.display != "none";
  1283. document.getElementById("searchLabelsExpand").style.display = current ? "none" : "";
  1284. document.getElementById("expandLabelsIcon").src = smf_images_url + (current ? "/expand.gif" : "/collapse.gif");
  1285. }
  1286. function updateRuleDef(optNum)
  1287. {
  1288. if (document.getElementById("ruletype" + optNum).value == "gid")
  1289. {
  1290. document.getElementById("defdiv" + optNum).style.display = "none";
  1291. document.getElementById("defseldiv" + optNum).style.display = "";
  1292. }
  1293. else if (document.getElementById("ruletype" + optNum).value == "bud" || document.getElementById("ruletype" + optNum).value == "")
  1294. {
  1295. document.getElementById("defdiv" + optNum).style.display = "none";
  1296. document.getElementById("defseldiv" + optNum).style.display = "none";
  1297. }
  1298. else
  1299. {
  1300. document.getElementById("defdiv" + optNum).style.display = "";
  1301. document.getElementById("defseldiv" + optNum).style.display = "none";
  1302. }
  1303. }
  1304. function updateActionDef(optNum)
  1305. {
  1306. if (document.getElementById("acttype" + optNum).value == "lab")
  1307. {
  1308. document.getElementById("labdiv" + optNum).style.display = "";
  1309. }
  1310. else
  1311. {
  1312. document.getElementById("labdiv" + optNum).style.display = "none";
  1313. }
  1314. }