239 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			239 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Vue
		
	
	
| <template>
 | |
|   <div>
 | |
|     <div style="position: relative">
 | |
|       <div :id="eid" style="text-align: left; min-width: 1080px"></div>
 | |
|       <div v-if="showExpand">
 | |
|         <div class="e-menu e-code" @click="editHTML">
 | |
|           <Icon type="md-code-working" size="22" />
 | |
|         </div>
 | |
|         <div class="e-menu e-preview" @click="fullscreenModal = true">
 | |
|           <Icon type="ios-eye" size="24" />
 | |
|         </div>
 | |
|         <div class="e-menu e-trash" @click="clear">
 | |
|           <Icon type="md-trash" size="18" />
 | |
|         </div>
 | |
|       </div>
 | |
|     </div>
 | |
| 
 | |
|     <Modal title="编辑html代码" v-model="showHTMLModal" :mask-closable="false" :width="900" :fullscreen="full">
 | |
|       <Input v-if="!full" v-model="dataEdit" :rows="15" type="textarea" style="max-height: 60vh; overflow: auto" />
 | |
|       <Input v-if="full" v-model="dataEdit" :rows="32" type="textarea" />
 | |
|       <div slot="footer">
 | |
|         <Button @click="full = !full" icon="md-expand">全屏开/关</Button>
 | |
|         <Button @click="editHTMLOk" type="primary" icon="md-checkmark-circle-outline">确定保存</Button>
 | |
|       </div>
 | |
|     </Modal>
 | |
|     <Modal title="预览" v-model="fullscreenModal" fullscreen>
 | |
|       <div v-html="data">{{ data }}</div>
 | |
|       <div slot="footer">
 | |
|         <Button @click="fullscreenModal = false">关闭</Button>
 | |
|       </div>
 | |
|     </Modal>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| import { uploadFile } from "@/libs/axios";
 | |
| import E from "wangeditor";
 | |
| import xss from "xss";
 | |
| // 表情包配置 自定义表情可在该js文件中统一修改
 | |
| import { sina } from "@/libs/emoji";
 | |
