render_namumark.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583
  1. function render_namumark(target) {
  2. function get_today() {
  3. var today_data = new Date();
  4. return '' +
  5. String(today_data.getFullYear()) + '-' +
  6. String(today_data.getMonth() + 1) + '-' +
  7. String(today_data.getDate()) + ' ' +
  8. (today_data.getHours() < 10 ? '0' + String(today_data.getHours()) : String(today_data.getHours())) + ':' +
  9. (today_data.getMinutes() < 10 ? '0' + String(today_data.getMinutes()) : String(today_data.getMinutes())) + ':' +
  10. (today_data.getSeconds() < 10 ? '0' + String(today_data.getSeconds()) : String(today_data.getSeconds())) +
  11. '';
  12. }
  13. function get_link_state(link_data) {
  14. var xhr = new XMLHttpRequest();
  15. xhr.open("GET", "/api/w/" + encodeURIComponent(link_data[0]) + "?exist=1", true);
  16. xhr.send(null);
  17. xhr.onreadystatechange = function() {
  18. var i = 0;
  19. while(1) {
  20. if(document.getElementsByClassName(link_data[1])[i]) {
  21. if(this.readyState === 4 && this.status === 200) {
  22. if(JSON.parse(this.responseText)['exist'] !== '1') {
  23. document.getElementsByClassName(link_data[1])[i].id = "not_thing";
  24. } else {
  25. document.getElementsByClassName(link_data[1])[i].id = "";
  26. }
  27. } else {
  28. document.getElementsByClassName(link_data[1])[i].id = "not_thing";
  29. }
  30. i += 1;
  31. } else {
  32. break;
  33. }
  34. }
  35. }
  36. }
  37. function get_file_state(file_data) {
  38. var file_part = file_data[0].match(/^([^.]+)\.(.+)$/);
  39. if(file_part) {
  40. var file_name = file_part[1];
  41. var file_type = '.' + file_part[2];
  42. } else {
  43. var file_name = file_data;
  44. var file_type = '';
  45. }
  46. var xhr = new XMLHttpRequest();
  47. xhr.open("GET", "/api/sha224/" + encodeURIComponent(file_name), true);
  48. xhr.send(null);
  49. xhr.onreadystatechange = function() {
  50. if(this.readyState === 4 && this.status === 200) {
  51. var xhr = new XMLHttpRequest();
  52. xhr.open("GET", "/api/w/file:" + encodeURIComponent(file_data[0]) + "?exist=1", true);
  53. xhr.send(null);
  54. var img_src = JSON.parse(this.responseText)['data'];
  55. xhr.onreadystatechange = function() {
  56. if(this.readyState === 4 && this.status === 200) {
  57. if(JSON.parse(this.responseText)['exist'] !== '1') {
  58. document.getElementById(file_data[1]).innerHTML = '' +
  59. '<a href="/upload?name=' + encodeURIComponent(file_data[0]) + '" id="not_thing">' + file_data[0] + '</a>' +
  60. '';
  61. } else {
  62. document.getElementById(file_data[1]).innerHTML = '' +
  63. '<img src="/image/' + img_src + file_type + '">' +
  64. '';
  65. }
  66. } else {
  67. document.getElementById(file_data[1]).innerHTML = '' +
  68. '<a href="/upload?name=' + encodeURIComponent(file_data[0]) + '" id="not_thing">' + file_data[0] + '</a>' +
  69. '';
  70. }
  71. }
  72. }
  73. }
  74. }
  75. function divi_link(link_data) {
  76. var link_part = link_data.match(/^([^|]+)\|(.+)$/);
  77. if(link_part) {
  78. return [link_part[2], link_part[1]]
  79. } else {
  80. return [link_data, link_data]
  81. }
  82. }
  83. var data = '\n' + document.getElementById(target).innerHTML + '\n';
  84. var title = window.location.pathname.replace(/^\/w\//, '');
  85. console.log(title);
  86. var mid_num = 0;
  87. var mid_stack = 0;
  88. var mid_list = [];
  89. var html_number = 0;
  90. data = data.replace(/(?:{{{(?:((?:(?! |{{{|}}}|&lt;).)*) ?)|(}}}))/g, function(all, in_data) {
  91. if(all === '}}}') {
  92. if(mid_stack > 0) {
  93. mid_stack -= 1;
  94. }
  95. if(mid_stack > 0) {
  96. return all;
  97. } else {
  98. if(mid_num > 0) {
  99. mid_num -= 1;
  100. }
  101. if(!mid_list[mid_num]) {
  102. var return_data = '';
  103. } else if(mid_list[mid_num] === 'pre') {
  104. var return_data = '</code></pre>';
  105. } else {
  106. var return_data = '</' + mid_list[mid_num] + '>';
  107. }
  108. if(return_data !== '') {
  109. mid_list.splice(mid_num, 1);
  110. return return_data;
  111. } else {
  112. return all;
  113. }
  114. }
  115. } else {
  116. if(mid_stack > 0) {
  117. mid_stack += 1;
  118. return all;
  119. } else {
  120. mid_num += 1;
  121. if(in_data.match(/^(#|@|\+|\-)/) && !in_data.match(/^(#|@|\+|\-){2}|(#|@|\+|\-)\\\\/)) {
  122. if(in_data.match(/^((#|@)([0-9a-f-A-F]{3}){1,2})/)) {
  123. mid_list.push('span');
  124. if(in_data.match(/^#/)) {
  125. return '<span style="color: ' + in_data + ';">';
  126. } else {
  127. return '<span style="background: ' + in_data + ';">';
  128. }
  129. } else if(in_data.match(/^((#|@)(\w+))/)) {
  130. mid_list.push('span');
  131. if(in_data.match(/^#/)) {
  132. return '<span style="color: ' + in_data.replace(/^#/, '') + ';">';
  133. } else {
  134. return '<span style="background: ' + in_data.replace(/^@/, '') + ';">';
  135. }
  136. } else if(in_data.match(/^(\+|-)([1-5])/)) {
  137. mid_list.push('span');
  138. var font_size_data = in_data.match(/^(\+|-)([1-5])/);
  139. if(font_size_data[1] == '+') {
  140. font_size_data = String(Number(font_size_data[2]) * 20 + 100);
  141. } else {
  142. font_size_data = String(100 - Number(font_size_data[2]) * 10);
  143. }
  144. return '<span style="font-size: ' + font_size_data + '%;">'
  145. } else if(in_data.match(/#!wiki/i)) {
  146. mid_list.push('div');
  147. if(data.match(/{{{#!wiki style=((?:(?!\n).)+) *\n/i)) {
  148. return '<div id="wiki_div_before">';
  149. } else {
  150. return '<div id="wiki_div">'
  151. }
  152. } else if(in_data.match(/#!syntax/i)) {
  153. mid_list.push('pre');
  154. mid_stack += 1;
  155. return '<pre><code id="syntax_before">';
  156. } else if(in_data.match(/#!folding/i)) {
  157. mid_list.push('div');
  158. return '<div id="folding_before">';
  159. } else if(in_data.match(/#!html/i)) {
  160. mid_list.push('span');
  161. html_number += 1;
  162. return '<span id="html_render_contect_' + String(html_number) + '">';
  163. } else {
  164. mid_list.push('code');
  165. mid_stack += 1;
  166. return '<code>' + in_data;
  167. }
  168. } else {
  169. mid_list.push('code');
  170. mid_stack += 1;
  171. return '<code>' + in_data;
  172. }
  173. }
  174. }
  175. });
  176. console.log(mid_stack);
  177. console.log(mid_num);
  178. console.log(mid_list);
  179. data = data.replace(/<\/div> *\n/ig, '</div>');
  180. data = data.replace(/<div id="folding_before">((?:(?!\n).)+) *\n/ig, function(all, in_data) {
  181. return in_data + ' [+]<div id="folding">';
  182. });
  183. data = data.replace(/<pre><code id="syntax_before">((?:(?!\n).)+) *\n/ig, function(all, in_data) {
  184. return '<pre><code>';
  185. });
  186. data = data.replace(/<div id="wiki_div_before">style=((?:(?!\n).)+) *\n/ig, function(all, in_data) {
  187. return '<div style=' + in_data.replace(/&quot;/g, "\"").replace(/&#039;/g, "'") + ' id="wiki_div">';
  188. });
  189. var nowiki_num = 0;
  190. var nowiki_list = {};
  191. data = data.replace(/<code>(\n*((?:(?!<\/code>).)+\n*)+)<\/code>/g, function(all, in_data) {
  192. nowiki_num += 1;
  193. nowiki_list['nowiki_' + String(nowiki_num)] = in_data;
  194. return '<span id="nowiki_' + String(nowiki_num) + '"></span>';
  195. });
  196. var math_list = [];
  197. var math_num = 0;
  198. data = data.replace(/\[math\(((?:(?!\)]).)+)\)]/ig, function(all, in_data) {
  199. var math_data = in_data.replace(/&amp;/g, "&").replace(/&lt;/g, "<").replace(/&gt;/g, ">").replace(/&quot;/g, "\"").replace(/&#039;/g, "'");
  200. math_num += 1;
  201. math_list.push(['math_' + String(math_num), math_data]);
  202. return '<span id="math_' + String(math_num) + '"></span>';
  203. });
  204. data = data.replace(/~~((?:(?!~~).)+)~~/g, '<s>$1</s>');
  205. data = data.replace(/--((?:(?!--).)+)--/g, '<s>$1</s>');
  206. data = data.replace(/__((?:(?!__).)+)__/g, '<u>$1</u>');
  207. data = data.replace(/'''((?:(?!''').)+)'''/g, '<b>$1</b>');
  208. data = data.replace(/''((?:(?!'').)+)''/g, '<i>$1</i>');
  209. data = data.replace(/\^\^((?:(?!\^\^).)+)\^\^/g, '<sup>$1</sup>');
  210. data = data.replace(/,,((?:(?!,,).)+),,/g, '<sub>$1</sub>');
  211. data = data.replace(/\n( {1,})\* ([^\n]+)/g, function(all, margin_data, in_data) {
  212. return '<li style="margin-left: ' + String(margin_data.length * 20) + 'px;">' + in_data + '</li>'
  213. });
  214. data = data.replace(/\n( {1,})/g, function(all, margin_data) {
  215. return '\n<span style="margin-left: ' + String(margin_data.length * 10) + 'px"></span>'
  216. });
  217. var link_list = [];
  218. var file_list = [];
  219. var link_num = 0;
  220. var file_num = 0;
  221. var category = ''
  222. while(1) {
  223. if(data.match(/\[\[((?:(?!\[\[|]]).)+)]]/)) {
  224. data = data.replace(/\[\[((?:(?!\[\[|]]).)+)]]/, function(all, in_data) {
  225. if(in_data.match(/^(?:category|분류):/i)) {
  226. var back_data = in_data.replace(/^(?:category|분류):/i, '');
  227. var front_data = back_data;
  228. back_data = 'category:' + back_data.replace(/#blur$/, '');
  229. if(front_data.match(/#blur$/)) {
  230. front_data = '#blur';
  231. }
  232. link_list.push([back_data, 'link_' + String(link_num)]);
  233. link_num += 1;
  234. if(category === '') {
  235. category += '<div id="cate_all"><hr><div id="cate">Category : '
  236. }
  237. category += '<a class="link_' + String(link_num - 1) + '" href="' + encodeURIComponent(back_data) + '">' + front_data + '</a> | ';
  238. return '';
  239. } else if(in_data.match(/^(?:file|파일):/i)) {
  240. file_list.push([in_data.replace(/^(?:file|파일):/i, ''), 'file_' + String(file_num)]);
  241. file_num += 1;
  242. return '<span id="file_' + String(file_num - 1) + '"></span>';
  243. } else if(in_data.match(/^http(?:s)?:\/\//i)) {
  244. var link_part = divi_link(in_data);
  245. var front_data = link_part[0];
  246. var back_data = link_part[1];
  247. return '<a id="out_link" href="' + back_data + '">' + front_data + '</a>';
  248. } else {
  249. var link_part = divi_link(in_data);
  250. var front_data = link_part[0];
  251. var back_data = link_part[1];
  252. link_list.push([back_data, 'link_' + String(link_num)]);
  253. link_num += 1;
  254. return '<a class="link_' + String(link_num - 1) + '" href="/w/' + encodeURIComponent(back_data) + '">' + front_data + '</a>';
  255. }
  256. });
  257. } else {
  258. break;
  259. }
  260. }
  261. if(category !== '') {
  262. category = category.replace(/ \| $/, '') + '</div></div>'
  263. }
  264. data = data.replace(/\[([^(\]]+)\(((?:(?!\)]).)+)\)]/g, function(all, name, in_data) {
  265. if(name.match(/^youtube|kakaotv|nicovideo$/i)) {
  266. var video_code = in_data.match(/^([^,]+)/);
  267. if(video_code) {
  268. video_code = video_code[1];
  269. } else {
  270. video_code = 'test';
  271. }
  272. if(name === 'youtube') {
  273. var video_src = 'https://www.youtube.com/embed/' + video_code
  274. } else if(name === 'kakaotv') {
  275. var video_src = 'https://tv.kakao.com/embed/player/cliplink/' + video_code +'?service=kakao_tv'
  276. } else {
  277. var video_src = 'https://embed.nicovideo.jp/watch/' + video_code
  278. }
  279. var width_data = in_data.match(/, *width=([^,]+)/);
  280. if(width_data) {
  281. width_data = width_data[1];
  282. } else {
  283. width_data = '560';
  284. }
  285. var height_data = in_data.match(/, *height=([^,]+)/);
  286. if(height_data) {
  287. height_data = height_data[1];
  288. } else {
  289. height_data = '315';
  290. }
  291. return '' +
  292. '<iframe ' +
  293. 'width="' + width_data + '" ' +
  294. 'height="' + height_data + '" ' +
  295. 'src="' + video_src + '" ' +
  296. 'allowfullscreen>' +
  297. '</iframe>' +
  298. '';
  299. } else if(name.match(/^ruby$/i)) {
  300. var main_text = in_data.match(/^([^,]+)/);
  301. if(main_text) {
  302. main_text = main_text[1];
  303. } else {
  304. main_text = 'test';
  305. }
  306. var ruby_text = in_data.match(/, *ruby=([^,]+)/);
  307. if(ruby_text) {
  308. ruby_text = ruby_text[1];
  309. } else {
  310. ruby_text = 'test';
  311. }
  312. var color_text = in_data.match(/, *color=([^,]+)/);
  313. if(color_text) {
  314. color_text = 'color:' + color_text[1];
  315. } else {
  316. color_text = '';
  317. }
  318. return '' +
  319. '<ruby>' +
  320. main_text +
  321. '<rp>(</rp>' +
  322. '<rt>' +
  323. '<span style="' + color_text + '">' + ruby_text + '</span>' +
  324. '</rt>' +
  325. '<rp>)</rp>' +
  326. '</ruby>' +
  327. '';
  328. } else if(name.match(/^anchor$/i)) {
  329. return '<span id="' + in_data + '"></span>';
  330. } else {
  331. return all;
  332. }
  333. });
  334. var toc_array = [0, 0, 0, 0, 0, 0];
  335. var before_data = 0;
  336. var edit_number = 0;
  337. var toc_data = '<div id="toc"><span id="toc_title">TOC</span>\n\n'
  338. data = data.replace(/\n(={1,6}) ?([^\n]+) (?:={1,6})/g, function(all, num, in_data) {
  339. num = num.length;
  340. edit_number += 1;
  341. if(before_data > num) {
  342. var i = num;
  343. while(1) {
  344. if(i == 6) {
  345. break;
  346. }
  347. toc_array[i] = 0;
  348. i += 1;
  349. }
  350. }
  351. before_data = num;
  352. toc_array[num - 1] += 1;
  353. num = String(num);
  354. var toc_num = (toc_array.join('.') + '.').replace(/0\./g, '');
  355. if(!toc_num.match(/\./)) {
  356. toc_num += '0.';
  357. }
  358. toc_data += '' +
  359. '<span style="margin-left: ' + String(10 * (toc_num.length / 2) - 10) + 'px;">' +
  360. '<a href="#s-' + toc_num.replace(/\.$/, '') + '">' + toc_num + '</a> ' + in_data +
  361. '</span>' +
  362. '\n' +
  363. '';
  364. return '' +
  365. '\n' +
  366. '<h' + num + ' id="s-' + toc_num.replace(/\.$/, '') + '">' +
  367. '<a href="#toc">' + toc_num + '</a> ' + in_data +
  368. '<span style="font-size: 12px">' +
  369. '<a href="/edit/' + title + '?section=' + String(edit_number) + '">(Edit)</a>' +
  370. '</span>' +
  371. '</h' + num + '>' +
  372. '';
  373. });
  374. toc_data += '</div>';
  375. data = data.replace(/<\/h([0-9])>\n/g, '</h$1>');
  376. data = data.replace(/\[([^\]]+)\]/g, function(all, name) {
  377. if(name.match(/^br$/i)) {
  378. return '\n'
  379. } else if(name.match(/^목차|tableofcontents$/i)) {
  380. return toc_data;
  381. } else if(name.match(/^date|datetime$/i)) {
  382. return get_today();
  383. } else {
  384. return all;
  385. }
  386. });
  387. var ref_num = 0;
  388. var ref_data = '<hr><ul id="footnote_data">';
  389. var name_ref_data = {};
  390. while(1) {
  391. if(data.match(/(?:\[\*([^ \]]*)(?: ((?:(?!\[\*|\]).)+))?\]|\[(?:각주|footnote)])/)) {
  392. data = data.replace(/(?:\[\*([^ \]]*)(?: ((?:(?!\[\*|\]).)+))?\]|\[(?:각주|footnote)])/, function(all, name_data, in_data) {
  393. if(all.match(/^\[(?:각주|footnote)]$/i)) {
  394. var new_ref_data = ref_data;
  395. ref_data = '<hr><ul id="footnote_data">';
  396. return new_ref_data + '</ul>';
  397. } else {
  398. ref_num += 1;
  399. if(name_data) {
  400. if(in_data) {
  401. name_ref_data[name_data] = in_data;
  402. ref_data += '' +
  403. '<li>' +
  404. '<a id="fn-' + name_data + '" href="#rfn-' + String(ref_num) + '">(' + name_data + ')</a> ' + in_data + ''
  405. '</li>' +
  406. ''
  407. } else {
  408. ref_data += '' +
  409. '<li>' +
  410. '<a href="#rfn-' + String(ref_num) + '">(' + name_data + ')</a>' +
  411. '</li>' +
  412. ''
  413. }
  414. } else {
  415. ref_data += '' +
  416. '<li>' +
  417. '<a id="fn-' + String(ref_num) + '" href="#rfn-' + String(ref_num) + '">(' + String(ref_num) + ')</a> ' + in_data + ''
  418. '</li>' +
  419. ''
  420. }
  421. if(name_data) {
  422. return '' +
  423. '<sup>' +
  424. '<a href="#fn-' + name_data + '" id="rfn-' + String(ref_num) + '" title="' + name_ref_data[name_data].replace(/<([^>]*)>/g, '') + '">' +
  425. '(' + name_data + ')' +
  426. '</a>' +
  427. '</sup>' +
  428. '';
  429. } else {
  430. return '' +
  431. '<sup>' +
  432. '<a href="#fn-' + String(ref_num) + '" id="rfn-' + String(ref_num) + '" title="' + in_data.replace(/<([^>]*)>/g, '') + '">' +
  433. '(' + String(ref_num) + ')' +
  434. '</a>' +
  435. '</sup>' +
  436. '';
  437. }
  438. }
  439. });
  440. } else {
  441. break;
  442. }
  443. }
  444. if(ref_data !== '<hr><ul id="footnote_data">') {
  445. data += ref_data + '</ul>';
  446. }
  447. var i = 1;
  448. while(1) {
  449. if(nowiki_list['nowiki_' + String(i)]) {
  450. data = data.replace('<span id="nowiki_' + String(i) + '"></span>', '<code>' + nowiki_list['nowiki_' + String(i)] + '</code>');
  451. i += 1;
  452. } else {
  453. break;
  454. }
  455. }
  456. data = data.replace(/^(\n| )+/g, '');
  457. data = data.replace(/(\n| )+$/g, '');
  458. data = data.replace(/\n/g, '<br>');
  459. data = data.replace(/&amp;/g, '&');
  460. data += category;
  461. document.getElementById(target).innerHTML = data;
  462. i = 0;
  463. while(1) {
  464. if(math_list[i]) {
  465. try {
  466. katex.render(math_list[i][1], document.getElementById(math_list[i][0]));
  467. } catch {
  468. try {
  469. document.getElementById(math_list[i][0]).innerHTML = '<span style="color: red;">' + math_list[i][1] + '</span>';
  470. } catch {}
  471. }
  472. i += 1;
  473. } else {
  474. break;
  475. }
  476. }
  477. i = 0;
  478. while(1) {
  479. if(link_list[i]) {
  480. get_link_state(link_list[i]);
  481. i += 1;
  482. } else {
  483. break;
  484. }
  485. }
  486. i = 0;
  487. while(1) {
  488. if(file_list[i]) {
  489. get_file_state(file_list[i]);
  490. i += 1;
  491. } else {
  492. break;
  493. }
  494. }
  495. render_html("html_render_contect");
  496. }