利用Vue实现一个markdown编辑器实例代码

作者:九路313 时间:2024-04-30 10:39:19 

前言

前段时间做项目的时候,需要一个Markdown编辑器,在网上找了一些开源的实现,但是都不满足需求

说实话,这些开源项目也很难满足需求公司项目的需求,与其实现一个大而全的项目,倒不如实现一个简单的,易于在源码上修改的项目,核心功能都有的,以供修改使用

本文的源码地址如下:https://github.com/jiulu313/HelloMarkDown(本地下载)

喜欢的朋友可以帮忙star一下,欢迎交流学习

先看一下本项目的效果图(图片经过压缩)

利用Vue实现一个markdown编辑器实例代码

本文的目的就是实现一个有核心功能的,简单,易于修改的项目

话不多说,来看思路

1 markdown内容如何转换成 html?

网上有一个开源的库叫 marked,地址如下:https://github.com/markedjs/marked.git

我们可以安装这个库,使用很简单,就一个函数,传进去markdown内容,就返回了html内容

2 markdown内容转换成了html,如何进行语法高亮?

网上也有一个开源的库,地址如下 :https://highlightjs.org/

我们可以使用这两个库

先把markdown内容解析成html内容

把html内容进行语法高亮

下面我们来一步一步实现代码

3 代码实现

默认你已经创建好了vue的项目 , 创建vue项目 vue init webpack demo
这里面不多讲。

3.1 安装两个库,分别执行下面两条命令


npm install marked --save
npm install highlight.js --save

3.2 首先创建一个 HelloMarkDown 的 Vue组件

布局文件的代码如下:


<template>
<div class="md_root_content" v-bind:style="{width:this.width,height: this.height}">

<!--功能按钮区-->
<div class="button_bar">
<span v-on:click="addBold"><B>B</B></span>
<span v-on:click="addUnderline"><B>U</B></span>
<span v-on:click="addItalic"><B>I</B></span>
</div>

<!--主要内容区-->
<div class="content_bar">

<!--markdown编辑器区-->
<div class="markdown_body">
<textarea ref="ref_md_edit" class="md_textarea_content" v-model="markString">
</textarea>
</div>

<!--解析成html区-->
<div class="html_body">
<p v-html="htmlString"></p>
</div>

</div>

</div>
</template>

主要分为上下两块,上面是功能区的布局

下面一块,分左右两部分,左边是markdown,右边是显示html部分

对应的样式代码如下:


<style scoped>

.md_root_content {
display: flex;
display: -webkit-flex;
flex-direction: column;
}

.button_bar {
width: 100%;
height: 40px;
background-color: #d4d4d4;
display: flex;
display: -webkit-flex;
align-items: center;
}

div.button_bar span {
width: 30px;
line-height: 40px;
text-align: center;
color: orange;
cursor: pointer;
}

.content_bar {
display: flex;
display: -webkit-flex;
width: 100%;
height: 100%;
}

.markdown_body {
width: 50%;
height: 100%;
display: flex;
display: -webkit-flex;
}

.html_body {
width: 50%;
height: 100%;
display: flex;
display: -webkit-flex;
background-color: #dfe9f1;
}

.md_textarea_content {
flex: 1;
height: 100%;
padding: 12px;
overflow: auto;
box-sizing: border-box;
resize: none;
outline: none;
border: none;
background-color: #f4f4f4;
font-size: 14px;
color: #232323;
line-height: 24px;
}
</style>

业务逻辑部分的代码如下:


<script>
import marked from 'marked' //解析mardown语法的库
import hljs from 'highlight.js' //对代码进行语法高亮的库

import testData from '../testData' //测试数据

export default {
name: "HelloMarkDown",

props: {
width: {
type: String,
default: '1000px'
},

height: {
type: String,
default: '600px'
}
},

data() {
return {
markString: '',
htmlString: '',
}
},

mounted(){
this.markString = testData
},

methods: {
//加粗
addBold() {
this.changeSelectedText("**","**")
},

//斜体
addItalic() {
this.changeSelectedText("***","***")
},

addUnderline() {
this.changeSelectedText("<u>","</u>")
},

changeSelectedText(startString,endString){
let t = this.$refs.ref_md_edit
if (window.getSelection) {
 if (t.selectionStart != undefined && t.selectionEnd != undefined) {

let str1 = t.value.substring(0, t.selectionStart)
 let str2 = t.value.substring(t.selectionStart, t.selectionEnd)
 let str3 = t.value.substring(t.selectionEnd)

let result = str1 + startString + str2 + endString + str3
 t.value = result
 this.markString = t.value
 }
}
}
},

watch: {

//监听markString变化
markString: function (value) {
marked.setOptions({
 renderer: new marked.Renderer(),
 gfm: true,
 tables: true,
 breaks: true,
 pedantic: false,
 sanitize: false,
 smartLists: true,
 smartypants: false
})

this.htmlString = marked(value)
},

//监听htmlString并对其高亮
htmlString: function (value) {
this.$nextTick(() => {
 const codes = document.querySelectorAll(".html_body pre code");

// elem 是一个 object
 codes.forEach(elem => {
 elem.innerHTML = "<ul><li>" + elem.innerHTML.replace(/\n/g, "\n</li><li>") + "\n</li></ul>"
 hljs.highlightBlock(elem);
 });
});
}
}

}
</script>

