最近在用uni-app写跨端的小程序APP应用。其中有富文本的需求,后台使用的vue-cli3+elementUI。vue做技术栈的话还是选择quill-editor比较多。

quill-editor上传图片和视频到服务器

quill-editor默认的图片是base64文件,一般来说,我们都需要保存至自己的服务器或者OSS。

quill-editor上传到服务器的关键是对quill-editor的图片和视频点击事件的劫持,劫持之后自定义一个上传的方法。基本配置在这里就不多做记录。可以按照需求查看文档或者百度。

editorOption: {
          theme: "snow", // or 'bubble'
          placeholder: "",
          modules: {
            imageResize: {},
            toolbar: {
              container: toolbarOptions,
              // container: "#toolbar",
              handlers: {
                //拦截默认的上传方式
                image: function(value) {
                    //当点击图片上传时,value会变为true
                  if (value) {
                    // 触发自定义的上传
                    document.querySelector(".avatar-uploader input").click();
                  } else {
                    this.quill.format("image", false);
                  }
                },
               video: function(value) {
                     //当点击视频上传时,value会变为true
                  if (value) {
                     // 触发上传
                    document.querySelector(".video-uploader input").click();
                  } else {
                    this.quill.format("video",false);
                  }
                },
              }
            }
          }
        },

文件上传的方法在这里就不做过多解释。下面说一下如何将上传之后的文件,插入到富文本中

uploadSuccessVideo(res,file){
        // res为文件服务器返回的数据
        // 获取富文本组件实例
        let quill = this.$refs.myQuillEditor.quill;
        // 如果上传成功
        if (res.code === 200) {
          // 使用getSelection来获取光标所在位置
          let length = quill.getSelection().index;
          // 插入“video”或者“image”  第三个参数为服务器端返回的地址
          quill.insertEmbed(length, "video", res.data);
          // 调整光标到最后
          quill.setSelection(length + 1);
          console.log(this.content)
        } else {
          this.$message.error("视频插入失败");
        }
        // loading动画消失
        this.quillUpdateImg = false;
 },

到这里自定义的文件服务器上传就完成了。

quill-image-resize-modules的使用

图片只有上传可远远不够,因为用户的图片大小规格不一,改变图片大小的需求是肯定会存在的。

这个插件在安装之后的引用遇到了许多问题,类似"Cannot read property 'register' of undefined"、"Cannot read property 'imports' of undefined" ,也百度了许多方案,但是很多行不通。这里记录下解决方案。

首先是quill-image-resize-module的引入,安装过程就不多做解释。

//引入
  import "quill/dist/quill.core.css";
  import "quill/dist/quill.snow.css";
  import 'quill/dist/quill.bubble.css';
  import { quillEditor } from 'vue-quill-editor'
  import * as Quill from 'quill'  
  import ImageResize from 'quill-image-resize-module'
  Quill.register('modules/imageResize', ImageResize)

//option配置
editorOption: {
          theme: "snow", // or 'bubble'
          placeholder: "您想说点什么?",
          modules: {
            imageResize: {},//这里就是配置
            toolbar: {
             //...
              }
            }
          }
        },

引入之后,然后运行! 接下来就开始报错,百度查找了。百度上有很多种webpack的配置,下面是本次在vue-cli3中的方案。

在vue.config.js中添加如下代码:(没有此文件的话,在根目录新建一个)

const webpack = require('webpack')

module.exports = {
         ... //这里是一些其他的配置


//关键配置
  configureWebpack: {
    plugins: [
      new webpack.ProvidePlugin({
        'window.Quill': 'quill/dist/quill.js',
        'Quill': 'quill/dist/quill.js'
      }),
    ]
  }
//

};

记住,每次修改vue.config.js之后都需要重新启动 npm run serve 到这里实现图片的缩放了。

uni-app或者小程序的解析

小程序是富文本默认是不解析视频的,需要一些其他的辅助来实现。微信小程序直接百度"wxparse"使用。uni-app的话去插件市场搜索"u-parse"。小程序的video的层级默认是高于普通标签的,如果video覆盖到了其他组件,可以对其他组件调节z-index去解决这个问题。

下边上传封装quill-editor组件,仅供参考配置需要自己修改。

<!--
    引入组件之后  v-model="content" 输出
-->

