render_namumark.js 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  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_list, i) {
  14. if(link_list[i]) {
  15. get_link_state(link_list, i + 1);
  16. var xhr = new XMLHttpRequest();
  17. xhr.open("GET", "/api/w/" + encodeURI(link_list[i][0]) + "?exist=1", true);
  18. xhr.send(null);
  19. xhr.onreadystatechange = function() {
  20. if(this.readyState === 4 && this.status === 200) {
  21. console.log(JSON.parse(this.responseText)['exist']);
  22. if(JSON.parse(this.responseText)['exist'] !== '1') {
  23. document.getElementById(link_list[i][1]).className = "not_thing";
  24. } else {
  25. document.getElementById(link_list[i][1]).className = "";
  26. }
  27. } else {
  28. document.getElementById(link_list[i][1]).className = "not_thing";
  29. }
  30. }
  31. }
  32. }
  33. var data = '\n' + document.getElementById(target).innerHTML + '\n';
  34. var nowiki_list = [];
  35. var mid_num = 0;
  36. var mid_stack = 0;
  37. var mid_list = [];
  38. var html_number = 0;
  39. data = data.replace(/(?:{{{(?:((?:(?! |{{{|}}}|&lt;).)*) ?)|(}}}))/g, function(all, in_data) {
  40. if(all === '}}}') {
  41. if(!mid_list[mid_num]) {
  42. return all;
  43. } else {
  44. if(mid_stack > 0) {
  45. mid_stack -= 1;
  46. }
  47. if(mid_stack > 0) {
  48. return all;
  49. } else {
  50. if(mid_num > 0) {
  51. mid_num -= 1;
  52. }
  53. if(mid_list[mid_num] === 'pre') {
  54. var return_data = '</code></pre>';
  55. } else {
  56. var return_data = '</' + mid_list[mid_num] + '>';
  57. }
  58. mid_list.splice(mid_num, 1);
  59. return return_data;
  60. }
  61. }
  62. } else {
  63. if(mid_stack > 0) {
  64. mid_stack += 1;
  65. return all;
  66. } else {
  67. mid_num += 1;
  68. if(in_data.match(/^(#|@|\+|\-)/) && !in_data.match(/^(#|@|\+|\-){2}|(#|@|\+|\-)\\\\/)) {
  69. if(in_data.match(/^((#|@)([0-9a-f-A-F]{3}){1,2})/)) {
  70. mid_list.push('span');
  71. if(in_data.match(/^#/)) {
  72. return '<span style="color: ' + in_data + ';">';
  73. } else {
  74. return '<span style="background: ' + in_data + ';">';
  75. }
  76. } else if(in_data.match(/^((#|@)(\w+))/)) {
  77. mid_list.push('span');
  78. if(in_data.match(/^#/)) {
  79. return '<span style="color: ' + in_data.replace(/^#/, '') + ';">';
  80. } else {
  81. return '<span style="background: ' + in_data.replace(/^@/, '') + ';">';
  82. }
  83. } else if(in_data.match(/^(\+|-)([1-5])/)) {
  84. mid_list.push('span');
  85. var font_size_data = in_data.match(/^(\+|-)([1-5])/);
  86. if(font_size_data[1] == '+') {
  87. font_size_data = String(Number(font_size_data[2]) * 20 + 100);
  88. } else {
  89. font_size_data = String(100 - Number(font_size_data[2]) * 10);
  90. }
  91. return '<span style="font-size: ' + font_size_data + '%;">'
  92. } else if(in_data.match(/#!wiki/i)) {
  93. mid_list.push('div');
  94. if(data.match(/{{{#!wiki style=((?:(?!\n).)+) *\n/i)) {
  95. return '<div id="wiki_div_before">';
  96. } else {
  97. return '<div id="wiki_div">'
  98. }
  99. } else if(in_data.match(/#!syntax/i)) {
  100. mid_list.push('pre');
  101. mid_stack += 1;
  102. return '<pre><code id="syntax_before">';
  103. } else if(in_data.match(/#!folding/i)) {
  104. mid_list.push('div');
  105. return '<div id="folding_before">';
  106. } else if(in_data.match(/#!html/i)) {
  107. mid_list.push('span');
  108. html_number += 1;
  109. return '<span id="html_render_contect_' + String(html_number) + '">';
  110. } else {
  111. mid_list.push('code');
  112. mid_stack += 1;
  113. return '<code>' + in_data;
  114. }
  115. } else {
  116. mid_list.push('code');
  117. mid_stack += 1;
  118. return '<code>' + in_data;
  119. }
  120. }
  121. }
  122. });
  123. console.log(mid_stack);
  124. console.log(mid_num);
  125. console.log(mid_list);
  126. data = data.replace(/<\/div> *\n/ig, '</div>');
  127. data = data.replace(/<div id="folding_before">((?:(?!\n).)+) *\n/ig, function(all, in_data) {
  128. return in_data + ' [+]<div id="folding">';
  129. });
  130. data = data.replace(/<pre><code id="syntax_before">((?:(?!\n).)+) *\n/ig, function(all, in_data) {
  131. return '<pre><code>';
  132. });
  133. data = data.replace(/<div id="wiki_div_before">style=((?:(?!\n).)+) *\n/ig, function(all, in_data) {
  134. return '<div style=' + in_data.replace(/&quot;/g, "\"").replace(/&#039;/g, "'") + ' id="wiki_div">';
  135. });
  136. var nowiki_num = 0;
  137. data = data.replace(/<code>((?:(?!<\/code>).)+)<\/code>/gm, function(all, in_data) {
  138. nowiki_num += 1;
  139. nowiki_list.push(['nowiki_' + String(nowiki_num), in_data]);
  140. return '<span id="nowiki_' + String(nowiki_num) + '"></span>';
  141. });
  142. var math_list = [];
  143. var math_num = 0;
  144. data = data.replace(/\[math\(((?:(?!\)]).)+)\)]/ig, function(all, in_data) {
  145. var math_data = in_data.replace(/&amp;/g, "&").replace(/&lt;/g, "<").replace(/&gt;/g, ">").replace(/&quot;/g, "\"").replace(/&#039;/g, "'");
  146. math_num += 1;
  147. math_list.push(['math_' + String(math_num), math_data]);
  148. return '<span id="math_' + String(math_num) + '"></span>';
  149. });
  150. data = data.replace(/~~((?:(?!~~).)+)~~/g, '<s>$1</s>');
  151. data = data.replace(/--((?:(?!--).)+)--/g, '<s>$1</s>');
  152. data = data.replace(/__((?:(?!__).)+)__/g, '<u>$1</u>');
  153. data = data.replace(/'''((?:(?!''').)+)'''/g, '<b>$1</b>');
  154. data = data.replace(/''((?:(?!'').)+)''/g, '<i>$1</i>');
  155. data = data.replace(/\^\^((?:(?!\^\^).)+)\^\^/g, '<sup>$1</sup>');
  156. data = data.replace(/,,((?:(?!,,).)+),,/g, '<sub>$1</sub>');
  157. data = data.replace(/\n( {1,})\* ([^\n]+)/g, function(all, margin_data, in_data) {
  158. return '<li style="margin-left: ' + String(margin_data.length * 20) + 'px;">' + in_data + '</li>'
  159. });
  160. data = data.replace(/\n( {1,})/g, function(all, margin_data) {
  161. return '\n<span style="margin-left: ' + String(margin_data.length * 10) + 'px"></span>'
  162. });
  163. var link_list = [];
  164. var link_num = 0;
  165. var category = '<div id="cate_all"><hr><div id="cate">Category : '
  166. data = data.replace(/\[\[((?:(?!]]).)+)]]/g, function(all, in_data) {
  167. if(in_data.match(/^(?:[^|]+)\|(?:.+)$/)) {
  168. return in_data.replace(/^([^|]+)\|([^|]+)$/, function(all, front_data, back_name) {
  169. link_list.push([front_data, 'link_' + String(link_num)]);
  170. link_num += 1;
  171. return '<a id="link_' + String(link_num - 1) + '" href="/w/' + encodeURI(front_data) + '">' + back_name + '</a>';
  172. });
  173. } else {
  174. if(in_data.match(/^(?:category|분류):/i)) {
  175. category += '<a href="' + encodeURI(in_data) + '">' + in_data + '</a> | ';
  176. return '';
  177. } else if(in_data.match(/^(?:file|파일):/i)) {
  178. } else {
  179. link_list.push([in_data, 'link_' + String(link_num)]);
  180. link_num += 1;
  181. return '<a id="link_' + String(link_num - 1) + '" href="/w/' + encodeURI(in_data) + '">' + in_data + '</a>';
  182. }
  183. }
  184. });
  185. data = data.replace(/\[([^(\]]+)\(((?:(?!\)]).)+)\)]/g, function(all, name, in_data) {
  186. if(name.match(/^youtube|kakaotv|nicovideo$/i)) {
  187. var video_code = in_data.match(/^([^,]+)/);
  188. if(video_code) {
  189. video_code = video_code[1];
  190. } else {
  191. video_code = 'test';
  192. }
  193. if(name === 'youtube') {
  194. var video_src = 'https://www.youtube.com/embed/' + video_code
  195. } else if(name === 'kakaotv') {
  196. var video_src = 'https://tv.kakao.com/embed/player/cliplink/' + video_code +'?service=kakao_tv'
  197. } else {
  198. var video_src = 'https://embed.nicovideo.jp/watch/' + video_code
  199. }
  200. var width_data = in_data.match(/, *width=([^,]+)/);
  201. if(width_data) {
  202. width_data = width_data[1];
  203. } else {
  204. width_data = '560';
  205. }
  206. var height_data = in_data.match(/, *height=([^,]+)/);
  207. if(height_data) {
  208. height_data = height_data[1];
  209. } else {
  210. height_data = '315';
  211. }
  212. return '' +
  213. '<iframe ' +
  214. 'width="' + width_data + '" ' +
  215. 'height="' + height_data + '" ' +
  216. 'src="' + video_src + '" ' +
  217. 'allowfullscreen>' +
  218. '</iframe>' +
  219. '';
  220. } else if(name.match(/^ruby$/i)) {
  221. var main_text = in_data.match(/^([^,]+)/);
  222. if(main_text) {
  223. main_text = main_text[1];
  224. } else {
  225. main_text = 'test';
  226. }
  227. var ruby_text = in_data.match(/, *ruby=([^,]+)/);
  228. if(ruby_text) {
  229. ruby_text = ruby_text[1];
  230. } else {
  231. ruby_text = 'test';
  232. }
  233. var color_text = in_data.match(/, *color=([^,]+)/);
  234. if(color_text) {
  235. color_text = 'color:' + color_text[1];
  236. } else {
  237. color_text = '';
  238. }
  239. return '' +
  240. '<ruby>' +
  241. main_text +
  242. '<rp>(</rp>' +
  243. '<rt>' +
  244. '<span style="' + color_text + '">' + ruby_text + '</span>' +
  245. '</rt>' +
  246. '<rp>)</rp>' +
  247. '</ruby>' +
  248. '';
  249. } else if(name.match(/^anchor$/i)) {
  250. return '<span id="' + in_data + '"></span>';
  251. } else {
  252. return all;
  253. }
  254. });
  255. var toc_array = [0, 0, 0, 0, 0, 0];
  256. var before_data = 0;
  257. var toc_data = '<div id="toc"><span id="toc_title">TOC</span>\n\n'
  258. data = data.replace(/\n(={1,6}) ?([^\n]+) (?:={1,6})/g, function(all, num, in_data) {
  259. num = num.length;
  260. if(before_data > num) {
  261. var i = num;
  262. while(1) {
  263. if(i == 6) {
  264. break;
  265. }
  266. toc_array[i] = 0;
  267. i += 1;
  268. }
  269. }
  270. before_data = num;
  271. toc_array[num - 1] += 1;
  272. num = String(num);
  273. var toc_num = (toc_array.join('.') + '.').replace(/0\./g, '');
  274. if(!toc_num.match(/\./)) {
  275. toc_num += '0.';
  276. }
  277. toc_data += '' +
  278. '<span style="margin-left: ' + String(10 * (toc_num.length / 2) - 10) + 'px;">' +
  279. '<a href="#s-' + toc_num.replace(/\.$/, '') + '">' + toc_num + '</a> ' + in_data +
  280. '</span>' +
  281. '\n' +
  282. '';
  283. return '\n<h' + num + ' id="s-' + toc_num.replace(/\.$/, '') + '"><a href="#toc">' + toc_num + '</a> ' + in_data + '</h' + num + '>';
  284. });
  285. toc_data += '</div>';
  286. data = data.replace(/<\/h([0-9])>\n/g, '</h$1>');
  287. data = data.replace(/\[([^\]]+)\]/g, function(all, name) {
  288. if(name.match(/^br$/i)) {
  289. return '\n'
  290. } else if(name.match(/^목차$/i)) {
  291. return toc_data;
  292. } else if(name.match(/^date|datetime$/i)) {
  293. return get_today();
  294. } else {
  295. return all;
  296. }
  297. });
  298. console.log(data);
  299. var ref_num = 0;
  300. var ref_data = '<hr><ul id="footnote_data">';
  301. var name_ref_data = {};
  302. data = data.replace(/(?:\[\*([^ \]]*)(?: ([^\]]+))?\]|\[(?:각주|footnote)])/g, function(all, name_data, in_data) {
  303. if(all.match(/^\[(?:각주|footnote)]$/i)) {
  304. var new_ref_data = ref_data;
  305. ref_data = '<hr><ul id="footnote_data">';
  306. return new_ref_data + '</ul>';
  307. } else {
  308. ref_num += 1;
  309. console.log(name_data);
  310. if(name_data) {
  311. if(in_data) {
  312. name_ref_data[name_data] = in_data;
  313. ref_data += '' +
  314. '<li>' +
  315. '<a id="fn-' + name_data + '" href="#rfn-' + String(ref_num) + '">(' + name_data + ')</a> ' + in_data + ''
  316. '</li>' +
  317. ''
  318. } else {
  319. ref_data += '' +
  320. '<li>' +
  321. '<a href="#rfn-' + String(ref_num) + '">(' + name_data + ')</a>' +
  322. '</li>' +
  323. ''
  324. }
  325. } else {
  326. ref_data += '' +
  327. '<li>' +
  328. '<a id="fn-' + String(ref_num) + '" href="#rfn-' + String(ref_num) + '">(' + String(ref_num) + ')</a> ' + in_data + ''
  329. '</li>' +
  330. ''
  331. }
  332. if(name_data) {
  333. return '' +
  334. '<sup>' +
  335. '<a href="#fn-' + name_data + '" id="rfn-' + String(ref_num) + '" title="' + name_ref_data[name_data].replace(/<([^>]*)>/g, '') + '">' +
  336. '(' + name_data + ')' +
  337. '</a>' +
  338. '</sup>' +
  339. '';
  340. } else {
  341. return '' +
  342. '<sup>' +
  343. '<a href="#fn-' + String(ref_num) + '" id="rfn-' + String(ref_num) + '" title="' + in_data.replace(/<([^>]*)>/g, '') + '">' +
  344. '(' + String(ref_num) + ')' +
  345. '</a>' +
  346. '</sup>' +
  347. '';
  348. }
  349. }
  350. });
  351. if(ref_data !== '<hr><ul id="footnote_data">') {
  352. data += ref_data + '</ul>';
  353. }
  354. i = 0;
  355. while(1) {
  356. if(nowiki_list[i]) {
  357. data = data.replace('<span id="' + nowiki_list[i][0] + '"></span>', '<code>' + nowiki_list[i][1] + '</code>');
  358. i += 1;
  359. } else {
  360. break;
  361. }
  362. }
  363. data = data.replace(/^(\n| )+/g, '');
  364. data = data.replace(/(\n| )+$/g, '');
  365. data = data.replace(/\n/g, '<br>');
  366. data = data.replace(/&amp;/g, '&');
  367. document.getElementById(target).innerHTML = data;
  368. var i = 0;
  369. while(1) {
  370. if(math_list[i]) {
  371. try {
  372. katex.render(math_list[i][1], document.getElementById(math_list[i][0]));
  373. } catch {
  374. try {
  375. document.getElementById(math_list[i][0]).innerHTML = '<span style="color: red;">' + math_list[i][1] + '</span>';
  376. } catch {}
  377. }
  378. i += 1;
  379. } else {
  380. break;
  381. }
  382. }
  383. get_link_state(link_list, 0);
  384. render_html("html_render_contect");
  385. }