vue前端富文本组件 quill-editor 粘贴图片的处理

页面组件:

<template>
  <div>
    <!-- 图片上传组件辅助 -->
    <el-upload
      class="avatar-uploader quill-img"
      :action="uploadImgUrl"
      name="file"
      :headers="headers"
      :show-file-list="false"
      :on-success="quillImgSuccess"
      :on-error="uploadError"
      :before-upload="quillImgBefore"
      accept='.jpg,.jpeg,.png,.gif'
    ></el-upload>

    <!-- 富文本组件 -->
    <quill-editor
      class="editor"
      v-model="content"
      ref="quillEditor"
      :options="editorOption"
      @blur="onEditorBlur($event)"
      @focus="onEditorFocus($event)"
      @change="onEditorChange($event)"
    ></quill-editor>
  </div>
</template>

组件配置:

editorOption: {
        theme: "snow", // or 'bubble'
        placeholder: "请输入内容",
        modules: {
          clipboard:{
            // 粘贴板,处理粘贴图片  *** 主要是这里
            matchers: [[Node.ELEMENT_NODE, this.desMatcher]],
          },
          toolbar: {
            container: this.message ? this.message : toolbarOptions,
            handlers: {
              image: function(value) {
                if (value) {
                  // 触发input框选择图片文件
                  document.querySelector(".quill-img input").click();
                } else {
                  this.quill.format("image", false);
                }
              }
            }
          },

        }
      },

监听粘贴板事件处理:

//添加匹粘贴板事件 粘贴处理
    desMatcher(node, Delta) {
      let ops = []
      Delta.ops.forEach(op => {
        if (op.insert && typeof op.insert === 'string') {// 如果粘贴了图片,这里会是一个对象,如果是字符串则可以添加
          ops.push({
            insert: op.insert,
          })
        }else{ //给出提醒
          if(op.insert.image){
            // 只给出提醒
            // this.$message.error('图片不可粘贴,请使用图片上传按钮哦!');

            // 处理上传图片
            if (op.insert.image.includes(';base64')) {
              let file = this.base64ToFile(op.insert.image, op.insert.image.name)
              var formData = new FormData();
              formData.append('file',file)
              uploadImagesFile(formData).then(res => {
                console.log("res", res);
                let quill = this.$refs.quillEditor.quill;
                if (res.code === 200) {
                  let length = quill.getSelection().index; //光标位置
                  quill.insertEmbed(length, "image", res.url); // 插入图片  图片地址
                  quill.setSelection(length + 1); //光标后移一位   调整光标到最后
                }
              }).catch(err => {
                console.error(err);
              });
            } else {
              ops.push({
                insert: op.insert,
              })
            }
          }
        }
      })
      Delta.ops = ops
      return Delta
    },

    base64ToFile(base64, fileName) {
      let data = base64.split(',');
      let type = data[0].match(/:(.*?);/)[1];
      let suffix = type.split('/')[1];
      const bstr = window.atob(data[1]);
      let n = bstr.length
      const u8arr = new Uint8Array(n)
      while (n--) {
        u8arr[n] = bstr.charCodeAt(n)
      }
      const file =  new File([u8arr], `${fileName}.${suffix}`, {
        type: type
      })
      return file;
    },

全部代码:

<template>
  <div>
    <!-- 图片上传组件辅助 -->
    <el-upload
      class="avatar-uploader quill-img"
      :action="uploadImgUrl"
      name="file"
      :headers="headers"
      :show-file-list="false"
      :on-success="quillImgSuccess"
      :on-error="uploadError"
      :before-upload="quillImgBefore"
      accept='.jpg,.jpeg,.png,.gif'
    ></el-upload>

    <!-- 富文本组件 -->
    <quill-editor
      class="editor"
      v-model="content"
      ref="quillEditor"
      :options="editorOption"
      @blur="onEditorBlur($event)"
      @focus="onEditorFocus($event)"
      @change="onEditorChange($event)"
    ></quill-editor>
  </div>
</template>

<script>
import { getToken } from '@/utils/auth'

// 工具栏配置
const toolbarOptions = [
  ["bold", "italic", "underline", "strike"],       // 加粗 斜体 下划线 删除线
  // ["blockquote", "code-block"],                    // 引用  代码块
  // [{ list: "ordered" }, { list: "bullet" }],       // 有序、无序列表
  // [{ indent: "-1" }, { indent: "+1" }],            // 缩进
  // [{ size: ["small", false, "large", "huge"] }],   // 字体大小
  [{ header: [1, 2, 3, 4, 5, 6, false] }],         // 标题
  // [{ color: [] }, { background: [] }],             // 字体颜色、字体背景颜色
  // [{ align: [] }],                                 // 对齐方式
  ["clean"],                                       // 清除文本格式
  //["link", "image", "video"]                     // 链接、图片、视频
  ["image"]                                        // 图片
];

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 {uploadImagesFile} from "@/utils/uploadImg";

