前言
之前在搭个人博客网站,需要一个markdown
编辑器,来进行博客的编写 看了网上的教程,决定使用simplemde
以为可以直接能拿来用的 不过实际运用的时候发现还是有要完善的地方 比如令人头疼的图片上传
最终效果
- markdown语法
- 实时预览
- 将图片上传服务器
- 最终生成html
安装及初始化
npm install simplemde --save
在html中加入一个textarea <textarea id="simplemde"></textarea>
在vue的生命周期函数mounted
中,添加simplemde
的实例化
var simplemde = new SimpleMDE({
el: document.getElementById(simplemde)
})
复制代码
el
通过dom指定为我们建立的textarea
元素,如果省略,则会自动抓取html
结构中的第一个textarea
绑定事件,使我们的内容数据始终与simplemde
获取到的键入数据同步
simplemde.codemirror.on("change", () => {
this.content = simplemde.value()
})
复制代码
上传图片
在原本的simplemde
中 点击图片按钮的效果是这样的
没办法,既然没有就只好自己做一个了
首先我们建立一个隐藏的input
<input style="display:none" accept="image/gif,image/jpeg,image/jpg,image/png" type="file" id="upInput" ref="upInput">
复制代码
接收图片格式的文件,点击即可弹出本地上传的文件选择框 之所以要隐藏,是因为我们并不想要这个按钮 我们还是想通过点击simplemde
的图片按钮来上传 虽然人家没啥用,但好看呀
所以我们就把这个input
给隐藏,只用一下它的click
方法 这样我们点击图片按钮就相当于在点击这个input
在simplemde
的源码里,找到图片按钮调用的函数 把原来的都注释掉,加上这两句
那么选择了图片之后,为了能即时预览 我们希望选择之后,就发到后端存储起来 在前端我们运用axios
+formdata
进行发送
var input = this.$refs.upInput
var formData = new FormData()
formData.append("i", input.files[0])
var config = {
headers: {
"Content-Type": "multipart/form-data"
}
}
this.$axios.post("/data/myupload", formData, config)
复制代码
后端我是用的node
,运用multer
模块来接收 multer
是专门用来处理mulipart/form-data
格式的数据的
var multer = require('multer')
//定义存储器
var storage = multer.diskStorage({
//存储路径
destination: function (req, file, cb) {
cb(null, '../static/upload/')
},
//存储文件名
filename: function (req, file, cb) {
cb(null, `${Date.now()}-${file.originalname}`)
}
})
//运用存储器
var upload = multer({ storage: storage })
//接受单图的上传
router.post('/data/myupload', upload.single('i'), function (req, res, next) {
//将存储后的文件名发还给前端
res.send(req.file.filename)
});
复制代码
前端收到文件名后,将其跟存储路径打包写进文本框中 也就是之前点击图片按钮看到的那串字符 写入后就可预览
this.$axios.post("/data/myupload", formData, config).then((res)=> {
var urlname=`![](/static/upload/${res.data})`
simplemde.value(`${this.content}\n${urlname}\n`)
})
复制代码
看起来万事大吉了 但其实还漏了一点
那就是input的click()本身不是异步的,但是你选择图片需要时间,在这过程中后面的代码(即便是异步代码)都执行了一遍,也就是说现在写的这些发送存储都在选完图片之前就执行完了
为了在选择完图片之后再执行 我们新增一个监听事件,监听input
的change
,把之前的代码都丢到这里面来
var input = this.$refs.upInput
input.addEventListener("change", () => {
var formData = new FormData()
formData.append("i", input.files[0])
var config = {
headers: {
"Content-Type": "multipart/form-data"
}
}
this.$axios.post("/data/myupload", formData, config).then((res)=> {
var urlname=`![](/static/upload/${res.data})`
simplemde.value(`${this.content}\n${urlname}\n`)
})
})
复制代码
这样就实现了我们的图片上传效果
显示
比如通过编辑器,我们写了一篇博客,并存储进了后台 想在别的组件中把它调出来显示 也就是字符串转为html
只需要调用simplemde
的原型链方法
this.contentMarkdown = SimpleMDE.prototype.markdown(content)
复制代码
然后把数据放到v-html中
<div v-html="contentMarkdown"></div>
复制代码
即可显示
再看一遍最终效果
PS
本文是将图片存进了服务器的项目目录中,其实也可以把图片上传到图床之类的 而额外的一些功能,比如直接拖拽图片进来,或者粘贴。等后期有时间研究一下再加上