vue-quill-editor二次封装,实现自定义文件上传方式

 更新时间:2024年03月07日 09:24:06   作者:jsmeng626  
这篇文章主要介绍了vue-quill-editor二次封装,实现自定义文件上传方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

实现步骤

  • 先将vue-quill-editor组件引入进来,自定义配置,在自定义配置中添加upload操作,但是由于vue-quill-editor组件不支持文件上传,该操作是无效的,逻辑需要自己去实现
  • 给添加的upload按钮自定义css样式,文件上传的图标,大小等
  • 模拟upload点击事件,在editor的配置中,有一个handler对象,来处理自定义的upload点击
  • 点击之后我们去让他触发element-ui中的文件上传组件的上传操作,所以我们将el-upload组件放在页面上,然后用css隐藏起来,在点击富文本中upload图标时,来模拟点击el-upload,弹出上传文件操作框
  • 然后自定义el-upload的上传方法,在上传后的回调中去调用自己的接口,拿到一个文件路径字符串
  • 再将这个字符串通过quill对象插入到富文本中,也就是vue-quill-editor组件依赖的父对象,掺入a链接是通过继承父类中的create方法来实现
  • 其他不明白的可以看代码中的注释

完整代码

1、quill-editor-upload自定义组件

<template>
  <div>
    <quill-editor ref="quillEditor" :options="editorOption" :content="content" @change="onEditorChange($event)"
      @blur="$emit('blur')"></quill-editor>
    <!-- 隐藏的upload -->
    <el-upload class="uploadFile" drag action="#" :http-request="uploadFile" :limit="1" :on-exceed="handleExceed"
      :on-remove="handleRemove" :file-list="fileList">
      <i class="el-icon-upload"></i>
      <div class="el-upload__text">
        将文件拖到此处,或<em>点击上传</em>
      </div>
    </el-upload>
  </div>
</template>
<script>
import Quill from 'quill';
import { quillEditor } from "vue-quill-editor";
import "quill/dist/quill.core.css";
import "quill/dist/quill.snow.css";
import "quill/dist/quill.bubble.css";
import { editorUploadPath } from "@/api/excel/excel.js";
 
// 工具栏配置
const toolbarOptions = [
  ['bold', 'italic', 'underline', 'strike'],        // toggled buttons
  ['blockquote', 'code-block'],
 
  [{ 'header': 1 }, { 'header': 2 }],               // custom button values
  [{ 'list': 'ordered' }, { 'list': 'bullet' }],
  [{ 'script': 'sub' }, { 'script': 'super' }],      // superscript/subscript
  [{ 'indent': '-1' }, { 'indent': '+1' }],          // outdent/indent
  [{ 'direction': 'rtl' }],                         // text direction
 
  [{ 'size': ['small', false, 'large', 'huge'] }],  // custom dropdown
  [{ 'header': [1, 2, 3, 4, 5, 6, false] }],
 
  [{ 'color': [] }, { 'background': [] }],          // dropdown with defaults from theme
  [{ 'font': [] }],
  [{ 'align': [] }],
  ['link', 'image', 'video', 'upload'],
  ['clean']                                         // remove formatting button
]
 
// 自定义插入a链接
var Link = Quill.import('formats/link');
class FileBlot extends Link {  // 继承Link Blot
  static create(value) {
    let node = undefined
    if (value && !value.href) {  // 适应原本的Link Blot
      node = super.create(value);
    }
    else {  // 自定义Link Blot
      node = super.create(value.href);
      node.setAttribute('download', value.innerText);  // 左键点击即下载
      node.innerText = value.innerText;
      node.download = value.innerText;
    }
    return node;
  }
}
FileBlot.blotName = 'link';
FileBlot.tagName = 'A';
Quill.register(FileBlot);
 
