最近在搭我的博客网站,须要一个markdown
编辑器,来进行博客的编写
看了网上的教程,决定使用simplemde
觉得能够直接能拿来用的
不过实际运用的时候发现仍是有要完善的地方
好比使人头疼的图片上传html
npm install simplemde --save
前端
在html中加入一个textarea<textarea id="simplemde"></textarea>
vue
在vue的生命周期函数mounted
中,添加simplemde
的实例化node
var simplemde = new SimpleMDE({ el: document.getElementById(simplemde) })
el
经过dom指定为咱们创建的textarea元素,若是省略,则会自动抓取html结构中的第一个textareaios
绑定事件,使咱们的内容数据始终与simplemde
获取到的键入数据同步npm
simplemde.codemirror.on("change", () => { this.content = simplemde.value() })
在本来的simplemde
中
点击图片按钮的效果是这样的
这是个啥??本地上传的选择框呢??axios
没办法,既然没有就只好本身作一个了后端
首先咱们创建一个隐藏的input
markdown
<input style="display:none" accept="image/gif,image/jpeg,image/jpg,image/png" type="file" id="upInput" ref="upInput">
接收图片格式的文件,点击便可弹出本地上传的文件选择框
之因此要隐藏,是由于咱们并不想要这个按钮
咱们仍是想经过点击simplemde
的图片按钮来上传
虽然人家没啥用,但好看呀app
因此咱们就把这个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=`` 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=`` simplemde.value(`${this.content}\n${urlname}\n`) }) })
这样就实现了咱们的图片上传效果
好比经过编辑器,咱们写了一篇博客,并存储进了后台
想在别的组件中把它调出来显示
也就是字符串转为html
只须要调用simplemde
的原型链方法
this.contentMarkdown = SimpleMDE.prototype.markdown(content)
而后把数据放到v-html中
<div v-html="contentMarkdown"></div>
便可显示
再看一遍最终效果
还有一些能够完善的地方好比直接拖拽图片进来,或者粘贴后期有时间研究一下再加上