JavaScript+Node.js写一款markdown解析器

JavaScript+Node.js写一款markdown解析器

目录

1. 准备工作

2. 处理图片&超链接

3. 处理blockquote

4. 处理标题

5. 处理字体

6. 处理代码块

7. 处理列表

8. 处理表格

9. 调用方法

1. 准备工作

首先编写getHtml函数,传入markdown文本字符串,这里使用fs读取markdown文件内容,返回值是转换过后的字符串。

const fs = require('fs'); const source = fs.readFileSync('./test.md', 'utf-8'); const getHtml = (source) => {     // 处理标题     return source; } const result = getHtml(source); console.log(result);

主要设计正则表达式和String.prototype.replace方法,replace接收的第一个参数可以是正则,第二个参数如果是函数那么返回值就是所替换的内容。

2. 处理图片&超链接

图片和超链接的语法很像,![图片](url),[超链接](url),使用正则匹配同时需要排除`。props会获取正则中的$,$1,$2。也就是匹配的字符整体,第一个括号内容,第二个括号内容。比如这里props[0]就是匹配到的完整内容,第四个参数props[3]是[]中的alt,第五个参数props[4]是链接地址。

const imageora = (source) => {     return source.replace(/(`?)(!?)\[(.*)\]\((.+)\)/gi, (...props) => {         switch (props[0].trim()[0]) {             case '!': return `<a href="${props[4]}" rel="external nofollow" alt="${props[3]}">${props[3]}</a>`;             case '[': return `<img src="${props[4]}" alt="${props[3]}"/>`;             default: return props[0];         }     }); } const getHtml = (source) => {     source = imageora(source);     return source; } 3. 处理blockquote

这里使用\x20匹配空格。如果匹配到内容,将文本props[3]放在blockquote标签返回就行了。

const block = (source) => {     return source.replace(/(.*)(`?)\>\x20+(.+)/gi, (...props) => {         switch (props[0].trim()[0]) {             case '>': return `<blockquote>${props[3]}</blockquote>`;             default: return props[0];         }     }); } 4. 处理标题

匹配必须以#开头,并且#的数量不能超过6,因为h6是最大的了,没有h7,最后props[2]是#后跟随的文本。

const formatTitle = (source) => {     return source.replace(/(.*#+)\x20?(.*)/g, (...props) => {         switch (props[0][0]) {             case '#': if (props[1].length <= 6) {                 return `<h${props[1].length}>${props[2].trim()}</h${props[1].length}>`;             };             default: return props[0];         }     }) } 5. 处理字体

写的开始复杂了

const formatFont = (source) => {     // 处理 ~ 包裹的文本     source = source.replace(/([`\\]*\~{2})(.*?)\~{2}/g, (...props) => {         switch (props[0].trim()[0]) {             case '~': return `<del>${props[2]}</del>`;;             default: return props[0];         }     });     // 处理 * - 表示的换行     source = source.replace(/([`\\]*)[* -]{3,}\n/g, (...props) => {         switch (props[0].trim()[0]) {             case '*': ;             case '-': return `<hr />`;             default: return props[0];         }     })     // 处理***表示的加粗或者倾斜。     source = source.replace(/([`\\]*\*{1,3})(.*?)(\*{1,3})/g, (...props) => {         switch (props[0].trim()[0]) {             case '*': if (props[1] === props[3]) {                 if (props[1].length === 1) {                     return `<em>${props[2]}</em>`;;                 } else if (props[1].length === 2) {                     return `<strong>${props[2]}</strong>`;;                 } else if (props[1].length === 3) {                     return `<strong><em>${props[2]}</em></strong>`;;                 }             };             default: return props[0];         }     });     return source; } 6. 处理代码块

使用正则匹配使用`包裹的代码块,props[1]是开头`的数量,props[5]是结尾`的数量,必须相等才生效。

const pre = (source) => {     source = source.replace(/([\\`]+)(\w+(\n))?([^!`]*?)(`+)/g, (...props) => {         switch (props[0].trim()[0]) {             case '`': if (props[1] === props[5]) {                 return `<pre>${props[3] || ''}${props[4]}</pre>`;             };             default: return props[0];         }     });     return source; } 7. 处理列表

这里只是处理了ul无序列表,写的同样很麻烦。主要我的思路是真复杂。而且bug肯定也不少。先匹配-+*加上空格,然后根据这一行前面的空格熟替换为ul。这样每一行都保证被ulli包裹。

