markdown.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516
  1. "use strict";
  2. class opennamu_render_markdown {
  3. // Init Part
  4. constructor(
  5. render_part_id,
  6. render_part_id_after,
  7. render_part_id_add,
  8. doc_name
  9. ) {
  10. this.doc_data = document.getElementById(render_part_id_add + render_part_id).innerHTML;
  11. this.doc_data = this.doc_data.replace(/&/g, '&');
  12. this.doc_data = '<brStart>\n' + this.doc_data + '\n<brEnd>';
  13. this.doc_data.replace(/\r/g, '');
  14. this.doc_name = doc_name;
  15. this.parser_data_temp = {};
  16. this.parser_data_temp_other = {};
  17. this.parser_data_temp_other['toc'] = '';
  18. this.parser_data_temp_other['footnote'] = '';
  19. this.parser_data_temp_other['category'] = '';
  20. this.parser_data_js = [];
  21. this.parser_count = {};
  22. this.parser_count['parser'] = 0;
  23. this.parser_count['nowiki'] = 0;
  24. this.render_part_id_add = render_part_id_add;
  25. this.render_part_id_after = render_part_id_after;
  26. }
  27. // Func Part
  28. do_func_parser_to_text(data, parser_type = 'parser') {
  29. let parser_data_temp = this.parser_data_temp;
  30. let parser_match;
  31. if(parser_type === 'nowiki') {
  32. parser_match = /<(\/?nowiki[0-9]+Span)>/;
  33. } else {
  34. parser_match = /<(\/?render[0-9]+Span)>/;
  35. }
  36. while(data.match(parser_match)) {
  37. data = data.replace(parser_match, function(match, x1) {
  38. return parser_data_temp[x1];
  39. });
  40. }
  41. return data;
  42. }
  43. do_func_xss_encode(data) {
  44. data = data.replace(/"/g, '&quot;');
  45. data = data.replace(/</g, '&lt;');
  46. data = data.replace(/</g, '&gt;');
  47. return data;
  48. }
  49. do_func_xss_decode(data) {
  50. data = data.replace(/&#x27;/g, '\'');
  51. data = data.replace(/&quot;/g, '"');
  52. data = data.replace(/&lt;/g, '<');
  53. data = data.replace(/&gt;/g, '<');
  54. return data;
  55. }
  56. // Render Part
  57. do_part_text() {
  58. let parser_count = this.parser_count['parser'];
  59. let parser_data_temp = this.parser_data_temp;
  60. this.doc_data = this.doc_data.replace(/~~((?:(?!~~).)+)~~/g, function(match, x1) {
  61. parser_count += 1;
  62. let parser_count_str = String(parser_count);
  63. parser_data_temp['render' + parser_count_str + 'Span'] = '<s>';
  64. parser_data_temp['/render' + parser_count_str + 'Span'] = '</s>';
  65. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  66. });
  67. this.doc_data = this.doc_data.replace(/\*\*((?:(?!\*\*).)+)\*\*/g, function(match, x1) {
  68. parser_count += 1;
  69. let parser_count_str = String(parser_count);
  70. parser_data_temp['render' + parser_count_str + 'Span'] = '<b>';
  71. parser_data_temp['/render' + parser_count_str + 'Span'] = '</b>';
  72. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  73. });
  74. this.doc_data = this.doc_data.replace(/__((?:(?!__).)+)__/g, function(match, x1) {
  75. parser_count += 1;
  76. let parser_count_str = String(parser_count);
  77. parser_data_temp['render' + parser_count_str + 'Span'] = '<b>';
  78. parser_data_temp['/render' + parser_count_str + 'Span'] = '</b>';
  79. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  80. });
  81. this.doc_data = this.doc_data.replace(/\*([^*]+)\*/g, function(match, x1) {
  82. parser_count += 1;
  83. let parser_count_str = String(parser_count);
  84. parser_data_temp['render' + parser_count_str + 'Span'] = '<i>';
  85. parser_data_temp['/render' + parser_count_str + 'Span'] = '</i>';
  86. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  87. });
  88. this.doc_data = this.doc_data.replace(/_([^_]+)_/g, function(match, x1) {
  89. parser_count += 1;
  90. let parser_count_str = String(parser_count);
  91. parser_data_temp['render' + parser_count_str + 'Span'] = '<i>';
  92. parser_data_temp['/render' + parser_count_str + 'Span'] = '</i>';
  93. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  94. });
  95. this.doc_data = this.doc_data.replace(/&lt;ins&gt;((?:(?!&lt;ins&gt;|&lt;\/ins&gt;).)+)&lt;\/ins&gt;/g, function(match, x1) {
  96. parser_count += 1;
  97. let parser_count_str = String(parser_count);
  98. parser_data_temp['render' + parser_count_str + 'Span'] = '<u>';
  99. parser_data_temp['/render' + parser_count_str + 'Span'] = '</u>';
  100. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  101. });
  102. this.doc_data = this.doc_data.replace(/&lt;sub&gt;((?:(?!&lt;sub&gt;|&lt;\/sub&gt;).)+)&lt;\/sub&gt;/g, function(match, x1) {
  103. parser_count += 1;
  104. let parser_count_str = String(parser_count);
  105. parser_data_temp['render' + parser_count_str + 'Span'] = '<sub>';
  106. parser_data_temp['/render' + parser_count_str + 'Span'] = '</sub>';
  107. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  108. });
  109. this.doc_data = this.doc_data.replace(/&lt;sup&gt;((?:(?!&lt;sup&gt;|&lt;\/sup&gt;).)+)&lt;\/sup&gt;/g, function(match, x1) {
  110. parser_count += 1;
  111. let parser_count_str = String(parser_count);
  112. parser_data_temp['render' + parser_count_str + 'Span'] = '<sup>';
  113. parser_data_temp['/render' + parser_count_str + 'Span'] = '</sup>';
  114. return '<render' + parser_count_str + 'Span>' + x1 + '</render' + parser_count_str + 'Span>';
  115. });
  116. this.parser_count['parser'] = parser_count;
  117. this.parser_data_temp = parser_data_temp;
  118. }
  119. do_part_heading() {
  120. let parser_count = this.parser_count['parser'];
  121. let parser_data_temp = this.parser_data_temp;
  122. let toc_data = '';
  123. let heading_n = 0;
  124. let heading_list = [0, 0, 0, 0, 0, 0];
  125. let heading_regex = /\n(#{1,6})([^\n]+)\n/;
  126. while(this.doc_data.match(heading_regex)) {
  127. this.doc_data = this.doc_data.replace(heading_regex, function(match, x1, x2) {
  128. let heading_level = x1.length - 1;
  129. let heading_level_str = String(heading_level + 1);
  130. heading_list[heading_level] += 1;
  131. for(let for_a = heading_level + 1; for_a < 6; for_a++) {
  132. heading_list[for_a] = 0;
  133. }
  134. let heading_list_str = '';
  135. for(let for_a = 0; for_a < 6; for_a++) {
  136. if(heading_list[for_a] !== 0) {
  137. heading_list_str += String(heading_list[for_a]) + '.'
  138. }
  139. }
  140. let heading_list_str_2 = heading_list_str.replace(/\.$/, '');
  141. heading_n += 1;
  142. let heading_n_str = String(heading_n);
  143. toc_data += '' +
  144. '<a href="#opennamuHeading' + heading_list_str_2 + '">' + heading_list_str + '</a> ' +
  145. '<span id="opennamuTOCcontent' + heading_n_str + '"></span>' +
  146. '<br>' +
  147. ''
  148. let heading_data = x2;
  149. heading_data = heading_data.replace(/^ /, '');
  150. return '' +
  151. '\n<brEnd>' +
  152. '<h' + heading_level_str + ' id="opennamuHeading' + heading_list_str_2 + '">' +
  153. '<a href="#opennamuTOC">' + heading_list_str + '</a> ' +
  154. '<span id="opennamuHeadingContent' + heading_n_str + '">' + heading_data + '</span>' +
  155. '</h' + heading_level_str + '>' +
  156. '<brStart>\n' +
  157. '';
  158. });
  159. }
  160. this.parser_data_temp_other['toc'] = toc_data;
  161. }
  162. do_part_image() {
  163. let render_main = this;
  164. let render_part_id_add = this.render_part_id_add;
  165. let parser_count = this.parser_count['parser'];
  166. let parser_data_temp = this.parser_data_temp;
  167. this.doc_data = this.doc_data.replace(/!\[([^\[\]]*)\]\(([^\(\)]*)\)/g, function(match, x1, x2) {
  168. if(x1 === '' && x2 === '') {
  169. return '<imageBlink>';
  170. } else {
  171. if(x2 !== '' && x2.match(/^https?:\/\//)) {
  172. parser_count += 1;
  173. let parser_count_str = String(parser_count);
  174. let image_src = render_main.do_func_xss_encode(x2);
  175. let image_alt;
  176. if(x1 !== '') {
  177. image_alt = render_main.do_func_xss_encode(x1);
  178. } else {
  179. image_alt = image_src;
  180. }
  181. parser_data_temp['render' + parser_count_str + 'Span'] = '<img alt="' + image_alt + '" src="' + image_src + '">';
  182. parser_data_temp['/render' + parser_count_str + 'Span'] = '';
  183. return '<render' + parser_count_str + 'Span></render' + parser_count_str + 'Span>';
  184. } else {
  185. parser_count += 1;
  186. let parser_count_str = String(parser_count);
  187. parser_data_temp['render' + parser_count_str + 'Span'] = '<img>';
  188. parser_data_temp['/render' + parser_count_str + 'Span'] = '';
  189. return '<render' + parser_count_str + 'Span></render' + parser_count_str + 'Span>';
  190. }
  191. }
  192. });
  193. this.doc_data = this.doc_data.replace(/<imageBlink>/g, '![]()');
  194. this.parser_count['parser'] = parser_count;
  195. this.parser_data_temp = parser_data_temp;
  196. }
  197. do_part_link() {
  198. let render_main = this;
  199. let render_part_id_add = this.render_part_id_add;
  200. let parser_count = this.parser_count['parser'];
  201. let parser_data_temp = this.parser_data_temp;
  202. this.doc_data = this.doc_data.replace(/\[([^\[\]]*)\]\(([^\(\)]*)\)/g, function(match, x1, x2) {
  203. if(x1 === '' && x2 === '') {
  204. return '<linkBlink>';
  205. } else {
  206. if(x2 !== '' && x2.match(/^https?:\/\//)) {
  207. parser_count += 1;
  208. let parser_count_str = String(parser_count);
  209. let link_main = render_main.do_func_xss_encode(x2);;
  210. let link_sub;
  211. if(x1 === '') {
  212. link_sub = x2;
  213. } else {
  214. link_sub = x1;
  215. }
  216. parser_data_temp['render' + parser_count_str + 'Span'] = '<a href="' + link_main + '">';
  217. parser_data_temp['/render' + parser_count_str + 'Span'] = '</a>';
  218. return '<render' + parser_count_str + 'Span>' + link_sub + '</render' + parser_count_str + 'Span>';
  219. } else {
  220. parser_count += 1;
  221. let parser_count_str = String(parser_count);
  222. let link_main;
  223. let link_sub;
  224. let link_title;
  225. if(x2 === '') {
  226. link_main = x1;
  227. link_sub = x1;
  228. } else if(x1 === '') {
  229. link_main = x2;
  230. link_sub = x2;
  231. } else {
  232. link_main = x2;
  233. link_sub = x1;
  234. }
  235. link_title = render_main.do_func_xss_encode(link_main);
  236. link_main = render_main.do_func_xss_decode(link_main);
  237. link_main = opennamu_do_url_encode(link_main);
  238. parser_data_temp['render' + parser_count_str + 'Span'] = '<a class="' + render_part_id_add + 'opennamuLink" title="' + link_title + '" href="/w/' + link_main + '">';
  239. parser_data_temp['/render' + parser_count_str + 'Span'] = '</a>';
  240. return '<render' + parser_count_str + 'Span>' + link_sub + '</render' + parser_count_str + 'Span>';
  241. }
  242. }
  243. });
  244. this.doc_data = this.doc_data.replace(/<linkBlink>/g, '[]()');
  245. this.parser_count['parser'] = parser_count;
  246. this.parser_data_temp = parser_data_temp;
  247. }
  248. do_part_footnote_list() {
  249. if(this.parser_data_temp_other['footnote'] !== '') {
  250. let footnote = '';
  251. footnote += '<ul id="footnote<underBar>data">';
  252. footnote += this.parser_data_temp_other['footnote'];
  253. footnote += '</ul>';
  254. this.parser_data_temp_other['footnote'] = '';
  255. return footnote;
  256. } else {
  257. return '';
  258. }
  259. }
  260. do_part_footnote() {
  261. let render_main = this;
  262. let parser_count = this.parser_count['parser'];
  263. let parser_data_temp = this.parser_data_temp;
  264. let parser_data_temp_other = this.parser_data_temp_other;
  265. let footnote_n = 0;
  266. let footnote_name_all = {};
  267. this.doc_data = this.doc_data.replace(/(?:\[\^((?:(?!\[\^| ).)*)(?: ((?:(?!\[\^|\]).)+))?\]|(\[fnote\(\)]))/g, function(match, x1, x2, x3) {
  268. if(x3 === undefined) {
  269. if(x1 === '' && x2 === undefined) {
  270. return '<footnoteBlink>';
  271. } else {
  272. footnote_n += 1;
  273. let footnote_name;
  274. let footnote_id = String(footnote_n);
  275. if(x1 === '') {
  276. footnote_name = footnote_id;
  277. } else {
  278. footnote_name = x1;
  279. }
  280. let footnote_content;
  281. if(x2 === undefined) {
  282. if(footnote_name_all[footnote_name]) {
  283. footnote_content = footnote_name_all[footnote_name];
  284. } else {
  285. footnote_content = '';
  286. footnote_name_all[footnote_name] = footnote_content;
  287. }
  288. } else {
  289. footnote_content = x2;
  290. footnote_name_all[footnote_name] = footnote_content;
  291. }
  292. let footnote_list = '';
  293. footnote_list += '<li>';
  294. footnote_list += '<a id="opennamuFnGo' + footnote_id + '" href="#opennamuFnIn' + footnote_id + '">'
  295. footnote_list += '(' + footnote_name + ')';
  296. footnote_list += '</a>';
  297. footnote_list += ' ';
  298. footnote_list += footnote_content;
  299. footnote_list += '</li>';
  300. parser_data_temp_other['footnote'] += footnote_list;
  301. parser_count += 1;
  302. let parser_count_str = String(parser_count);
  303. let footnote_data = '';
  304. footnote_data += '<sup>'
  305. footnote_data += '<a id="opennamuFnIn' + footnote_id + '" href="#opennamuFnGo' + footnote_id + '">';
  306. footnote_data += '(' + footnote_name + ')';
  307. footnote_data += '</a>';
  308. footnote_data += '</sup>';
  309. parser_data_temp['render' + parser_count_str + 'Span'] = footnote_data;
  310. parser_data_temp['/render' + parser_count_str + 'Span'] = '';
  311. return '<render' + parser_count_str + 'Span></render' + parser_count_str + 'Span>';
  312. }
  313. } else {
  314. return render_main.do_part_footnote_list();
  315. }
  316. });
  317. this.doc_data = this.doc_data.replace(/<footnoteBlink>/g, '[^]');
  318. this.doc_data += this.do_part_footnote_list();
  319. this.parser_count['parser'] = parser_count;
  320. this.parser_data_temp = parser_data_temp;
  321. this.parser_data_temp_other = parser_data_temp_other;
  322. }
  323. do_part_macro() {
  324. let render_main = this;
  325. let render_part_id_add = this.render_part_id_add;
  326. let parser_count = this.parser_count['parser'];
  327. let parser_data_temp = this.parser_data_temp;
  328. let parser_data_temp_other = this.parser_data_temp_other;
  329. this.doc_data = this.doc_data.replace(/\[([^\[\(<>]+)\(((?:(?!\(|\)\]|<|>).)*)\)\]/g, function(match, x1, x2) {
  330. if(x1 === 'anchor') {
  331. parser_count += 1;
  332. let parser_count_str = String(parser_count);
  333. let anchor_data = render_main.do_func_parser_to_text(x2, 'nowiki');
  334. anchor_data = render_main.do_func_xss_encode(anchor_data);
  335. parser_data_temp['render' + parser_count_str + 'Span'] = '<span id="' + anchor_data + '">';
  336. parser_data_temp['/render' + parser_count_str + 'Span'] = '</span>';
  337. return '<render' + parser_count_str + 'Span></render' + parser_count_str + 'Span>'
  338. } else if(x1 === 'category') {
  339. let category = x2.split(',');
  340. let category_data = render_main.do_func_parser_to_text(category[0], 'nowiki');
  341. let link_main = 'category:' + category_data;
  342. let link_title = render_main.do_func_xss_encode(link_main);
  343. link_main = render_main.do_func_xss_decode(link_main);
  344. link_main = opennamu_do_url_encode(link_main);
  345. parser_data_temp_other['category'] += '<a class="' + render_part_id_add + 'opennamuLink" title="' + link_title + '" href="/w/' + link_main + '">';
  346. parser_data_temp_other['category'] += category_data;
  347. parser_data_temp_other['category'] += '</a>';
  348. parser_data_temp_other['category'] += ' | ';
  349. return '';
  350. } else if(x1 === 'toc') {
  351. if(parser_data_temp_other['toc'] !== '') {
  352. return '' +
  353. '<div class="opennamuTOC">' +
  354. '<span class="opennamuTOCtitle">' +
  355. 'TOC' +
  356. '</span>' +
  357. '<br>' +
  358. '<br>' +
  359. parser_data_temp_other['toc'] +
  360. '</div>' +
  361. '';
  362. } else {
  363. return '';
  364. }
  365. }
  366. else {
  367. return '';
  368. }
  369. });
  370. this.parser_count['parser'] = parser_count;
  371. this.parser_data_temp = parser_data_temp;
  372. this.parser_data_temp_other = parser_data_temp_other;
  373. }
  374. do_part_final() {
  375. this.doc_data = this.doc_data.replace(/<brStart>\n?/g, '');
  376. this.doc_data = this.doc_data.replace(/\n?<brEnd>/g, '');
  377. this.doc_data = this.doc_data.replace(/<underBar>/g, '_');
  378. this.doc_data = this.doc_data.replace(/\n/g, '<br>');
  379. if(this.parser_data_temp_other['category'] !== '') {
  380. let category = this.parser_data_temp_other['category'];
  381. category = category.replace(/ \| $/, '');
  382. this.doc_data += '<hr class="main_hr">';
  383. this.doc_data += '<div id="cate_all"><div id="cate">';
  384. this.doc_data += 'Category : ';
  385. this.doc_data += category;
  386. this.doc_data += '</div></div>';
  387. }
  388. }
  389. // Main Part
  390. do_main() {
  391. this.do_part_heading();
  392. this.do_part_footnote();
  393. this.do_part_macro();
  394. this.do_part_image();
  395. this.do_part_link();
  396. this.do_part_text();
  397. this.do_part_final();
  398. this.doc_data = this.do_func_parser_to_text(this.doc_data);
  399. document.getElementById(this.render_part_id_add + this.render_part_id_after).innerHTML = this.doc_data;
  400. for(let x1 in this.parser_data_js) {
  401. eval(this.parser_data_js[x1]);
  402. }
  403. new opennamu_render_wiki(
  404. render_part_id_add = this.render_part_id_add
  405. ).do_main();
  406. }
  407. }