| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202 |
- import markdownTable from 'markdown-table';
- import stringWidth from 'string-width';
- /**
- * Utility for markdown table
- */
- class MarkdownTableUtil {
- constructor() {
- // https://github.com/markdown-it/markdown-it/blob/d29f421927e93e88daf75f22089a3e732e195bd2/lib/rules_block/table.js#L83
- // https://regex101.com/r/7BN2fR/7
- this.tableAlignmentLineRE = /^[-:|][-:|\s]*$/;
- this.tableAlignmentLineNegRE = /^[^-:]*$/; // it is need to check to ignore empty row which is matched above RE
- this.linePartOfTableRE = /^\|[^\r\n]*|[^\r\n]*\|$|([^|\r\n]+\|[^|\r\n]*)+/; // own idea
- this.getBot = this.getBot.bind(this);
- this.getEot = this.getEot.bind(this);
- this.getBol = this.getBol.bind(this);
- this.getStrFromBot = this.getStrFromBot.bind(this);
- this.getStrToEot = this.getStrToEot.bind(this);
- this.parseFromTableStringToMarkdownTable = this.parseFromTableStringToMarkdownTable.bind(this);
- this.replaceMarkdownTableWithReformed = this.replaceMarkdownTableWithReformed.bind(this);
- }
- /**
- * return the postion of the BOT(beginning of table)
- * (It is assumed that current line is a part of table)
- */
- getBot(editor) {
- const firstLine = editor.getDoc().firstLine();
- const curPos = editor.getCursor();
- let line = curPos.line - 1;
- for (; line >= firstLine; line--) {
- const strLine = editor.getDoc().getLine(line);
- if (!this.linePartOfTableRE.test(strLine)) {
- break;
- }
- }
- const botLine = Math.max(firstLine, line + 1);
- return { line: botLine, ch: 0 };
- }
- /**
- * return the postion of the EOT(end of table)
- * (It is assumed that current line is a part of table)
- */
- getEot(editor) {
- const lastLine = editor.getDoc().lastLine();
- const curPos = editor.getCursor();
- let line = curPos.line + 1;
- for (; line <= lastLine; line++) {
- const strLine = editor.getDoc().getLine(line);
- if (!this.linePartOfTableRE.test(strLine)) {
- break;
- }
- }
- const eotLine = Math.min(line - 1, lastLine);
- const lineLength = editor.getDoc().getLine(eotLine).length;
- return { line: eotLine, ch: lineLength };
- }
- /**
- * return the postion of the BOL(beginning of line)
- */
- getBol(editor) {
- const curPos = editor.getCursor();
- return { line: curPos.line, ch: 0 };
- }
- /**
- * return strings from BOT(beginning of table) to current position
- */
- getStrFromBot(editor) {
- const curPos = editor.getCursor();
- return editor.getDoc().getRange(this.getBot(editor), curPos);
- }
- /**
- * return strings from current position to EOT(end of table)
- */
- getStrToEot(editor) {
- const curPos = editor.getCursor();
- return editor.getDoc().getRange(curPos, this.getEot(editor));
- }
- /**
- * returns markdown table whose described by 'markdown-table' format
- * ref. https://github.com/wooorm/markdown-table
- * @param {string} lines all of table
- */
- parseFromTableStringToMarkdownTable(strMDTable) {
- const arrMDTableLines = strMDTable.split(/(\r\n|\r|\n)/);
- let contents = [];
- let aligns = [];
- for (let n = 0; n < arrMDTableLines.length; n++) {
- const line = arrMDTableLines[n];
- if (this.tableAlignmentLineRE.test(line) && !this.tableAlignmentLineNegRE.test(line)) {
- // parse line which described alignment
- const alignRuleRE = [
- { align: 'c', regex: /^:-+:$/ },
- { align: 'l', regex: /^:-+$/ },
- { align: 'r', regex: /^-+:$/ },
- ];
- let lineText = '';
- lineText = line.replace(/^\||\|$/g, ''); // strip off pipe charactor which is placed head of line and last of line.
- lineText = lineText.replace(/\s*/g, '');
- aligns = lineText.split(/\|/).map(col => {
- const rule = alignRuleRE.find(rule => col.match(rule.regex));
- return (rule != undefined) ? rule.align : '';
- });
- }
- else if (this.linePartOfTableRE.test(line)) {
- // parse line whether header or body
- let lineText = '';
- lineText = line.replace(/\s*\|\s*/g, '|');
- lineText = lineText.replace(/^\||\|$/g, ''); // strip off pipe charactor which is placed head of line and last of line.
- const row = lineText.split(/\|/);
- contents.push(row);
- }
- }
- return (new MarkdownTable(contents, { align: aligns, stringLength: stringWidth }));
- }
- /**
- * return boolean value whether the current position of cursor is end of line
- */
- isEndOfLine(editor) {
- const curPos = editor.getCursor();
- return (curPos.ch == editor.getDoc().getLine(curPos.line).length);
- }
- /**
- * add a row at the end
- * (This function overwrite directory markdown table specified as argument.)
- * @param {MarkdownTable} markdown table
- */
- addRowToMarkdownTable(mdtable) {
- const numCol = mdtable.table.length > 0 ? mdtable.table[0].length : 1;
- let newRow = [];
- (new Array(numCol)).forEach(() => newRow.push('')); // create cols
- mdtable.table.push(newRow);
- }
- /**
- * returns markdown table that is merged all of markdown table in array
- * (The merged markdown table options are used for the first markdown table.)
- * @param {Array} array of markdown table
- */
- mergeMarkdownTable(mdtable_list) {
- if (mdtable_list == undefined
- || !(mdtable_list instanceof Array)) {
- return undefined;
- }
- let newTable = [];
- const options = mdtable_list[0].options; // use option of first markdown-table
- mdtable_list.forEach((mdtable) => {
- newTable = newTable.concat(mdtable.table);
- });
- return (new MarkdownTable(newTable, options));
- }
- /**
- * replace markdown table which is reformed by markdown-table
- * @param {MarkdownTable} markdown table
- */
- replaceMarkdownTableWithReformed(editor, table) {
- const curPos = editor.getCursor();
- // replace the lines to strTableLinesFormated
- const strTableLinesFormated = table.toString();
- editor.getDoc().replaceRange(strTableLinesFormated, this.getBot(editor), this.getEot(editor));
- // set cursor to first column
- editor.getDoc().setCursor(curPos.line + 1, 2);
- }
- }
- /**
- * markdown table class for markdown-table module
- * ref. https://github.com/wooorm/markdown-table
- */
- class MarkdownTable {
- constructor(table, options) {
- this.table = table || [];
- this.options = options || {};
- this.toString = this.toString.bind(this);
- }
- toString() {
- return markdownTable(this.table, this.options);
- }
- }
- // singleton pattern
- const instance = new MarkdownTableUtil();
- Object.freeze(instance);
- export default instance;
|