| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320 |
- <template>
- <div class="ibps-ueditor" v-on="listeners">
- <script ref="script" :name="name" type="text/plain" />
- </div>
- </template>
- <script>
- // 参考代码 https://github.com/HaoChuan9421/vue-ueditor-wrap 的v2.4.2版本[2019-05-30]
- import LoadEvent from './utils/Event.js'
- import Debounce from './utils/Debounce.js'
- import PopupManager from '@/utils/popup'
- import defaultConfig from './ueditor.config'
- export default {
- name: 'ibps-ueditor',
- inject: {
- elForm: {
- default: ''
- },
- elFormItem: {
- default: ''
- }
- },
- props: {
- // v-model 实现方式
- mode: {
- type: String,
- default: 'observer',
- validator: function(value) {
- // 1. observer 借助 MutationObserver API https://developer.mozilla.org/zh-CN/docs/Web/API/MutationObserver
- // 2. listener 借助 UEditor 的 contentChange 事件 https://ueditor.baidu.com/doc/#UE.Editor:contentChange
- return ['observer', 'listener'].indexOf(value) !== -1
- }
- },
- value: {
- type: String,
- default: ''
- },
- config: {
- type: Object,
- default: function() {
- return {}
- }
- },
- init: {
- type: Function,
- default: function() {
- return () => {}
- }
- },
- readonly: {
- type: Boolean,
- default: false
- },
- destroy: {
- type: Boolean,
- default: false
- },
- name: {
- type: String,
- default: ''
- },
- observerDebounceTime: {
- type: Number,
- default: 50,
- validator: function(value) {
- return value >= 20
- }
- },
- observerOptions: {
- type: Object,
- default: function() {
- // https://developer.mozilla.org/en-US/docs/Web/API/MutationObserverInit
- return {
- attributes: true, // 是否监听 DOM 元素的属性变化
- attributeFilter: ['src', 'style', 'type', 'name'], // 只有在该数组中的属性值的变化才会监听
- characterData: true, // 是否监听文本节点
- childList: true, // 是否监听子节点
- subtree: true // 是否监听后代元素
- }
- }
- },
- // 本组件提供对普通 Vue 项目和 Nuxt 项目开箱即用的支持,但如果是自己搭建的 Vue SSR 项目,可能需要自行区分是客户端还是服务端环境并跳过环境检测,直接初始化
- forceInit: {
- type: Boolean,
- default: false
- },
- zIndex: {
- type: Number
- }
- },
- data() {
- return {
- editor: null,
- id: null,
- status: 0,
- initValue: '',
- defaultConfig: defaultConfig()
- }
- },
- computed: {
- mixedConfig() {
- return Object.assign({
- readonly: this.readonly,
- zIndex: this.getZindex()
- }, this.defaultConfig, this.config)
- },
- listeners() {
- return {
- ...this.$listeners
- }
- }
- },
- // v-model语法糖实现
- watch: {
- value: {
- handler(value) {
- // 修复值为空无法双向绑定的问题
- if (value === null || value === undefined) {
- value = ''
- }
- // 0: 尚未初始化 1: 开始初始化但尚未ready 2 初始化完成并已ready
- switch (this.status) {
- case 0:
- this.status = 1
- this.initValue = value;
- // 判断执行环境是服务端还是客户端,这里的 process.client 是 Nuxt 添加的环境变量
- (this.forceInit || (typeof process !== 'undefined' && process.client) || typeof window !== 'undefined') && this._checkDependencies().then(() => {
- this.$refs.script ? this._initEditor() : this.$nextTick(() => this._initEditor())
- })
- break
- case 1:
- this.initValue = value
- break
- case 2:
- this._setContent(value)
- break
- default:
- break
- }
- },
- immediate: true
- }
- },
- deactivated() {
- this.editor && this.editor.removeListener('contentChange', this.contentChangeHandler)
- this.observer && this.observer.disconnect()
- },
- beforeDestroy() {
- if (this.destroy && this.editor && this.editor.destroy) {
- this.editor.destroy()
- }
- if (this.observer && this.observer.disconnect) {
- this.observer.disconnect()
- }
- },
- methods: {
- // 添加自定义按钮(自定义按钮,自定义弹窗等操作从 2.2.0 版本开始不再考虑直接集成,这会使得组件和 UEditor 过度耦合,但为了兼容一些老版用户的写法,这个方法依然保留)
- registerButton({ name, icon, tip, handler, index, UE = window.UE }) {
- UE.registerUI(name, (editor, name) => {
- editor.registerCommand(name, {
- execCommand: () => {
- handler(editor, name)
- }
- })
- const btn = new UE.ui.Button({
- name,
- title: tip,
- cssRules: `background-image: url(${icon}) !important;background-size: cover;`,
- onclick() {
- editor.execCommand(name)
- }
- })
- editor.addListener('selectionchange', () => {
- const state = editor.queryCommandState(name)
- if (state === -1) {
- btn.setDisabled(true)
- btn.setChecked(false)
- } else {
- btn.setDisabled(false)
- btn.setChecked(state)
- }
- })
- return btn
- }, index, this.id)
- },
- // 实例化编辑器
- _initEditor() {
- this.$refs.script.id = this.id = 'ibps_editor_' + Math.random().toString(16).slice(-6) // 这么做是为了支持 Vue SSR,因为如果把 id 属性放在 data 里会导致服务端和客户端分别计算该属性的值,而造成 id 不匹配无法初始化的 BUG
- this.init()
- this.$emit('before-init', this.id, this.mixedConfig)
- this.$emit('beforeInit', this.id, this.mixedConfig) // 虽然这个驼峰的写法会导致使用 DOM 模版时出现监听事件自动转小写的 BUG,但如果经过编译的话并不会有这个问题,为了兼容历史版本,不做删除,参考 https://vuejs.org/v2/guide/components-custom-events.html#Event-Names
- this.editor = window.UE.getEditor(this.id, this.mixedConfig)
- this.editor.addListener('ready', () => {
- if (this.status === 2) { // 使用 keep-alive 组件会出现这种情况
- this.editor.setContent(this.value)
- } else {
- this.status = 2
- this.$emit('ready', this.editor)
- if (this.initValue) {
- this.editor.setContent(this.initValue)
- }
- }
- if (this.mode === 'observer' && window.MutationObserver) {
- this._observerChangeListener()
- } else {
- this._normalChangeListener()
- }
- })
- // zxh 增加click 事件
- this.editor.addListener('click', (editor) => {
- this.$emit('click', editor)
- })
- },
- // 检测依赖,确保 UEditor 资源文件已加载完毕
- _checkDependencies() {
- return new Promise((resolve, reject) => {
- // 判断ueditor.config.js和ueditor.all.js是否均已加载(仅加载完ueditor.config.js时UE对象和UEDITOR_CONFIG对象存在,仅加载完ueditor.all.js时UEDITOR_CONFIG对象存在,但为空对象)
- const scriptsLoaded = !!window.UE && !!window.UEDITOR_CONFIG && Object.keys(window.UEDITOR_CONFIG).length !== 0 && !!window.UE.getEditor
- if (scriptsLoaded) {
- resolve()
- } else if (window['$loadEnv']) { // 利用订阅发布,确保同时渲染多个组件时,不会重复创建script标签
- window['$loadEnv'].on('scriptsLoaded', () => {
- resolve()
- })
- } else {
- window['$loadEnv'] = new LoadEvent()
- // 如果在其他地方只引用ueditor.all.min.js,在加载ueditor.config.js之后仍需要重新加载ueditor.all.min.js,所以必须确保ueditor.config.js已加载
- this._loadConfig().then(() => this._loadCore()).then(() => {
- resolve()
- window['$loadEnv'].emit('scriptsLoaded')
- })
- }
- })
- },
- _loadConfig() {
- return new Promise((resolve, reject) => {
- if (window.UE && window.UEDITOR_CONFIG && Object.keys(window.UEDITOR_CONFIG).length !== 0) {
- resolve()
- return
- }
- const configScript = document.createElement('script')
- configScript.type = 'text/javascript'
- configScript.src = this.mixedConfig.UEDITOR_HOME_URL + 'ueditor.config.js'
- document.getElementsByTagName('head')[0].appendChild(configScript)
- configScript.onload = function() {
- if (window.UE && window.UEDITOR_CONFIG && Object.keys(window.UEDITOR_CONFIG).length !== 0) {
- resolve()
- } else {
- console.error('加载ueditor.config.js失败,请检查您的配置地址UEDITOR_HOME_URL填写是否正确!\n', configScript.src)
- }
- }
- })
- },
- _loadCore() {
- return new Promise((resolve, reject) => {
- if (window.UE && window.UE.getEditor) {
- resolve()
- return
- }
- const coreScript = document.createElement('script')
- coreScript.type = 'text/javascript'
- coreScript.src = this.mixedConfig.UEDITOR_HOME_URL + 'ueditor.all.min.js'
- document.getElementsByTagName('head')[0].appendChild(coreScript)
- coreScript.onload = function() {
- if (window.UE && window.UE.getEditor) {
- resolve()
- } else {
- console.error('加载ueditor.all.min.js失败,请检查您的配置地址UEDITOR_HOME_URL填写是否正确!\n', coreScript.src)
- }
- }
- })
- },
- // 设置内容
- _setContent(value) {
- if (this.editor) {
- value === this.editor.getContent() || this.editor.setContent(value)
- }
- },
- contentChangeHandler() {
- this.$emit('input', this.editor.getContent())
- },
- execCommand(cmd, params) {
- if (this.editor) {
- this.editor.execCommand(cmd, params)
- }
- },
- // 基于 UEditor 的 contentChange 事件
- _normalChangeListener() {
- if (this.editor) {
- this.editor.addListener('contentChange', this.contentChangeHandler)
- }
- },
- // 基于 MutationObserver API
- _observerChangeListener() {
- const changeHandle = (mutationsList) => {
- if (this.editor.document.getElementById('baidu_pastebin')) {
- return
- }
- this.$emit('input', this.editor.getContent())
- }
- // 函数防抖
- this.observer = new MutationObserver(Debounce(changeHandle, this.observerDebounceTime))
- this.observer.observe(this.editor.body, this.observerOptions)
- },
- /**
- * zxh 修复zindex 不是最高的被遮住
- */
- getZindex() {
- if (this.zIndex) {
- return this.zIndex
- }
- const zIndex = PopupManager.getZIndex()
- return zIndex - 1
- }
- }
- }
- </script>
|