export default {
  name: "quill-editor-upload",
  components: { quillEditor },
  // 自定义v-model,prop为props中的属性,代表v-model传入的值;event表示反向输出给v-model的事件名
  model: {
    prop: 'content',
    event: 'changeContent'
  },
  // 自定义v-model, v-model传入的值
  props: {
    content: {
      type: String,
      default: ''
    }
  },
  data() {
    return {
      fileList: [], // 文件列表
      // editor配置
      editorOption: {
        placeholder: '请输入正文......',
        modules: {
          toolbar: {
            container: toolbarOptions,  // 工具栏
            handlers: {
              // 模拟upload按钮点击事件
              'upload': (value => {
                if (value) {
                  document.querySelector('.uploadFile input').click()
                }
              })
            }
          }
        }
      }
    }
  },
  methods: {
    // 自定义v-model,将editor选择器change事件提供的html传出去
    onEditorChange(e) {
      this.$emit('changeContent', e.html)
    },
    // 自定义文件上传
    uploadFile(res) {
      // console.log('上传成功', res);
      let myFormData = new FormData();
      myFormData.append("file", res.file);
      editorUploadPath(myFormData).then((response) => {
        if (response.code === 0) {
          let fileNameLength = res.file.name.length
          // 插入链接
          let quill = this.$refs.quillEditor.quill
          let length = quill.getSelection().index;
          quill.insertEmbed(length, 'link', { href: response.msg, innerText: res.file.name }, "api")
          quill.setSelection(length + fileNameLength)
          this.fileList = []
        }
      });
    },
    // 文件超出限制提示
    handleExceed() {
      this.$message.warning(`当前限制选择 1 个文件,请删除后更新!`);
    },
    // 移除文件
    handleRemove() {
      this.fileList = [];
    },
  }
}
</script>
<style scoped>
/* 文件上传图标样式 */
/deep/ .quill-editor .ql-snow.ql-toolbar .ql-formats .ql-upload {
  background: url("../assets/file.png");
  background-size: 16px 16px;
  background-position: center center;
  background-repeat: no-repeat;
}
 
/* 隐藏upload上传,用模拟事件触发 */
.uploadFile {
  width: 0;
  height: 0;
  display: none;
}
</style>

2、使用该组件

<el-form-item label="内容" prop="content">
    <quillEditorUpload v-model="form.content"></quillEditorUpload>
</el-form-item>

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • vue项目中data数据之间互相访问的实现

    vue项目中data数据之间互相访问的实现

    本文主要介绍了vue项目中data数据之间互相访问的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • 在Vue中使用antv的示例代码

    在Vue中使用antv的示例代码

    这篇文章主要介绍了在Vue中使用antv的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-06-06
  • Element使用el-table组件二次封装

    Element使用el-table组件二次封装

    这篇文章主要为大家介绍了Element使用el-table组件二次封装示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • LogicFlow内置插件使用实例

    LogicFlow内置插件使用实例

    这篇文章主要为大家介绍了LogicFlow内置插件使用实例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • Vue实现双向绑定的方法

    Vue实现双向绑定的方法

    这篇文章主要介绍了Vue实现双向绑定的方法,了解vue的双向数据绑定原理以及核心代码模块,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • vue-router钩子函数实现路由守卫

    vue-router钩子函数实现路由守卫

    这篇文章主要介绍了vue-router钩子函数实现路由守卫,对vue感兴趣的同学,可以参考下
    2021-04-04
  • vue如何在引入的el-tree前添加图标

    vue如何在引入的el-tree前添加图标

    这篇文章主要介绍了vue如何在引入的el-tree前添加图标问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-03-03
  • vue2创建高复用组件的方法示例

    vue2创建高复用组件的方法示例

    Vue2中的高复用组件通常是指那些设计得足够通用,并能多次在项目中重复使用的组件,本文给大家详细介绍了vue2创建高复用组件的方法示例,并通过代码示例讲解的非常详细,需要的朋友可以参考下
    2024-07-07
  • 动态Axios的配置步骤详解

    动态Axios的配置步骤详解

    这篇文章主要给大家分享介绍了关于动态Axios的配置步骤,文中通过示例代码介绍的非常详细,通过这个教程大家可以很方便的实现动态Axios的配置,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧。
    2018-01-01
  • 解决vue里a标签值解析变量,跳转页面,前面加默认域名端口的问题

    解决vue里a标签值解析变量,跳转页面,前面加默认域名端口的问题

    这篇文章主要介绍了解决vue里a标签值解析变量,跳转页面,前面加默认域名端口的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-07-07

最新评论