markdown.js 21 KB

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