render_onmark.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  1. // Tool
  2. function do_url_change(data) {
  3. return encodeURIComponent(data);
  4. }
  5. function do_link_change(data, data_nowiki, no_change = 0) {
  6. data = data.replace(/^:/, '');
  7. if(no_change === 0) {
  8. data = data.replace(/^사용자:/, 'user:');
  9. data = data.replace(/^분류:/, 'category:');
  10. data = data.replace(/^파일:/, 'file:');
  11. }
  12. var data_var = data.split('#');
  13. var link_main = data.replace(/#(.*)$/, '');
  14. var link_sub = data_var.length !== 1 ? ('#' + data_var[data_var.length - 1]) : '';
  15. return [link_main, link_sub];
  16. }
  17. function do_js_safe_change(data) {
  18. data = data.replace(/\\/g, '\\\\');
  19. data = data.replace(/"/g, '\\"');
  20. return data;
  21. }
  22. function do_math_try_insert(name_ob, data) {
  23. return '' +
  24. 'try {\n' +
  25. 'katex.render("' + data + '", document.getElementById(\"' + name_ob + '\"));\n' +
  26. '} catch {\n' +
  27. 'document.getElementById(\"' + name_ob + '\").innerHTML = "<span style=\'color: red;\'>' + data + '</span>";\n' +
  28. '}\n' +
  29. ''
  30. }
  31. function do_data_try_insert(name_ob, data) {
  32. return '' +
  33. 'if(document.getElementById("' + name_ob + '")) {\n' +
  34. 'document.getElementById("' + name_ob + '").innerHTML = "' + data + '";\n' +
  35. '}\n' +
  36. ''
  37. }
  38. function do_return_date() {
  39. var today_data = new Date();
  40. return '' +
  41. String(today_data.getFullYear()) + '-' +
  42. ((today_data.getMonth() + 1) < 10 ? '0' : '') + String(today_data.getMonth() + 1) + '-' +
  43. (today_data.getDate() < 10 ? '0' : '') + String(today_data.getDate()) + ' ' +
  44. (today_data.getHours() < 10 ? '0' : '') + String(today_data.getHours()) + ':' +
  45. (today_data.getMinutes() < 10 ? '0' : '') + String(today_data.getMinutes()) + ':' +
  46. (today_data.getSeconds() < 10 ? '0' : '') + String(today_data.getSeconds()) +
  47. '';
  48. }
  49. function do_xss_change(data) {
  50. data = data.replace(/&lt;/g, '<');
  51. data = data.replace(/&gt;/g, '>');
  52. data = data.replace(/&amp;/g, '&');
  53. return data;
  54. }
  55. // Sub
  56. function do_onmark_text_render(data) {
  57. data = data.replace(/'''((?:(?!''').)+)'''/g, '<b>$1</b>');
  58. data = data.replace(/''((?:(?!'').)+)''/g, '<i>$1</i>');
  59. data = data.replace(/__((?:(?!__).)+)__/g, '<u>$1</u>');
  60. data = data.replace(/\^\^((?:(?!\^\^).)+)\^\^/g, '<sup>$1</sup>');
  61. data = data.replace(/,,((?:(?!,,).)+),,/g, '<sub>$1</sub>');
  62. data = data.replace(/--((?:(?!--).)+)--/g, '<s>$1</s>');
  63. data = data.replace(/~~((?:(?!~~).)+)~~/g, '<s>$1</s>');
  64. return data;
  65. }
  66. function do_onmark_heading_render(data) {
  67. var heading_re = /\n(={1,6}) ?([^=]+) ?={1,6}\n/;
  68. var heading_level_all = [0, 0, 0, 0, 0, 0];
  69. var toc_data = '<div id="toc"><div id="toc_title">TOC</div>\n';
  70. while(1) {
  71. var heading_data = data.match(heading_re);
  72. if(!heading_data) {
  73. break;
  74. }
  75. var heading_level = heading_data[1].length;
  76. heading_level_all[heading_level - 1] += 1;
  77. var i = 6;
  78. while(i > heading_level - 1) {
  79. heading_level_all[i] = 0;
  80. i -= 1;
  81. }
  82. heading_level = String(heading_level);
  83. var heading_level_string = '';
  84. i = 0;
  85. while(i < 6) {
  86. if(heading_level_all[i] !== 0) {
  87. heading_level_string += String(heading_level_all[i]) + '.';
  88. }
  89. i += 1;
  90. }
  91. var heading_level_string_no_end = heading_level_string.replace(/\.$/, '');
  92. toc_data += '' +
  93. '<span style="margin-left: ' + String((heading_level_string.match(/\./g).length - 1) * 10) + 'px;">' +
  94. '<a href="#s-' + heading_level_string_no_end + '">' +
  95. heading_level_string + ' ' +
  96. '</a>' + heading_data[2] +
  97. '</span>' +
  98. '\n' +
  99. ''
  100. data = data.replace(heading_re,
  101. '<h' + heading_level + ' id="s-' + heading_level_string_no_end + '">' +
  102. '<a href="#toc">' + heading_level_string + '</a> ' + heading_data[2] +
  103. '</h' + heading_level + '>' +
  104. '\n'
  105. );
  106. }
  107. data = data.replace(/\[(?:toc|목차)\]/g, toc_data + '</div>');
  108. return data;
  109. }
  110. function do_onmark_link_render(data, data_js, name_doc, name_include, data_nowiki) {
  111. var num_link = 0;
  112. var category_data = '';
  113. var category_re = /^(분류|category):/i;
  114. var file_re = /^(파일|file):/i;
  115. data = data.replace(/\[\[(((?!\]\]).)+)\]\]/g, function(x, x_1) {
  116. var link_split = x_1.split('|');
  117. var link_real = link_split[0];
  118. var link_out = link_split[1] ? link_split[1] : link_split[0];
  119. var link_out_2 = link_split[1] ? link_split[1] : '';
  120. num_link += 1;
  121. var num_link_str = String(num_link - 1);
  122. if(link_real.match(file_re)) {
  123. var file_name = link_real.replace(file_re, '');
  124. console.log(file_name);
  125. return '';
  126. } else if(link_real.match(category_re)) {
  127. var category_link = link_real.replace(category_re, '');
  128. category_data = (category_data === '' ? '<div id="cate_all"><div id="cate">Category : ' : category_data);
  129. category_data += '' +
  130. '<a class="' + name_include + 'link_finder" ' +
  131. 'href="/w/category:' + do_url_change(category_link) + '">' +
  132. category_link +
  133. '</a> | ' +
  134. ''
  135. return '';
  136. } else if(link_real.match(/^http(s)?:\/\//)) {
  137. var i = 0;
  138. while(i < 2) {
  139. if(i === 0) {
  140. var var_link_type = 'href';
  141. } else {
  142. var var_link_type = 'title';
  143. }
  144. data_js += '' +
  145. 'document.getElementsByName("' + name_include + 'set_link_' + num_link_str + '")[0].' + var_link_type + ' = ' +
  146. '"' + do_js_safe_change(link_real) + '";' +
  147. '\n' +
  148. '';
  149. i += 1;
  150. }
  151. return '<a id="out_link" ' +
  152. 'name="' + name_include + 'set_link_' + num_link_str + '" ' +
  153. 'title=""' +
  154. 'href="">' + link_out + '</a>';
  155. } else {
  156. var i = 0;
  157. while(i < 2) {
  158. if(i === 0) {
  159. var link_data_var = do_link_change(link_real, data_nowiki);
  160. var link_main = link_data_var[0];
  161. var link_sub = link_data_var[1];
  162. var var_link_type = 'href';
  163. var var_link_data = '/w/' + do_url_change(link_main) + link_sub;
  164. } else {
  165. var var_link_type = 'title';
  166. var var_link_data = do_js_safe_change(link_real);
  167. }
  168. data_js += '' +
  169. 'document.getElementsByName("' + name_include + 'set_link_' + num_link_str + '")[0].' + var_link_type + ' = ' +
  170. '"' + var_link_data + '";' +
  171. '\n' +
  172. '';
  173. i += 1;
  174. }
  175. return '<a class="' + name_include + 'link_finder" ' +
  176. 'name="' + name_include + 'set_link_' + num_link_str + '" ' +
  177. 'title="" ' +
  178. 'href="">' + link_out + '</a>';
  179. }
  180. });
  181. data += (category_data === '' ? '' : (category_data.replace(/\| $/, '') + '</div></div>'));
  182. return [data, data_js];
  183. }
  184. function do_onmark_footnote_render(data, name_include) {
  185. var footnote_end_data = '';
  186. var footnote_all_data = {};
  187. var footnote_re = /(?:\[\*([^ \]]*)(?: ((?:(?!\]).)+))?\]|\[(footnote|각주)\])/;
  188. var i = 1;
  189. while(1) {
  190. var footnote_data = data.match(footnote_re);
  191. if(!footnote_data) {
  192. break;
  193. }
  194. if(!footnote_data[3]) {
  195. if(!footnote_data[2]) {
  196. var footnote_line_data = '';
  197. } else {
  198. var footnote_line_data = footnote_data[2];
  199. }
  200. if(!footnote_data[1]) {
  201. var footnote_name = String(i);
  202. } else {
  203. var footnote_name = footnote_data[1];
  204. }
  205. if(!footnote_all_data[footnote_name]) {
  206. footnote_all_data[footnote_name] = footnote_line_data;
  207. }
  208. footnote_line_data = footnote_all_data[footnote_name];
  209. footnote_end_data += '' +
  210. '<li>' +
  211. '<a href="javascript:do_open_foot(\'' + name_include + 'fn-' + String(i) + '\', 1);" ' +
  212. 'id="' + name_include + 'cfn-' + String(i) + '">' +
  213. '(' + footnote_name + ')' +
  214. '</a> <span id="' + name_include + 'fn-' + String(i) + '">' + footnote_line_data + '</span>' +
  215. '</li>' +
  216. '';
  217. data = data.replace(footnote_re, '' +
  218. '<sup>' +
  219. '<a href="javascript:do_open_foot(\'' + name_include + 'fn-' + String(i) + '\', 0);" ' +
  220. 'id="' + name_include + 'rfn-' + String(i) + '">' +
  221. '(' + footnote_name + ')' +
  222. '</a>' +
  223. '</sup><span id="' + name_include + 'dfn-' + String(i) + '"></span>' +
  224. '');
  225. i += 1;
  226. } else {
  227. if(footnote_end_data !== '') {
  228. data = data.replace(footnote_re, '<ul id="footnote_data">' + footnote_end_data + '</ul>');
  229. }
  230. footnote_end_data = '';
  231. }
  232. }
  233. if(footnote_end_data !== '') {
  234. data += '<ul id="footnote_data">' + footnote_end_data + '</ul>';
  235. }
  236. return data;
  237. }
  238. function do_onmark_macro_render(data) {
  239. data = data.replace(/\[([^[\](]+)\(((?:(?!\)\]).)+)\)\]/g, function(x, x_1, x_2) {
  240. x_1 = x_1.toLowerCase();
  241. if(x_1 === 'youtube' || x_1 === 'kakaotv' || x_1 === 'nicovideo') {
  242. var video_code = x_2.match(/^([^,]+)/);
  243. video_code = video_code ? video_code[1] : '';
  244. var video_width = x_2.match(/,(?: *)width=([0-9]+)/);
  245. video_width = video_width ? (video_width[1] + 'px') : '640px';
  246. var video_height = x_2.match(/,(?: *)height=([0-9]+)/);
  247. video_height = video_height ? (video_height[1] + 'px') : '360px';
  248. if(x_1 === 'youtube') {
  249. var video_start = x_2.match(/,(?: *)start=([0-9]+)/);
  250. video_start = video_start ? ('?' + video_start[1]) : '';
  251. video_code = video_code.replace(/^https:\/\/www\.youtube\.com\/watch\?v=/, '');
  252. video_code = video_code.replace(/^https:\/\/youtu\.be\//, '');
  253. var video_src = 'https://www.youtube.com/embed/' + video_code + video_start
  254. } else if(x_1 === 'kakaotv') {
  255. video_code = video_code.replace(/^https:\/\/tv\.kakao\.com\/channel\/9262\/cliplink\//, '');
  256. video_code = video_code.replace(/^http:\/\/tv\.kakao\.com\/v\//, '');
  257. var video_src = 'https://tv.kakao.com/embed/player/cliplink/' + video_code +'?service=kakao_tv'
  258. } else {
  259. var video_src = 'https://embed.nicovideo.jp/watch/' + video_code
  260. }
  261. return '<iframe style="width: ' + video_width + '; height: ' + video_height + ';" src="' + video_src + '" frameborder="0" allowfullscreen></iframe>';
  262. } else if(x_1 === 'anchor') {
  263. return '<span id="' + x_2 + '"></span>';
  264. } else {
  265. return '<macro_start>' + x_1 + '(' + x_2 + ')<macro_end>';
  266. }
  267. });
  268. data = data.replace(/\[([^[*()\]]+)\]/g, function(x, x_1) {
  269. x_1 = x_1.toLowerCase();
  270. if(x_1 === 'date') {
  271. return do_return_date();
  272. } else if(x_1 === 'clearfix') {
  273. return '<div style="clear:both"></div>';
  274. } else if(x_1 === 'br') {
  275. return '<br>';
  276. } else {
  277. return '<macro_start>' + x_1 + '<macro_end>';
  278. }
  279. });
  280. data = data.replace(/<macro_start>/g, '[');
  281. data = data.replace(/<macro_end>/g, ']');
  282. return data;
  283. }
  284. function do_onmark_middle_render(data, data_js, name_include) {
  285. var middle_stack = [];
  286. var middle_re = /(?:{{{([^{} ]+)|(}}}))/;
  287. var syntax_on = 0;
  288. var html_n = 0;
  289. while(1) {
  290. var middle_data = data.match(middle_re);
  291. if(!middle_data) {
  292. break;
  293. }
  294. if(middle_data[2]) {
  295. if(middle_stack.length === 0) {
  296. data = data.replace(middle_re, '<middle_end>');
  297. } else {
  298. data = data.replace(middle_re, middle_stack[middle_stack.length - 1]);
  299. middle_stack.pop();
  300. }
  301. } else {
  302. if(middle_data[1].match(/^(?:(#(?:[0-9a-f-A-F]{3}){1,2})|#([a-zA-Z]+))/)) {
  303. var color = middle_data[1].match(/^(?:(#(?:[0-9a-f-A-F]{3}){1,2})|#([a-zA-Z]+))/);
  304. color = color[1] ? color[1] : color[2];
  305. data = data.replace(middle_re, '<span style="color: ' + color + ';">');
  306. middle_stack.push('</span>');
  307. } else if(middle_data[1].match(/^(\+|-)([1-5])/)) {
  308. var font = middle_data[1].match(/^(\+|-)([1-5])/);
  309. if(font[1] === '+') {
  310. var font_size = String(100 + (20 * Number(font[2]))) + '%';
  311. } else {
  312. var font_size = String(100 - (10 * Number(font[2]))) + '%';
  313. }
  314. data = data.replace(middle_re, '<span style="font-size: ' + font_size + ';">');
  315. middle_stack.push('</span>');
  316. } else if(middle_data[1] === '#!wiki') {
  317. var wiki_re = /{{{#!wiki(?: style=["']([^"']*)["']\n)?/;
  318. var wiki = data.match(wiki_re);
  319. var wiki_style = wiki[1] ? wiki[1] : '';
  320. data = data.replace(wiki_re, '<div_wiki_start style="' + wiki_style + '">');
  321. middle_stack.push('<div_wiki_end>');
  322. } else if(middle_data[1] === '#!html') {
  323. html_n += 1;
  324. data = data.replace(middle_re, '<span id="' + name_include + 'render_contect_' + String(html_n) + '">');
  325. middle_stack.push('</span>');
  326. } else if(middle_data[1] === '#!folding') {
  327. data = data.replace(middle_re, '<div>');
  328. middle_stack.push('</div>');
  329. } else {
  330. data = data.replace(middle_re, '<middle_start>' + middle_data[1]);
  331. }
  332. }
  333. }
  334. while(middle_stack.length !== 0) {
  335. data += middle_stack[middle_stack.length - 1];
  336. middle_stack.pop();
  337. }
  338. data = data.replace(/\n<div_wiki_end>/g, '<div_wiki_end>');
  339. data = data.replace(/<middle_start>/g, '{{{');
  340. data = data.replace(/<middle_end>/g, '}}}');
  341. return [data, data_js];
  342. }
  343. function do_onmark_last_render(data) {
  344. // middle_render 마지막 처리
  345. data = data.replace(/<div_wiki_start /g, '<div ');
  346. data = data.replace(/<div_wiki_end>/g, '</div>');
  347. // heading_render 마지막 처리
  348. data = data.replace(/(<\/h[0-9]>)\n/g, '$1');
  349. // list_render 마지막 처리
  350. data = data.replace(/(<\/ul>)\n/g, '$1');
  351. // br 마지막 처리
  352. data = data.replace(/^(\n| )+/, '');
  353. data = data.replace(/(\n| )+$/, '');
  354. data = data.replace(/\n/g, '<br>');
  355. return data;
  356. }
  357. function do_onmark_include_render(data, data_js, name_include) {
  358. var include_re = /\[include\(((?:(?!\)\]).)+)\)\]/;
  359. var i = 0;
  360. while(1) {
  361. i += 1;
  362. var include_data = data.match(include_re);
  363. if(!include_data) {
  364. break;
  365. }
  366. data = data.replace(include_re,
  367. '<a id="' + name_include + 'include_link" class="include_' + String(i) + '" href="">(' + include_data[1] + ')</a>' +
  368. '<div id="' + name_include + 'include_' + String(i) + '"></div>'
  369. );
  370. data_js += 'load_include("' + do_js_safe_change(include_data[1]) + '", "' + name_include + 'include_' + String(i) + '", []);\n'
  371. }
  372. return [data, data_js];
  373. }
  374. function do_onmark_nowiki_before_render(data, data_js, name_include, data_nowiki, num_nowiki = 0) {
  375. data = data.replace(/\\(.)/g, function(x, x_1) {
  376. num_nowiki += 1;
  377. data_nowiki[name_include + 'nowiki_' + String(num_nowiki)] = x_1;
  378. data_js += do_data_try_insert(name_include + 'nowiki_' + String(num_nowiki), do_js_safe_change(x_1));
  379. return '<span id="' + name_include + 'nowiki_' + String(num_nowiki) + '"></span>';
  380. });
  381. return [data, data_js, data_nowiki, num_nowiki];
  382. }
  383. function do_onmark_table_render(data) {
  384. return data;
  385. }
  386. function do_onmark_list_render(data) {
  387. var list_re = /\n((?:(?:(?: )+)\* (?:(?:(?!\n).)+)\n)+)/;
  388. var list_short_re = /((?: )+)\* ((?:(?!\n).)+)\n/g;
  389. while(1) {
  390. var list_data = data.match(list_re);
  391. if(!list_data) {
  392. break;
  393. }
  394. var list_end_data = '<ul>' + list_data[1].replace(list_short_re, function(x, x_1, x_2) {
  395. return '<li style="margin-left: ' + String(x_1.length * 20) + 'px;">' + x_2 + '</li>';
  396. }) + '</ul>';
  397. data = data.replace(list_re, '\n' + list_end_data + '\n');
  398. }
  399. return data;
  400. }
  401. function do_onmark_math_render(data, data_js, name_include) {
  402. data = data.replace(/<math>((?:(?!<\/math>).)+)<\/math>/g, '[math($1)]');
  403. var i = 0;
  404. data = data.replace(/\[math\((((?!\)]).)+)\)]/g, function(x, x_1) {
  405. i += 1;
  406. data_js += do_math_try_insert(name_include + 'math_' + String(i), do_js_safe_change(do_xss_change(x_1)));
  407. return '<span id="' + name_include + 'math_' + String(i) + '"></span>';
  408. });
  409. return [data, data_js];
  410. }
  411. // Main
  412. function do_onmark_render(test_mode = 1, name_id = '', name_include = '', name_doc = '') {
  413. if(test_mode === 0) {
  414. var data = '\n' + document.getElementById(name_id).innerHTML.replace(/\r/g, '') + '\n';
  415. } else {
  416. var data = '\n' + (
  417. ``
  418. ) + '\n';
  419. }
  420. var data_js = '';
  421. var data_backlink = [];
  422. var data_nowiki = {};
  423. var data_var = do_onmark_math_render(data, data_js, name_include);
  424. data = data_var[0];
  425. data_js = data_var[1];
  426. data_var = do_onmark_nowiki_before_render(data, data_js, name_include, data_nowiki);
  427. data = data_var[0];
  428. data_js = data_var[1];
  429. data_nowiki = data_var[2];
  430. var num_nowiki = data_var[3];
  431. data_var = do_onmark_include_render(data, data_js, name_include);
  432. data = data_var[0];
  433. data_js = data_var[1];
  434. data_var = do_onmark_middle_render(data, data_js, name_include);
  435. data = data_var[0];
  436. data_js = data_var[1];
  437. data = do_onmark_text_render(data);
  438. data = do_onmark_heading_render(data);
  439. data = do_onmark_table_render(data);
  440. data_var = do_onmark_link_render(data, data_js, name_doc, name_include, data_nowiki);
  441. data = data_var[0];
  442. data_js = data_var[1];
  443. data = do_onmark_macro_render(data);
  444. data = do_onmark_list_render(data);
  445. data = do_onmark_footnote_render(data, name_include);
  446. data = do_onmark_last_render(data, name_include);
  447. data_js += '' +
  448. 'get_link_state("' + name_include + '");\n' +
  449. 'get_file_state("' + name_include + '");\n' +
  450. ''
  451. data_js = 'render_html("' + name_include + 'render_contect");\n' + data_js
  452. if(test_mode === 0) {
  453. document.getElementById(name_id).innerHTML = data;
  454. eval(data_js);
  455. } else {
  456. console.log([data, data_js]);
  457. }
  458. }
  459. do_onmark_render();