前言 自定义文字大小前文已经写过了,业务需要富文本还需要上传视频,所以这里记录一下如何自定义Quill富文本的视频上传。
只拦截视频上传功能是不够的,因为默认插入的是iframe标签,还需要更改为video标签,VideoBlot.js文件的作用就是用于把默认的iframe改为video标签。
官方其实预设了一些Blot,我们可以基于这些进行扩展:
Text Blot(文本) 表示普通的文本内容,是最基本的Blot类型。 Block Blot(块) 表示一个块级元素,如段落、标题等。它可以包含多个Text Blot。 Inline Blot(内联) 表示内联元素,如加粗、斜体、链接等。它可以包含多个Text Blot。 Container Blot(容器) 表示一个容器元素,可以包含其他的Blot类型。常见的容器Blot有List(列表)和Table(表格)。 除了这些基本的Blot类型,Quill还提供了一些特殊的Blot类型,用于处理特定的功能和样式,例如: Embed Blot(嵌入) 表示嵌入式内容,如图片、视频等。Embed Blot可以包含其他的Blot类型,用于表示嵌入内容的各个部分。 扩展视频,我们通过Block Blot就行了,blot有一些预设静态属性,因为它本身是一个class,其中blotName、tagName都是必填的,className可写可不写,写了他就是你配置的tagName元素的类名。
1.新建VideoBlot.js文件 从utils目录下ruoyi-ui/src/utils/VideoBlot.js文件,主要用于自定义嵌入类型的Blot用于替换默认的iframe标签。
import Quill from 'quill' let BlockEmbed = Quill.import('blots/block/embed') class VideoBlot extends BlockEmbed { static create(value) { let node = super.create() node.setAttribute('src', value.url) node.setAttribute('controls', true) node.setAttribute('width', '100%') node.setAttribute('style', 'height: auto; margin: 10px 0;') return node } static value(node) { return { url: node.getAttribute('src') } } } VideoBlot.blotName = 'video' VideoBlot.tagName = 'video' Quill.register(VideoBlot, true) 2.修改Editor组件 文件路径ruoyi-ui/src/components/Editor/index.vue
主要修改了以下几点:
引入VideoBlot.js文件 init() 方法初始化方法拦截video上传按钮,手动触发上传组件 handleBeforeUpload() 方法添加视频格式和提示内容 handleUploadSuccess() 方法添加视频插入逻辑 <template> <div> <el-upload :action="uploadUrl" :before-upload="handleBeforeUpload" :on-success="handleUploadSuccess" :on-error="handleUploadError" name="file" :show-file-list="false" :headers="headers" style="display: none" ref="upload" v-if="this.type == 'url'" > </el-upload> <div class="editor" ref="editor" :style="styles"></div> </div> </template> <script> import Quill from "quill"; import "quill/dist/quill.core.css"; import "quill/dist/quill.snow.css"; import "quill/dist/quill.bubble.css"; import "@/utils/VideoBlot" import { getToken } from "@/utils/auth"; // 定义自定义字号 let fontSize = ['10px', '12px', '14px', '16px', '18px', '20px', '24px', '36px']; // 注册自定义字号 let Size = Quill.import('attributors/style/size'); Size.whitelist = fontSize; Quill.register(Size, true); export default { name: "Editor", props: { /* 编辑器的内容 */ value: { type: String, default: "", }, /* 高度 */ height: { type: Number, default: null, }, /* 最小高度 */ minHeight: { type: Number, default: null, }, /* 只读 */ readOnly: { type: Boolean, default: false, }, /* 上传文件大小限制(MB) */ fileSize: { type: Number, default: 100, }, /* 类型(base64格式、url格式) */ type: { type: String, default: "url", } }, data() { return { uploadUrl: process.env.VUE_APP_BASE_API + "/common/upload", // 上传的图片服务器地址 headers: { Authorization: "Bearer " + getToken() }, Quill: null, currentValue: "", options: { theme: "snow", bounds: document.body, debug: "warn", modules: { // 工具栏配置 toolbar: [ ["bold", "italic", "underline", "strike"], // 加粗 斜体 下划线 删除线 ["blockquote", "code-block"], // 引用 代码块 [{ list: "ordered" }, { list: "bullet" }], // 有序、无序列表 [{ indent: "-1" }, { indent: "+1" }], // 缩进 [{ 'size': ['10px', '12px', '14px', '16px', '18px', '20px', '24px', '36px'] }], // 自定义字号 [{ header: [1, 2, 3, 4, 5, 6, false] }], // 标题 [{ color: [] }, { background: [] }], // 字体颜色、字体背景颜色 [{ align: [] }], // 对齐方式 ["clean"], // 清除文本格式 ["link", "image", "video"] // 链接、图片、视频 ], }, placeholder: "请输入内容", readOnly: this.readOnly, }, }; }, computed: { styles() { let style = {}; if (this.minHeight) { style.minHeight = `${this.minHeight}px`; } if (this.height) { style.height = `${this.height}px`; } return style; }, }, watch: { value: { handler(val) { if (val !== this.currentValue) { this.currentValue = val === null ? "" : val; if (this.Quill) { this.Quill.clipboard.dangerouslyPasteHTML(this.currentValue); } } }, immediate: true, }, }, mounted() { this.init(); }, beforeDestroy() { this.Quill = null; }, methods: { init() { const editor = this.$refs.editor; this.Quill = new Quill(editor, this.options); // 如果设置了上传地址则自定义图片上传事件 if (this.type == 'url') { let toolbar = this.Quill.getModule("toolbar"); toolbar.addHandler("image", (value) => { if (value) { this.$refs.upload.$children[0].$refs.input.click(); } else { this.quill.format("image", false); } }); } // 修改工具栏的视频处理器 let toolbar = this.Quill.getModule("toolbar") toolbar.addHandler("video", () => { this.$refs.upload.$children[0].$refs.input.click() }) this.Quill.clipboard.dangerouslyPasteHTML(this.currentValue); this.Quill.on("text-change", (delta, oldDelta, source) => { const html = this.$refs.editor.children[0].innerHTML; const text = this.Quill.getText(); const quill = this.Quill; this.currentValue = html; this.$emit("input", html); this.$emit("on-change", { html, text, quill }); }); this.Quill.on("text-change", (delta, oldDelta, source) => { this.$emit("on-text-change", delta, oldDelta, source); }); this.Quill.on("selection-change", (range, oldRange, source) => { this.$emit("on-selection-change", range, oldRange, source); }); this.Quill.on("editor-change", (eventName, ...args) => { this.$emit("on-editor-change", eventName, ...args); }); }, // 上传前校检格式和大小 handleBeforeUpload(file) { const type = ["image/jpeg", "image/jpg", "image/png", "image/svg", "video/mp4"]; // 统一文件类型校验 if (!type.includes(file.type)) { this.$message.error(`仅支持 ${type.join(", ")} 格式`); return false; } // 校检文件大小 if (this.fileSize) { const isLt = file.size / 1024 / 1024 < this.fileSize; if (!isLt) { this.$message.error(`上传文件大小不能超过 ${this.fileSize} MB!`); return false; } } return true; }, handleUploadSuccess(res, file) { if (res.code == 200) { const quill = this.Quill const range = quill.getSelection().index // 视频插入逻辑 if (file.raw.type.startsWith('video/')) { const videoUrl = process.env.VUE_APP_BASE_API + res.fileName quill.insertEmbed(range, 'video', { url: videoUrl }) // 关键修改 } // 图片逻辑保持不变 else { quill.insertEmbed(range, "image", process.env.VUE_APP_BASE_API + res.fileName) } quill.setSelection(range + 1) } }, handleUploadError() { this.$message.error("图片插入失败"); }, }, }; </script> <style> .editor, .ql-toolbar { white-space: pre-wrap !important; line-height: normal !important; } .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: "等宽字体"; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="10px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="10px"]::before { content: '10px'; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="12px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="12px"]::before { content: '12px'; } /* 其他字号样式 */ .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="14px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="14px"]::before { content: '14px'; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="16px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="16px"]::before { content: '16px'; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="18px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="18px"]::before { content: '18px'; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="20px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="20px"]::before { content: '20px'; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="24px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="24px"]::before { content: '24px'; } .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="36px"]::before, .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="36px"]::before { content: '36px'; } </style> 3.效果展示 ...