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

View File

@ -1,77 +1,103 @@
<script setup lang="ts"> <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<{ 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) { const editorModel = ref<string>()
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('上传失败')
// onMounted(() => {
resolve('上传成功') initEditor()
}) })
.catch((error) => {
// const editorRef = ref()
reject(error) 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> </script>
<template> <template>
<div class="tinymce-boxz"> <div class="h-2xl">
<Editor <div ref="editorRef" />
v-model="data"
api-key="no-api"
:init="initConfig"
/>
</div> </div>
</template> </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> <template>
<n-card title="MarkDown编辑器"> <n-card title="MarkDown编辑器">
<n-space vertical :size="12"> <n-space vertical :size="12">
<n-alert :show-icon="false" type="info"> <n-alert title="基于 md-editor-v3 封装" type="success" />
使用 md-editor-v3 实现
</n-alert>
<MarkDownEditor v-model="text" /> <MarkDownEditor v-model="text" />
</n-space> </n-space>
</n-card> </n-card>

View File

@ -9,8 +9,17 @@ onMounted(() => {
</script> </script>
<template> <template>
<n-card> <n-card title="富文本编辑器">
<RichTextEditor v-model="text" /> <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> </n-card>
</template> </template>