<template>
    <div class="customEditor">
        <!-- 图片上传组件辅助-->
        <el-upload
                class="avatar-uploader"
                :action="uploadURl"
                name="img"
                accept=".png,.jpeg,.jpg"
                :headers="header"
                :show-file-list="false"
                :on-success="uploadSuccess"
                :on-error="uploadError"
                :before-upload="beforeUpload">
        </el-upload>

        <!-- 视频上传组件辅助-->
        <el-upload
                class="video-uploader"
                :action="uploadURl"
                name="video"
                accept=".mp4,.3gp"
                :headers="header"
                :show-file-list="false"
                :on-success="uploadSuccessVideo"
                :on-error="uploadError"
                :before-upload="beforeUpload">
        </el-upload>

        <quill-editor
                class="editor"
                v-model="content"
                ref="myQuillEditor"
                :options="editorOption"
                @blur="onEditorBlur($event)" @focus="onEditorFocus($event)"
                @change="onEditorChange($event)">
        </quill-editor>
    </div>
</template>

<script>
  const toolbarOptions = [
    ["bold", "italic", "underline", "strike"], // 加粗 斜体 下划线 删除线
    ["blockquote", "code-block"], // 引用  代码块
    [{ header: 1 }, { header: 2 }], // 1、2 级标题
    [{ list: "ordered" }, { list: "bullet" }], // 有序、无序列表
    [{ script: "sub" }, { script: "super" }], // 上标/下标
    [{ indent: "-1" }, { indent: "+1" }], // 缩进
    [{ size: ["small", false, "large", "huge"] }], // 字体大小
    [{ header: [1, 2, 3, 4, 5, 6, false] }], // 标题
    [{ color: [] }, { background: [] }], // 字体颜色、字体背景颜色
    [{ font: [] }], // 字体种类
    [{ align: [] }], // 对齐方式
    ["clean"], // 清除文本格式
    ["link", "image", "video"] // 链接、图片、视频
  ];
  import "quill/dist/quill.core.css";
  import "quill/dist/quill.snow.css";
  import 'quill/dist/quill.bubble.css';
  import { quillEditor } from 'vue-quill-editor'
  import * as Quill from 'quill'
  import ImageResize from 'quill-image-resize-module'
  Quill.register('modules/imageResize', ImageResize)
  import ConfigPath from '@/configs/path.js'
  export default {
    name: 'customEditor',
    props: {
      value: {
        type: String
      }
    },
    components: {
      quillEditor
    },

    data() {
      return {
        content: this.value,
        imgList:[],
        uploadURl:'',//服务其上传的地址
        quillUpdateImg: false,
        editorOption: {
          theme: "snow",
          placeholder: "您想说点什么?",
          modules: {
            imageResize: {},
            toolbar: {
              container: toolbarOptions,
              handlers: {
                image: function(value) {
                  if (value) {
                    document.querySelector(".avatar-uploader input").click();
                  } else {
                    this.quill.format("image", false);
                  }
                },
                video: function(value) {
                  if (value) {
                    document.querySelector(".video-uploader input").click();
                  } else {
                    this.quill.format("video",false);
                  }
                },
              }
            }
          }
        },
        header: {
          // token: sessionStorage.token
        } // 有的图片服务器要求请求头需要有token
      };
    },

    methods: {
      onEditorBlur() {
        //失去焦点事件
      },
      onEditorFocus() {
        //获得焦点事件
      },
      onEditorChange() {
        //内容改变事件
        this.$emit("input", this.content);
      },
      uploadSuccessVideo(res,file){
        // res为图片服务器返回的数据
        // 获取富文本组件实例
        let quill = this.$refs.myQuillEditor.quill;
        // 如果上传成功
        if (res.code == 200) {
          // 获取光标所在位置
          let length = quill.getSelection().index;
          // 插入视频
          quill.insertEmbed(length, "video", res.url);
          // 调整光标到最后
          quill.setSelection(length + 1);
          console.log(this.content)
        } else {
          this.$message.error("视频插入失败");
        }
        // loading动画消失
        this.quillUpdateImg = false;
      },
      // 富文本图片上传前
      beforeUpload() {
        // 显示loading动画
        this.quillUpdateImg = true;
      },
      uploadSuccess(res, file) {
        // res为图片服务器返回的数据
        // 获取富文本组件实例
        let quill = this.$refs.myQuillEditor.quill;
        // 如果上传成功
        if (res.code == 200) {
          // 获取光标所在位置
          this.imgList.push(res.data)
          this.$emit('imgList',this.imgList)
          let length = quill.getSelection().index;
          // 插入图片  res.url为服务器返回的图片地址
          quill.insertEmbed(length, "image", res.url);
          // 调整光标到最后
          quill.setSelection(length + 1);
        } else {
          this.$message.error("图片插入失败");
        }
        // loading动画消失
        this.quillUpdateImg = false;
      },
      // 富文本图片上传失败
      uploadError() {
        // loading动画消失
        this.quillUpdateImg = false;
        this.$message.error("图片插入失败");
      }
    }
  }
</script>

<style lang="scss">
    .customEditor{
        .editor {
            line-height: normal !important;
            height: 400px;
        }
        .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>

Logo

前往低代码交流专区

更多推荐