feat: replace rice text editor to quill2.0

This commit is contained in:
chansee97 2024-05-12 02:51:31 +08:00
parent 3b3b964067
commit 922e82d12f
4 changed files with 100 additions and 67 deletions

View File

@ -51,7 +51,6 @@
},
"dependencies": {
"@alova/scene-vue": "^1.4.6",
"@tinymce/tinymce-vue": "^5.1.1",
"@vueuse/core": "^10.9.0",
"alova": "^2.19.0",
"colord": "^2.9.3",
@ -61,6 +60,7 @@
"pinia": "^2.1.7",
"pinia-plugin-persist": "^1.0.0",
"qs": "^6.12.0",
"quill": "^2.0.1",
"radash": "^12.1.0",
"vue": "^3.4.21",
"vue-i18n": "^9.11.1",

View File

@ -1,77 +1,103 @@
<script setup lang="ts">
import Editor from '@tinymce/tinymce-vue'
import Quill from 'quill'
import 'quill/dist/quill.snow.css'
defineOptions({
name: 'RichTextEditor',
})
const props = defineProps<{
modelValue: string
disabled?: boolean
}>()
const emit = defineEmits(['update:modelValue'])
const model = defineModel()
const data = useVModel(props, 'modelValue', emit)
let editorInst = null
function imagesUploadHandler(blobInfo: any, _progress: number) {
return new Promise((resolve, reject) => {
const formData = new FormData()
formData.append('file', blobInfo.blob())
fetch('www.example.com/upload', {
method: 'POST',
body: formData,
})
.then((response) => {
if (!response.ok)
throw new Error('上传失败')
const editorModel = ref<string>()
//
resolve('上传成功')
})
.catch((error) => {
//
reject(error)
})
onMounted(() => {
initEditor()
})
const editorRef = ref()
function initEditor() {
const options = {
modules: {
toolbar: [
{ header: [1, 2, 3, 4, 5, 6, false] }, //
'bold', //
'italic', //
'strike', // 线
{ size: ['small', false, 'large', 'huge'] }, //
{ font: [] }, //
{ color: [] }, //
{ background: [] }, //
'link', //
'image', //
'blockquote', //
'link', //
'image', //
'video', //
{ list: 'bullet' }, //
{ list: 'ordered' }, //
{ script: 'sub' }, //
{ script: 'super' }, //
{ align: [] }, //
'formula', //
'clean', // remove formatting button
],
},
placeholder: 'Insert text here ...',
theme: 'snow',
}
const quill = new Quill(editorRef.value, options)
quill.on('text-change', (_delta, _oldDelta, _source) => {
editorModel.value = quill.getSemanticHTML()
})
}
const initConfig = {
language: 'zh_CN', //
min_height: 700,
content_css: 'dark',
placeholder: '请输入内容', // textarea
branding: false,
font_formats:
'微软雅黑=Microsoft YaHei,Helvetica Neue,PingFang SC,sans-serif;苹果苹方=PingFang SC,Microsoft YaHei,sans-serif;宋体=simsun,serif;仿宋体=FangSong,serif;黑体=SimHei,sans-serif;Arial=arial,helvetica,sans-serif;Arial Black=arial black,avant garde;Book Antiqua=book antiqua,palatino;', //
plugins:
'print preview searchreplace autolink directionality visualblocks visualchars fullscreen code codesample table charmap hr pagebreak nonbreaking anchor insertdatetime advlist lists wordcount textpattern autosave emoticons', // axupimgs indent2em
toolbar: [
'fullscreen undo redo restoredraft | cut copy paste pastetext | forecolor backcolor bold italic underline strikethrough anchor | alignleft aligncenter alignright alignjustify outdent indent | bullist numlist | blockquote subscript superscript removeformat ',
'styleselect formatselect fontselect fontsizeselect | table emoticons charmap hr pagebreak insertdatetime selectall visualblocks | code preview | indent2em lineheight formatpainter',
],
paste_data_images: true, //
//
images_upload_handler: imagesUploadHandler,
if (props.disabled)
quill.enable(false)
editorInst = quill
}
watch(
() => model.value,
(newValue, _oldValue) => {
if (newValue && newValue !== editorModel.value) {
editorInst!.setContents(editorInst!.clipboard.convert({
html: newValue,
}))
}
else if (!newValue) {
editorInst!.setContents([])
}
},
)
watch(editorModel, (newValue, oldValue) => {
if (newValue && newValue !== oldValue)
model.value = newValue
else if (!newValue)
quillInstance!.setContents([])
})
watch(
() => props.disabled,
(newValue, _oldValue) => {
editorInst!.enable(!newValue)
},
)
onBeforeUnmount(() => quillInstance = null)
</script>
<template>
<div class="tinymce-boxz">
<Editor
v-model="data"
api-key="no-api"
:init="initConfig"
/>
<div class="h-2xl">
<div ref="editorRef" />
</div>
</template>
<style>
.tinymce-boxz > textarea {
display: none;
}
/* 隐藏apikey没有绑定这个域名的提示 */
.tox-notifications-container .tox-notification--warning {
display: none !important;
}
.tox.tox-tinymce {
max-width: 100%;
}
</style>

View File

@ -5,9 +5,7 @@ const text = ref('# Hello Editor ![图片描述](https://via.placeholder.com/350
<template>
<n-card title="MarkDown编辑器">
<n-space vertical :size="12">
<n-alert :show-icon="false" type="info">
使用 md-editor-v3 实现
</n-alert>
<n-alert title="基于 md-editor-v3 封装" type="success" />
<MarkDownEditor v-model="text" />
</n-space>
</n-card>

View File

@ -9,8 +9,17 @@ onMounted(() => {
</script>
<template>
<n-card>
<RichTextEditor v-model="text" />
<n-card title="富文本编辑器">
<n-space vertical :size="12">
<n-alert title="基于 Quill 封装" type="success" />
<n-space :size="12">
<RichTextEditor v-model="text" />
<div>
<n-h2>v-html 预览</n-h2>
<div v-html="text" />
</div>
</n-space>
</n-space>
</n-card>
</template>