第二步判断相邻ul之间相差的个数,如果相等则表示应该是同一个ul的li,替换掉</ul><ul>为空,如果后一个ul大于前一个ul,则表示后面有退格,新生成一个<ul>包裹退格后的li,如果是最后一个ul则补齐前面所有的</ul>。

const list = (source) => {     source = source.replace(/.*?[\x20\t]*([\-\+\*]{1})\x20(.*)/g, (...props) => {         if (/^[\t\x20\-\+\*]/.test(props[0])) {             return props[0].replace(/([\t\x20]*)[\-\+\*]\x20(.*)/g, (...props) => {                 const len = props[1].length || '';                 return `<ul${len}><li>${props[2]}</li></ul${len}>`;             })         } else {             return props[0];         }     });     const set = new Set();     source = source.replace(/<\/ul(\d*)>(\n<ul(\d*)>)?/g, (...props) => {         set.add(props[1]);         if (props[1] == props[3]) {             return '';         } else if (props[1] < props[3]) {             return '<ul>';         } else {             const arr = [...set];             const end = arr.indexOf(props[1]);             let start = arr.indexOf(props[3]);             if (start > 0) {                 return '</ul>'.repeat(end - start);             } else {                 return '</ul>'.repeat(end + 1);             }                     }     });     return source.replace(/<(\/?)ul(\d*)>/g, '<$1ul>'); } 8. 处理表格 const table = (source) => {     source = source.replace(/\|.*\|\n\|\s*-+\s*\|.*\|\n/g, (...props) => {         let str = '<table><tr>';         const data = props[0].split(/\n/)[0].split('|');         for (let i = 1; i < data.length - 1; i++) {             str += `<th>${data[i].trim()}</th>`         }         str += '<tr></table>';         return str;     });     return formatTd(source); } const formatTd = (source) => {     source = source.replace(/<\/table>\|.*\|\n/g, (...props) => {         let str = '<tr>';         const data = props[0].split('|');         for (let i = 1; i < data.length - 1; i++) {             str += `<td>${data[i].trim()}</td>`         }         str += '<tr></table>';         return str;     });     if (source.includes('</table>|')) {         return formatTd(source);     }     return source; } 9. 调用方法 const getHtml = (source) => {     source = imageora(source);     source = block(source);     source = formatTitle(source);     source = formatFont(source);     source = pre(source);     source = list(source);     source = table(source);     return source; } const result = getHtml(source); console.log(result);

到此这篇关于JavaScript+Node.js写一款markdown解析器的文章就介绍到这了,更多相关写一款markdown解析器内容请搜索易知道(ezd.cc)以前的文章或继续浏览下面的相关文章希望大家以后多多支持易知道(ezd.cc)!

推荐阅读

    js设置div的边框|怎样给div设置边框

    js设置div的边框|怎样给div设置边框,,1. 怎样给div设置边框1、首先新建一个html文件,输入基本的内容,这里设置一个div,并把它的class设置为de

    js设置样式|js设置样式类

    js设置样式|js设置样式类,,js设置样式    javascript改变CSS样式分为局部和全局,分别如下:  一、局部改变样式    有三种方法:直接

    js用代码实现简单购物车

    js用代码实现简单购物车,,图: 选择所有按钮: 复制代码代码如下所示: 选择 笔记本电脑:3000元 笔记本电脑:3000元 笔记本电脑:3000元 笔记本电脑:3

    js设置背景色|js设置颜色

    js设置背景色|js设置颜色,,js设置背景色首先通过js定位到div的子元素,再通过setatteibute方法给属性添加背景色。js设置颜色js改变字体的颜

    Safari调试iOS中的js

    Safari调试iOS中的js,页面,设备,概述对于HTML5的开发,大家都知道Chrome的DevTools工具有强大的功能和友好的用户体验,不仅能快速方便调试Jav

    Bootstrap的js插件之模态框|modal

    Bootstrap的js插件之模态框|modal,模态,饭盒,.modal——指明div元素包裹模态框;.fade——给模态框添加淡入淡出效果;.modal-dialog——包裹

    js默认事件汇总

    js默认事件汇总,事件,表单,默认事件   就是浏览器通过HTML标签或DOM元素提供的一些功能性的默认行为。比如在a标签href属性上的跳转,右键

    js获取元素宽度

    js获取元素宽度,内联,样式,js获取元素宽度1、使用内联样式,即直接把CSS写在HTML元素的style属性中<div > </div>复制代码通过以下js代码