markdown.js 21 KB

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