export default {
  props: {
    /* 编辑器的内容 */
    value: {
      type: String
    },
    /* 图片大小 */
    maxSize: {
      type: Number,
      default: 4000 //kb
    },
    message: Array
  },
  components: { quillEditor },
  data() {
    return {
      content: this.value,
      editorOption: {
        theme: "snow", // or 'bubble'
        placeholder: "请输入内容",
        modules: {
          clipboard:{
            // 粘贴板,处理粘贴图片  *** 主要是这里
            matchers: [[Node.ELEMENT_NODE, this.desMatcher]],
          },
          toolbar: {
            container: this.message ? this.message : toolbarOptions,
            handlers: {
              image: function(value) {
                if (value) {
                  // 触发input框选择图片文件
                  document.querySelector(".quill-img input").click();
                } else {
                  this.quill.format("image", false);
                }
              }
            }
          },

        }
      },
      uploadImgUrl: process.env.VUE_APP_BASE_API + "/common/uploadImagesFile", // 上传的图片服务器地址
      headers: {
        Authorization: 'Bearer ' + getToken()
      }
    };
  },
  watch: {
    value: function() {
      this.content = this.value;
    }
  },
  methods: {
    onEditorBlur() {
      //失去焦点事件
    },
    onEditorFocus() {
      //获得焦点事件
    },
    onEditorChange() {
      //内容改变事件
      this.$emit("input", this.content);
    },

    //添加匹粘贴板事件 粘贴处理
    desMatcher(node, Delta) {
      let ops = []
      Delta.ops.forEach(op => {
        if (op.insert && typeof op.insert === 'string') {// 如果粘贴了图片,这里会是一个对象,如果是字符串则可以添加
          ops.push({
            insert: op.insert,
          })
        }else{ //给出提醒
          if(op.insert.image){
            // 只给出提醒
            // this.$message.error('图片不可粘贴,请使用图片上传按钮哦!');

            // 处理上传图片
            if (op.insert.image.includes(';base64')) {
              let file = this.base64ToFile(op.insert.image, op.insert.image.name)
              var formData = new FormData();
              formData.append('file',file)
              uploadImagesFile(formData).then(res => {
                console.log("res", res);
                let quill = this.$refs.quillEditor.quill;
                if (res.code === 200) {
                  let length = quill.getSelection().index; //光标位置
                  quill.insertEmbed(length, "image", res.url); // 插入图片  图片地址
                  quill.setSelection(length + 1); //光标后移一位   调整光标到最后
                }
              }).catch(err => {
                console.error(err);
              });
            } else {
              ops.push({
                insert: op.insert,
              })
            }
          }
        }
      })
      Delta.ops = ops
      return Delta
    },

    base64ToFile(base64, fileName) {
      let data = base64.split(',');
      let type = data[0].match(/:(.*?);/)[1];
      let suffix = type.split('/')[1];
      const bstr = window.atob(data[1]);
      let n = bstr.length
      const u8arr = new Uint8Array(n)
      while (n--) {
        u8arr[n] = bstr.charCodeAt(n)
      }
      const file =  new File([u8arr], `${fileName}.${suffix}`, {
        type: type
      })
      return file;
    },


    // 富文本图片上传前
    quillImgBefore(file) {
      let fileType = file.type;
			if(fileType === 'image/jpeg' || fileType === 'image/png'){
				return true;
			}else {
				this.$message.error('请插入图片类型文件(jpg/jpeg/png)');
				return false;
			}
    },

    quillImgSuccess(res, file) {
      // res为图片服务器返回的数据
      // 获取富文本组件实例
      let quill = this.$refs.quillEditor.quill;
      // 如果上传成功
      if (res.code == 200) {
        // 获取光标所在位置
        let length = quill.getSelection().index;
        // 插入图片  res.url为服务器返回的图片地址
        quill.insertEmbed(length, "image", res.url);
        // 调整光标到最后
        quill.setSelection(length + 1);
      } else {
        this.$message.error("图片插入失败");
      }
    },
    // 富文本图片上传失败
    uploadError() {
      // loading动画消失
      this.$message.error("图片插入失败");
    }
  }
};
</script>