script中的代码解释


props: {
width: {
type: String,
default: '1000px'
},

height: {
type: String,
default: '600px'
}
},

width: 组件的宽度

height:组件的高度


data() {
return {
markString: '',
htmlString: '',
}
},

markString:保存我们输入的markdown内容

htmlString:保存markdown内容转换成的html内容,也就是通过marked函数转换过来的


mounted(){
 this.markString = testData
},

显示默认数据


//加粗
 addBold() {
 this.changeSelectedText("**","**")
 },

//斜体
 addItalic() {
 this.changeSelectedText("***","***")
 },

//加下划线
 addUnderline() {
 this.changeSelectedText("<u>","</u>")
 },

这三个函数都是调用了 changeSelectedText 函数

主要是对鼠标选中的内容进行改变,比如加粗效果,是在选中文本的两边分别添加 **

所以changeSelectedText函数的作用就是在选中的文本两边添加不同的md的符号

比如

this.changeSelectedText("","") ,就是在选中的文本左边和右边都添加**

然后再把最新的内容赋值给 this.$refs.ref_md_edit.value,同时也两会给markString

这样就可以做到选中文本加粗效果了


//监听markString变化
 markString: function (value) {
 marked.setOptions({
  renderer: new marked.Renderer(),
  gfm: true,
  tables: true,
  breaks: true,
  pedantic: false,
  sanitize: false,
  smartLists: true,
  smartypants: false
 })

this.htmlString = marked(value)
 },

此时是监听markString的变化

然后调用marked函数进行转换成html内容,并赋值给htmlString

marked.setOptions 是设置一些配置,有兴趣的可以查一下这些配置的作用


//监听htmlString并对其高亮
 htmlString: function (value) {
 this.$nextTick(() => {
  const codes = document.querySelectorAll(".html_body pre code");

// elem 是一个 object
  codes.forEach(elem => {
  elem.innerHTML = "<ul><li>" + elem.innerHTML.replace(/\n/g, "\n</li><li>") + "\n</li></ul>"
  hljs.highlightBlock(elem);
  });
 });
 }

原本通过 highlight.js这个库在显示语法高亮的时候,是没有行号的。这里我进行了扩展

通过 document.querySelectorAll(".html_body pre code") 找到nodeList

然后对其循环,动态添加 ul , li, 这样就可以显示行号了

不过这需要对 highlight的css文件添加几个样式

源码里面我把highlight中的css文件全部copy到项目中了,使用的是github.css

具体位置是在项目中的 assets/markdown/styles/github.css

如果想使用其它的主题,可以自己修改其它的对应的css文件,这里使用了github的主题,所以只修改了github.css这一个文件

有兴趣的可以查看一下

github.css文件的提交记录

具体的思路就是这些,水平有限,难免有bug,如有发现,欢迎提出

来源:104.116.116.112.58.47.47.119.119.119.46.99.110.98.108.111.103.115.46.99.111.109.47.115.116.97.114.116.49.50.50.53.47.112.47.49.48.56.56.55.48.54.49.46.104.116.109.108.

标签:vue,markdown,编辑器
0
投稿

猜你喜欢

  • vue cli+axios踩坑记录+拦截器使用方式,代理跨域proxy

    2023-07-02 16:38:30
  • Pandas读存JSON数据操作示例详解

    2022-05-24 08:14:03
  • node.js回调函数之阻塞调用与非阻塞调用

    2024-05-05 09:21:26
  • 开源MySQL公司停止提供企业版源代码tar包

    2009-01-14 13:02:00
  • javascript json字符串到json对象转义问题

    2023-07-02 05:18:22
  • asp 去掉html中的table正则代码函数

    2011-04-06 10:48:00
  • php设置编码格式的方法

    2023-11-14 12:01:22
  • Python按行读取文件的简单实现方法

    2023-06-14 22:55:49
  • Python深入学习之内存管理

    2021-11-18 12:54:10
  • Python 调用VC++的动态链接库(DLL)

    2023-06-19 09:01:44
  • 短视频(douyin)去水印工具的实现代码

    2024-04-29 13:40:15
  • Flask框架踩坑之ajax跨域请求实现

    2023-12-25 01:28:02
  • 设计上的小细节

    2010-06-24 21:44:00
  • 深入理解Python虚拟机中描述器的实现原理

    2021-08-05 21:20:48
  • vue基于input实现密码的显示与隐藏功能

    2024-05-09 15:28:20
  • Go每日一库之quicktemplate的使用

    2023-06-28 01:03:15
  • 从浅入深带你掌握Golang数据结构map

    2023-06-21 16:42:43
  • 如何使用Python实现名片管理系统

    2022-11-24 03:13:09
  • Python项目跨域问题解决方案

    2022-12-01 17:39:32
  • python+selenium行为链登录12306(滑动验证码滑块)

    2023-01-16 16:08:01
  • asp之家 网络编程 m.aspxhome.com