xm
2024-06-14 722af26bc6fec32bb289b1df51a9016a4935610f
提交 | 用户 | 时间
722af2 1 <template>
X 2   <div>
3     <el-upload
4       :action="uploadUrl"
5       :before-upload="handleBeforeUpload"
6       :on-success="handleUploadSuccess"
7       :on-error="handleUploadError"
8       name="file"
9       :show-file-list="false"
10       :headers="headers"
11       style="display: none"
12       ref="upload"
13       v-if="this.type == 'url'"
14     >
15     </el-upload>
16     <div class="editor" ref="editor" :style="styles"></div>
17   </div>
18 </template>
19
20 <script>
21 import Quill from "quill";
22 import "quill/dist/quill.core.css";
23 import "quill/dist/quill.snow.css";
24 import "quill/dist/quill.bubble.css";
25 import { getToken } from "@/utils/auth";
26
27 export default {
28   name: "Editor",
29   props: {
30     /* 编辑器的内容 */
31     value: {
32       type: String,
33       default: "",
34     },
35     /* 高度 */
36     height: {
37       type: Number,
38       default: null,
39     },
40     /* 最小高度 */
41     minHeight: {
42       type: Number,
43       default: null,
44     },
45     /* 只读 */
46     readOnly: {
47       type: Boolean,
48       default: false,
49     },
50     // 上传文件大小限制(MB)
51     fileSize: {
52       type: Number,
53       default: 5,
54     },
55     /* 类型(base64格式、url格式) */
56     type: {
57       type: String,
58       default: "url",
59     }
60   },
61   data() {
62     return {
63       uploadUrl: process.env.VUE_APP_BASE_API + "/system/oss/upload", // 上传的图片服务器地址
64       headers: {
65         Authorization: "Bearer " + getToken()
66       },
67       Quill: null,
68       currentValue: "",
69       options: {
70         theme: "snow",
71         bounds: document.body,
72         debug: "warn",
73         modules: {
74           // 工具栏配置
75           toolbar: [
76             ["bold", "italic", "underline", "strike"],       // 加粗 斜体 下划线 删除线
77             ["blockquote", "code-block"],                    // 引用  代码块
78             [{ list: "ordered" }, { list: "bullet" }],       // 有序、无序列表
79             [{ indent: "-1" }, { indent: "+1" }],            // 缩进
80             [{ size: ["small", false, "large", "huge"] }],   // 字体大小
81             [{ header: [1, 2, 3, 4, 5, 6, false] }],         // 标题
82             [{ color: [] }, { background: [] }],             // 字体颜色、字体背景颜色
83             [{ align: [] }],                                 // 对齐方式
84             ["clean"],                                       // 清除文本格式
85             ["link", "image", "video"]                       // 链接、图片、视频
86           ],
87         },
88         placeholder: "请输入内容",
89         readOnly: this.readOnly,
90       },
91     };
92   },
93   computed: {
94     styles() {
95       let style = {};
96       if (this.minHeight) {
97         style.minHeight = `${this.minHeight}px`;
98       }
99       if (this.height) {
100         style.height = `${this.height}px`;
101       }
102       return style;
103     },
104   },
105   watch: {
106     value: {
107       handler(val) {
108         if (val !== this.currentValue) {
109           this.currentValue = val === null ? "" : val;
110           if (this.Quill) {
111             this.Quill.pasteHTML(this.currentValue);
112           }
113         }
114       },
115       immediate: true,
116     },
117   },
118   mounted() {
119     this.init();
120   },
121   beforeDestroy() {
122     this.Quill = null;
123   },
124   methods: {
125     init() {
126       const editor = this.$refs.editor;
127       this.Quill = new Quill(editor, this.options);
128       // 如果设置了上传地址则自定义图片上传事件
129       if (this.type == 'url') {
130         let toolbar = this.Quill.getModule("toolbar");
131         toolbar.addHandler("image", (value) => {
132           this.uploadType = "image";
133           if (value) {
134             this.$refs.upload.$children[0].$refs.input.click();
135           } else {
136             this.quill.format("image", false);
137           }
138         });
139       }
140       this.Quill.pasteHTML(this.currentValue);
141       this.Quill.on("text-change", (delta, oldDelta, source) => {
142         const html = this.$refs.editor.children[0].innerHTML;
143         const text = this.Quill.getText();
144         const quill = this.Quill;
145         this.currentValue = html;
146         this.$emit("input", html);
147         this.$emit("on-change", { html, text, quill });
148       });
149       this.Quill.on("text-change", (delta, oldDelta, source) => {
150         this.$emit("on-text-change", delta, oldDelta, source);
151       });
152       this.Quill.on("selection-change", (range, oldRange, source) => {
153         this.$emit("on-selection-change", range, oldRange, source);
154       });
155       this.Quill.on("editor-change", (eventName, ...args) => {
156         this.$emit("on-editor-change", eventName, ...args);
157       });
158     },
159     // 上传前校检格式和大小
160     handleBeforeUpload(file) {
161       // 校检文件大小
162       if (this.fileSize) {
163         const isLt = file.size / 1024 / 1024 < this.fileSize;
164         if (!isLt) {
165           this.$message.error(`上传文件大小不能超过 ${this.fileSize} MB!`);
166           return false;
167         }
168       }
169       return true;
170     },
171     handleUploadSuccess(res, file) {
172       // 获取富文本组件实例
173       let quill = this.Quill;
174       // 如果上传成功
175       if (res.code == 200) {
176         // 获取光标所在位置
177         let length = quill.getSelection().index;
178         // 插入图片  res.url为服务器返回的图片地址
179         quill.insertEmbed(length, "image", res.data.url);
180         // 调整光标到最后
181         quill.setSelection(length + 1);
182       } else {
183         this.$message.error(res.msg);
184       }
185     },
186     handleUploadError() {
187       this.$message.error("图片插入失败");
188     },
189   },
190 };
191 </script>
192
193 <style>
194 .editor, .ql-toolbar {
195   white-space: pre-wrap !important;
196   line-height: normal !important;
197 }
198 .quill-img {
199   display: none;
200 }
201 .ql-snow .ql-tooltip[data-mode="link"]::before {
202   content: "请输入链接地址:";
203 }
204 .ql-snow .ql-tooltip.ql-editing a.ql-action::after {
205   border-right: 0px;
206   content: "保存";
207   padding-right: 0px;
208 }
209
210 .ql-snow .ql-tooltip[data-mode="video"]::before {
211   content: "请输入视频地址:";
212 }
213
214 .ql-snow .ql-picker.ql-size .ql-picker-label::before,
215 .ql-snow .ql-picker.ql-size .ql-picker-item::before {
216   content: "14px";
217 }
218 .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="small"]::before,
219 .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="small"]::before {
220   content: "10px";
221 }
222 .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="large"]::before,
223 .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="large"]::before {
224   content: "18px";
225 }
226 .ql-snow .ql-picker.ql-size .ql-picker-label[data-value="huge"]::before,
227 .ql-snow .ql-picker.ql-size .ql-picker-item[data-value="huge"]::before {
228   content: "32px";
229 }
230
231 .ql-snow .ql-picker.ql-header .ql-picker-label::before,
232 .ql-snow .ql-picker.ql-header .ql-picker-item::before {
233   content: "文本";
234 }
235 .ql-snow .ql-picker.ql-header .ql-picker-label[data-value="1"]::before,
236 .ql-snow .ql-picker.ql-header .ql-picker-item[data-value="1"]::before {
237   content: "标题1";
238 }
239 .ql-snow .ql-picker.ql-header .ql-picker-label[data-value="2"]::before,
240 .ql-snow .ql-picker.ql-header .ql-picker-item[data-value="2"]::before {
241   content: "标题2";
242 }
243 .ql-snow .ql-picker.ql-header .ql-picker-label[data-value="3"]::before,
244 .ql-snow .ql-picker.ql-header .ql-picker-item[data-value="3"]::before {
245   content: "标题3";
246 }
247 .ql-snow .ql-picker.ql-header .ql-picker-label[data-value="4"]::before,
248 .ql-snow .ql-picker.ql-header .ql-picker-item[data-value="4"]::before {
249   content: "标题4";
250 }
251 .ql-snow .ql-picker.ql-header .ql-picker-label[data-value="5"]::before,
252 .ql-snow .ql-picker.ql-header .ql-picker-item[data-value="5"]::before {
253   content: "标题5";
254 }
255 .ql-snow .ql-picker.ql-header .ql-picker-label[data-value="6"]::before,
256 .ql-snow .ql-picker.ql-header .ql-picker-item[data-value="6"]::before {
257   content: "标题6";
258 }
259
260 .ql-snow .ql-picker.ql-font .ql-picker-label::before,
261 .ql-snow .ql-picker.ql-font .ql-picker-item::before {
262   content: "标准字体";
263 }
264 .ql-snow .ql-picker.ql-font .ql-picker-label[data-value="serif"]::before,
265 .ql-snow .ql-picker.ql-font .ql-picker-item[data-value="serif"]::before {
266   content: "衬线字体";
267 }
268 .ql-snow .ql-picker.ql-font .ql-picker-label[data-value="monospace"]::before,
269 .ql-snow .ql-picker.ql-font .ql-picker-item[data-value="monospace"]::before {
270   content: "等宽字体";
271 }
272 </style>