<style>
.editor {
  line-height: normal !important;
  height: 350px;
  padding-bottom: 50px;
}
.quill-img {
  display: none;
}
.ql-snow .ql-tooltip[data-mode="link"]::before {
  content: "请输入链接地址:";
}
.ql-snow .ql-tooltip.ql-editing a.ql-action::after {
  border-right: 0px;
  content: "保存";
  padding-right: 0px;
}

.ql-snow .ql-tooltip[data-mode="video"]::before {
  content: "请输入视频地址:";
}

.ql-snow .ql-picker.ql-size .ql-picker-label::before,
.ql-snow .ql-picker.ql-size .ql-picker-item::before {
  content: "14px";
}
.ql-snow .ql-picker.ql-size .ql-picker-label[data-value="small"]::before,
.ql-snow .ql-picker.ql-size .ql-picker-item[data-value="small"]::before {
  content: "10px";
}
.ql-snow .ql-picker.ql-size .ql-picker-label[data-value="large"]::before,
.ql-snow .ql-picker.ql-size .ql-picker-item[data-value="large"]::before {
  content: "18px";
}
.ql-snow .ql-picker.ql-size .ql-picker-label[data-value="huge"]::before,
.ql-snow .ql-picker.ql-size .ql-picker-item[data-value="huge"]::before {
  content: "32px";
}

.ql-snow .ql-picker.ql-header .ql-picker-label::before,
.ql-snow .ql-picker.ql-header .ql-picker-item::before {
  content: "文本";
}
.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="1"]::before,
.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="1"]::before {
  content: "标题1";
}
.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="2"]::before,
.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="2"]::before {
  content: "标题2";
}
.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="3"]::before,
.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="3"]::before {
  content: "标题3";
}
.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="4"]::before,
.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="4"]::before {
  content: "标题4";
}
.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="5"]::before,
.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="5"]::before {
  content: "标题5";
}
.ql-snow .ql-picker.ql-header .ql-picker-label[data-value="6"]::before,
.ql-snow .ql-picker.ql-header .ql-picker-item[data-value="6"]::before {
  content: "标题6";
}

.ql-snow .ql-picker.ql-font .ql-picker-label::before,
.ql-snow .ql-picker.ql-font .ql-picker-item::before {
  content: "标准字体";
}
.ql-snow .ql-picker.ql-font .ql-picker-label[data-value="serif"]::before,
.ql-snow .ql-picker.ql-font .ql-picker-item[data-value="serif"]::before {
  content: "衬线字体";
}
.ql-snow .ql-picker.ql-font .ql-picker-label[data-value="monospace"]::before,
.ql-snow .ql-picker.ql-font .ql-picker-item[data-value="monospace"]::before {
  content: "等宽字体";
}
</style>

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
vue-quill-editor粘贴图片的实现思路是将复制的图片上传至服务器,并将服务器返回的图片地址插入到富文本编辑器中。首先,在mounted方法中添加一个监听paste事件的事件处理函数。当粘贴事件发生时,判断粘贴的内容中是否包含图片文件。如果包含图片文件,则将图片文件以FormData的形式进行上传。上传成功后,获取到服务器返回的图片地址,然后将该地址插入到富文本编辑器的光标位置之后,实现粘贴图片的功能。下面是一个具体的实现代码示例: ```javascript mounted() { let quill = this.$refs.myQuillEditor.quill; this.$forceUpdate(); quill.root.addEventListener("paste", (evt) => { if (evt.clipboardData && evt.clipboardData.files && evt.clipboardData.files.length) { evt.preventDefault(); [].forEach.call(evt.clipboardData.files, (file) => { if (!file.type.match(/^image\/(gif|jpe?g|a?png|bmp)/i)) { return; } const formData = new FormData(); formData.append("pictureFile", file); makdwnImg(formData) .then((res) => { let quill = this.$refs.myQuillEditor.quill; if (res.data.code == 200) { let length = quill.getSelection().index; quill.insertEmbed(length, "image", res.data.data); quill.setSelection(length + 1); } }) .catch((err) => { console.error(err); }); }); } }, false); }, ``` 在这个示例代码中,首先获取到vue-quill-editor实例的quill对象。然后,使用addEventListener方法监听paste事件,并在事件处理函数中判断粘贴的内容是否包含图片文件。如果包含图片文件,则使用FormData对象将文件进行上传,并在上传成功后获取到服务器返回的图片地址。最后,使用quill对象的insertEmbed方法将图片地址插入到富文本编辑器的光标位置之后,并调整光标位置到插入图片之后的位置。这样就实现了vue-quill-editor粘贴图片的功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值