升级富文本

pull/975/head
zhangdaiscott 2023-12-29 21:24:03 +08:00
parent 308326a5f9
commit d4f134b2b5
1 changed files with 28 additions and 47 deletions

View File

@ -8,52 +8,28 @@
v-show="editorRef" v-show="editorRef"
:disabled="disabled" :disabled="disabled"
/> />
<textarea :id="tinymceId" ref="elRef" :style="{ visibility: 'hidden' }" v-if="!initOptions.inline"></textarea> <Editor :id="tinymceId" ref="elRef" :disabled="disabled" :init="initOptions" :style="{ visibility: 'hidden' }" v-if="!initOptions.inline"></Editor>
<slot v-else></slot> <slot v-else></slot>
</div> </div>
</template> </template>
<script lang="ts"> <script lang="ts">
import type { Editor, RawEditorSettings } from 'tinymce';
import tinymce from 'tinymce/tinymce'; import tinymce from 'tinymce/tinymce';
import Editor from '@tinymce/tinymce-vue'
import 'tinymce/themes/silver'; import 'tinymce/themes/silver';
import 'tinymce/icons/default/icons'; import 'tinymce/icons/default/icons';
import 'tinymce/plugins/advlist'; import 'tinymce/models/dom';
import 'tinymce/plugins/anchor';
import 'tinymce/plugins/autolink'; // tinymce
import 'tinymce/plugins/autosave'; // https://www.tiny.cloud/docs/plugins/
import 'tinymce/plugins/code';
import 'tinymce/plugins/codesample';
import 'tinymce/plugins/directionality';
import 'tinymce/plugins/fullscreen'; import 'tinymce/plugins/fullscreen';
import 'tinymce/plugins/hr';
import 'tinymce/plugins/insertdatetime';
import 'tinymce/plugins/link'; import 'tinymce/plugins/link';
import 'tinymce/plugins/lists'; import 'tinymce/plugins/lists';
import 'tinymce/plugins/media';
import 'tinymce/plugins/nonbreaking';
import 'tinymce/plugins/noneditable';
import 'tinymce/plugins/pagebreak';
import 'tinymce/plugins/paste';
import 'tinymce/plugins/preview'; import 'tinymce/plugins/preview';
import 'tinymce/plugins/print';
import 'tinymce/plugins/save';
import 'tinymce/plugins/searchreplace';
import 'tinymce/plugins/spellchecker';
import 'tinymce/plugins/tabfocus';
// import 'tinymce/plugins/table';
import 'tinymce/plugins/template';
import 'tinymce/plugins/textpattern';
import 'tinymce/plugins/visualblocks';
import 'tinymce/plugins/visualchars';
import 'tinymce/plugins/wordcount';
import 'tinymce/plugins/image'; import 'tinymce/plugins/image';
import 'tinymce/plugins/table';
import 'tinymce/plugins/textcolor';
import 'tinymce/plugins/contextmenu';
import { defineComponent, computed, nextTick, ref, unref, watch, onDeactivated, onBeforeUnmount } from 'vue'; import { defineComponent, computed, nextTick, ref, unref, watch, onDeactivated, onBeforeUnmount } from 'vue';
import ImgUpload from './ImgUpload.vue'; import ImgUpload from './ImgUpload.vue';
import { toolbar, plugins, simplePlugins, simpleToolbar, menubar } from './tinymce'; import {simpleToolbar, menubar, simplePlugins} from './tinymce';
import { buildShortUUID } from '/@/utils/uuid'; import { buildShortUUID } from '/@/utils/uuid';
import { bindHandlers } from './helper'; import { bindHandlers } from './helper';
import { onMountedOrActivated } from '/@/hooks/core/onMountedOrActivated'; import { onMountedOrActivated } from '/@/hooks/core/onMountedOrActivated';
@ -74,11 +50,11 @@
toolbar: { toolbar: {
type: [Array as PropType<string[]>, String], type: [Array as PropType<string[]>, String],
default: toolbar, default: simpleToolbar,
}, },
plugins: { plugins: {
type: Array as PropType<string[]>, type: [Array as PropType<string[]>, String],
default: plugins, default: simplePlugins,
}, },
menubar: { menubar: {
type: [Object, String], type: [Object, String],
@ -105,12 +81,14 @@
export default defineComponent({ export default defineComponent({
name: 'Tinymce', name: 'Tinymce',
components: { ImgUpload }, components: { ImgUpload,Editor },
inheritAttrs: false, inheritAttrs: false,
props: tinymceProps, props: tinymceProps,
emits: ['change', 'update:modelValue', 'inited', 'init-error'], emits: ['change', 'update:modelValue', 'inited', 'init-error'],
setup(props, { emit, attrs }) { setup(props, { emit, attrs }) {
const editorRef = ref<Nullable<Editor>>(null); console.log("---Tinymce---初始化---")
const editorRef = ref<Nullable<any>>(null);
const fullscreen = ref(false); const fullscreen = ref(false);
const tinymceId = ref<string>(buildShortUUID('tiny-vue')); const tinymceId = ref<string>(buildShortUUID('tiny-vue'));
const elRef = ref<Nullable<HTMLElement>>(null); const elRef = ref<Nullable<HTMLElement>>(null);
@ -138,14 +116,14 @@
return ['zh_CN', 'en'].includes(lang) ? lang : 'zh_CN'; return ['zh_CN', 'en'].includes(lang) ? lang : 'zh_CN';
}); });
const initOptions = computed((): RawEditorSettings => { const initOptions = computed(() => {
const { height, options, toolbar, plugins, menubar } = props; const { height, options, toolbar, plugins, menubar } = props;
const publicPath = import.meta.env.VITE_PUBLIC_PATH || '/'; const publicPath = import.meta.env.VITE_PUBLIC_PATH || '/';
return { return {
selector: `#${unref(tinymceId)}`, selector: `#${unref(tinymceId)}`,
height, height,
toolbar, toolbar,
menubar: menubar, menubar: false,
plugins, plugins,
language_url: publicPath + 'resource/tinymce/langs/' + langName.value + '.js', language_url: publicPath + 'resource/tinymce/langs/' + langName.value + '.js',
language: langName.value, language: langName.value,
@ -155,10 +133,11 @@
object_resizing: true, object_resizing: true,
toolbar_mode: 'sliding', toolbar_mode: 'sliding',
auto_focus: true, auto_focus: true,
toolbar_groups: true, // toolbar_groups: true,
skin: skinName.value, skin: skinName.value,
skin_url: publicPath + 'resource/tinymce/skins/ui/' + skinName.value, skin_url: publicPath + 'resource/tinymce/skins/ui/' + skinName.value,
images_upload_handler: (blobInfo, success) => { images_upload_handler: (blobInfo, process) =>
new Promise((resolve, reject) => {
let params = { let params = {
file: blobInfo.blob(), file: blobInfo.blob(),
filename: blobInfo.filename(), filename: blobInfo.filename(),
@ -168,18 +147,20 @@
if (res.success) { if (res.success) {
if (res.message == 'local') { if (res.message == 'local') {
const img = 'data:image/jpeg;base64,' + blobInfo.base64(); const img = 'data:image/jpeg;base64,' + blobInfo.base64();
success(img); resolve(img);
} else { } else {
let img = getFileAccessHttpUrl(res.message); let img = getFileAccessHttpUrl(res.message);
success(img); resolve(img);
} }
} else {
reject('上传失败!');
} }
}; };
uploadFile(params, uploadSuccess); uploadFile(params, uploadSuccess);
}, }),
content_css: publicPath + 'resource/tinymce/skins/ui/' + skinName.value + '/content.min.css', content_css: publicPath + 'resource/tinymce/skins/ui/' + skinName.value + '/content.min.css',
...options, ...options,
setup: (editor: Editor) => { setup: (editor: any) => {
editorRef.value = editor; editorRef.value = editor;
editor.on('init', (e) => initSetup(e)); editor.on('init', (e) => initSetup(e));
}, },
@ -191,7 +172,7 @@
const getdDisabled = options && Reflect.get(options, 'readonly'); const getdDisabled = options && Reflect.get(options, 'readonly');
const editor = unref(editorRef); const editor = unref(editorRef);
// update-begin-author:taoyan date:20220407 for: disabled // update-begin-author:taoyan date:20220407 for: disabled
if (editor) { if (editor && editor?.setMode) {
editor.setMode(getdDisabled || attrs.disabled === true ? 'readonly' : 'design'); editor.setMode(getdDisabled || attrs.disabled === true ? 'readonly' : 'design');
} }
if (attrs.disabled === true) { if (attrs.disabled === true) {
@ -208,7 +189,7 @@
if (!editor) { if (!editor) {
return; return;
} }
editor.setMode(attrs.disabled ? 'readonly' : 'design'); editor?.setMode && editor.setMode(attrs.disabled ? 'readonly' : 'design');
} }
); );
@ -239,7 +220,7 @@
function initEditor() { function initEditor() {
const el = unref(elRef); const el = unref(elRef);
if (el) { if (el && el?.style && el?.style?.visibility) {
el.style.visibility = ''; el.style.visibility = '';
} }
tinymce tinymce