| export default {
 | |
|   name: "editor",
 | |
|   props: {
 | |
|     eid: {
 | |
|       type: String,
 | |
|       default: "editor",
 | |
|     },
 | |
|     value: String,
 | |
|     base64: {
 | |
|       type: Boolean,
 | |
|       default: false,
 | |
|     },
 | |
|     showExpand: {
 | |
|       type: Boolean,
 | |
|       default: true,
 | |
|     },
 | |
|     openXss: {
 | |
|       type: Boolean,
 | |
|       default: false,
 | |
|     },
 | |
|   },
 | |
|   data() {
 | |
|     return {
 | |
|       editor: "", // 初始化富文本编辑器
 | |
|       data: this.value, // 富文本数据
 | |
|       dataEdit: "", // 编辑数据
 | |
|       showHTMLModal: false, // 显示html
 | |
|       full: false, // html全屏开关
 | |
|       fullscreenModal: false, // 显示全屏预览
 | |
|     };
 | |
|   },
 | |
|   methods: {
 | |
|     // 初始化编辑器
 | |
|     initEditor() {
 | |
|       let that = this;
 | |
|       this.editor = new E(`#${this.eid}`);
 | |
|       // 编辑内容绑定数据
 | |
|       this.editor.config.onchange = (html) => {
 | |
|         if (this.openXss) {
 | |
|           this.data = xss(html);
 | |
|         } else {
 | |
|           this.data = html;
 | |
|         }
 | |
|         this.$emit("input", this.data);
 | |
|         this.$emit("on-change", this.data);
 | |
|       };
 | |
|       this.editor.config.showFullScreen = false;
 | |
|       // z-index
 | |
|       this.editor.config.zIndex = 100;
 | |
|       if (this.base64) {
 | |
|         // 使用 base64 保存图片
 | |
|         this.editor.config.uploadImgShowBase64 = true;
 | |
|       } else {
 | |
|         // 配置上传图片服务器端地址
 | |
|         this.editor.config.uploadImgServer = uploadFile;
 | |
|         // lili如要header中传入token鉴权
 | |
|         this.editor.config.uploadImgHeaders = {
 | |
|           accessToken: that.getStore("accessToken"),
 | |
|         };
 | |
|         this.editor.config.uploadFileName = "file";
 | |
|         this.editor.config.uploadImgHooks = {
 | |
|           before: function (xhr, editor, files) {
 | |
|             // 图片上传之前触发
 | |
|           },
 | |
|           success: function (xhr, editor, result) {
 | |
|             // 图片上传并返回结果,图片插入成功之后触发
 | |
|           },
 | |
|           fail: function (xhr, editor, result) {
 | |
|             // 图片上传并返回结果,但图片插入错误时触发
 | |
|             that.$Message.error("上传图片失败");
 | |
|           },
 | |
|           error: function (xhr, editor) {
 | |
|             // 图片上传出错时触发
 | |
|             that.$Message.error("上传图片出错");
 | |
|           },
 | |
|           timeout: function (xhr, editor) {
 | |
|             // 图片上传超时时触发
 | |
|             that.$Message.error("上传图片超时");
 | |
|           },
 | |
|           // 如果服务器端返回的不是 {errno:0, data: [...]} 这种格式,可使用该配置
 | |
|           customInsert: function (insertImg, result, editor) {
 | |
|             if (result.success == true) {
 | |
|               let url = result.result;
 | |
|               insertImg(url);
 | |
|               that.$Message.success("上传图片成功");
 | |
|             } else {
 | |
|               that.$Message.error(result.message);
 | |
|             }
 | |
|           },
 | |
|         };
 | |
|       }
 | |
|       this.editor.config.customAlert = function (info) {
 | |
|         // info 是需要提示的内容
 | |
|         // that.$Message.info(info);
 | |
|       };
 | |
|       // 字体
 | |
|       this.editor.config.fontNames = ["微软雅黑", "宋体", "黑体", "Arial"];
 | |
|       // 表情面板可以有多个 tab ,因此要配置成一个数组。数组每个元素代表一个 tab 的配置
 | |
|       this.editor.config.emotions = [
 | |
|         {
 | |
|           // tab 的标题
 | |
|           title: "新浪",
 | |
|           // type -> 'emoji' / 'image'
 | |
|           type: "image",
 | |
|           // content -> 数组
 | |
|           content: sina,
 | |
|         },
 | |
|       ];
 | |
|       if (this.value) {
 | |
|         if (this.openXss) {
 | |
|           this.editor.txt.html(xss(this.value));
 | |
|         } else {
 | |
|           this.editor.txt.html(this.value);
 | |
|         }
 | |
|       }
 | |
|       this.editor.create();
 | |
|     },
 | |
|     // html预览
 | |
|     editHTML() {
 | |
|       this.dataEdit = this.data;
 | |
|       this.showHTMLModal = true;
 | |
|     },
 | |
|     // 保存
 | |
|     editHTMLOk() {
 | |
|       this.editor.txt.html(this.dataEdit);
 | |
|       this.$emit("input", this.data);
 | |
|       this.$emit("on-change", this.data);
 | |
|       this.showHTMLModal = false;
 | |
|     },
 | |
|     // 清空编辑器
 | |
|     clear() {
 | |
|       this.$Modal.confirm({
 | |
|         title: "确认清空",
 | |
|         content: "确认要清空编辑器内容?清空后不能撤回",
 | |
|         onOk: () => {
 | |
|           this.data = "";
 | |
|           this.editor.txt.html(this.data);
 | |
|           this.$emit("input", this.data);
 | |
|           this.$emit("on-change", this.data);
 | |
|         },
 | |
|       });
 | |
|     },
 | |
|     setData(value) {
 | |
|       // 设置数据
 | |
|       if (!this.editor) {
 | |
|         this.initEditor();
 | |
|       }
 | |
|       if (value && value != this.data) {
 | |
|         this.data = value;
 | |
|         this.editor.txt.html(this.data);
 | |
|         this.$emit("input", this.data);
 | |
|         this.$emit("on-change", this.data);
 | |
|       }
 | |
|     },
 | |
|   },
 | |
|   watch: {
 | |
|     value: {
 | |
|       handler: function (val) {
 | |
|         // 赋值给富文本
 | |
|         this.setData(val);
 | |
|       },
 | |
|     },
 | |
|   },
 | |
| 
 | |
|   mounted() {
 | |
|     this.initEditor();
 | |
|   },
 | |
| };
 | |
| </script>
 | |
| 
 | |
| <style lang="scss" scoped>
 | |
| .e-menu {
 | |
|   z-index: 101;
 | |
|   position: absolute;
 | |
|   cursor: pointer;
 | |
|   color: #999;
 | |
|   :hover {
 | |
|     color: #333;
 | |
|   }
 | |
| }
 | |
| .w-e-toolbar {
 | |
|   // 给工具栏换行
 | |
|   flex-wrap: wrap;
 | |
| }
 | |
| .e-code {
 | |
|   top: 6px;
 | |
|   left: 976px;
 | |
| }
 | |
| .e-preview {
 | |
|   top: 6px;
 | |
|   left: 1014px;
 | |
| }
 | |
| .e-trash {
 | |
|   top: 4px;
 | |
|   left: 1047px;
 | |
| }
 | |
| </style>
 | |
| 
 |