当前位置: 首页 > news >正文

ck-editor5的研究 (4):初步使用 CKEditor5 的插件功能

前言

在上一篇文章中—— ck-editor5 的研究(3):初步使用 CKEditor5 的事件系统和API ,我们已经初步了解了 CKEditor5 的工作方式

那么这篇文章,我们将初步使用 CKEditor5 的插件功能,我将会写一个自己的插件,让编辑器聚焦,并自动插入一段文本。大概的效果如下:
在这里插入图片描述

编写插件只需 3 步

1. 第一步:先搭建目录

仍然是先搭建目录结构:

  1. 新建一个 ckeditor4.ts ,使用基本的配置
  2. 新建一个 ck-editor4.client.vue 接入插件功能
  3. 新建一个 demo4/index.vue 测试编辑器
  4. 新建一个 my-plugin.ts ,用来编写我们自己的插件

他们的代码如下:

// ckeditor4.ts
import ClassicEditor from '@ckeditor/ckeditor5-editor-classic/src/classiceditor';
import { Essentials } from '@ckeditor/ckeditor5-essentials';
import { Autoformat } from '@ckeditor/ckeditor5-autoformat';
import { Bold, Italic } from '@ckeditor/ckeditor5-basic-styles';
import { BlockQuote } from '@ckeditor/ckeditor5-block-quote';
import { Heading } from '@ckeditor/ckeditor5-heading';
import { Link } from '@ckeditor/ckeditor5-link';
import { List } from '@ckeditor/ckeditor5-list';
import { Paragraph } from '@ckeditor/ckeditor5-paragraph';export default class MyClassicEditor extends ClassicEditor {static override builtinPlugins = [Paragraph,Essentials,Autoformat,Bold,Italic,BlockQuote,Heading,Link,List,];static override defaultConfig = {toolbar: {items: ['heading', '|', 'bold', 'italic', 'link', 'bulletedList', 'numberedList', 'blockQuote', 'undo', 'redo'],},language: 'en',};
}
<template><div ref="editorRef"></div>
</template><script setup lang="ts">
import MyClassicEditor from './ckeditor4';defineOptions({ name: 'CkEditor4' });type EmitsType = {ready: [editor: MyClassicEditor];destroy: [];
};
const emit = defineEmits<EmitsType>();const editorRef = useTemplateRef('editorRef');
let instance: MyClassicEditor | null = null;async function initEditor() {try {if (!editorRef.value) {return;}instance = await MyClassicEditor.create(editorRef.value);// console.log('instance :>> ', instance);emit('ready', instance); // 暴露编辑器实例,给父组件使用} catch (error: any) {console.log('error :>> ', error.stack);}
}onMounted(() => {nextTick(initEditor);
});
onBeforeUnmount(() => {if (instance) {instance.destroy();instance = null;}emit('destroy');
});
</script>

my-plugin.ts目前还是空的,而 demo4/index.vue 测试页面如下:
在这里插入图片描述
表现到页面上就是这样:在这里插入图片描述
到此为止,目录结构搭建完成,接下来准备开发插件

2. 研究文档,并编写插件

先找到官方文档 扩展功能, 进行观察,我们发现,编写插件有2种方式,第一种是使用构造函数,第二种是编写一个类名:在这里插入图片描述
于是我们进入到 my-plugin.ts 写2个插件,分别通过 构造函数 和 类名 的形式编写并进行导出:

值得注意的是,使用类名的方法,会有一个固定的 init 方法,它会被编辑器自动调用

import { Plugin } from '@ckeditor/ckeditor5-core';
import type { Editor } from '@ckeditor/ckeditor5-core';function MyPlugin1(editor: Editor) {console.log('MyPlugin1 初始化了');console.log('编辑器的实例对象是', editor);
}class MyPlugin2 extends Plugin {constructor(editor: Editor) {super(editor);console.log('MyPlugin2 constructor 执行了');}init() {console.log('MyPlugin2 init');console.log('编辑器的实例对象是', this.editor);}
}export { MyPlugin1, MyPlugin2 };

紧接着,在 ckeditor4.ts 中导入,并注册插件:

在这里插入图片描述
接着刷新页面,看到我们自己写的插件的确是被执行了,控制台有打印内容:
在这里插入图片描述

3. 在插件中添加自己的逻辑

我的现在想做一件事:初始化 2 秒之后,让编辑器聚焦,再过 2 秒后,插入一段文本,并把光标聚焦到编辑器末尾

于是对 MyPlugin2 进行改写:

class MyPlugin2 extends Plugin {constructor(editor: Editor) {super(editor);console.log('MyPlugin2 constructor 执行了');}init() {console.log('MyPlugin2 init');console.log('编辑器的实例对象是', this.editor);// 2 秒后聚焦编辑器setTimeout(() => {this.editor.focus();// 再过 2 秒后插入文本setTimeout(() => {this.editor.model.change((writer) => {const text = writer.createText('我将会在第 4 秒后被插入到编辑器中, 并且会光标被移动到编辑器的末尾');this.editor.model.insertContent(text);// 获取根节点, 并设置光标位置在末尾const root = this.editor.model.document.getRoot();if (!root) {return;}const position = writer.createPositionAt(root, 'end');writer.setSelection(position);});}, 2000);}, 2000);}
}

测试最终效果

打开 demo4/index.vue 页面,我们可以看到自己写的插件起作用了:
在这里插入图片描述

最后总结

回顾一下目前学的内容:

  1. 我们已经成功把 CKEditor5 集成到了 nuxt 中,并且封装成了一个通用的 vue 组件,可以实际在项目中使用了。
  2. 我们认识了 CKEditor5 的事件系统、常用API、插件编写。知道了 CKEditor5 的大致设计概念,比如 Model 和 2种View (即: Editing view + Data view),已经可以通过 js 对编辑器进行一定的操作了。
  3. 我们用过了好几个 Model 的方法:比如 editor.model.change((writer) => {}) 和 editor.model.document.on(‘change:data’, () => {})
  4. 我们也用过了 Editing view 的方法:比如 editor.editing.view.document.on(‘enter’, () => {})
  5. 我们也用过了 Data view 的方法:比如 editor.data.processor.toView(content) 和 editor.data.toModel(viewFragment);

而这一篇内容,我们又初步明白了 CKEditor5 的插件功能。插件有 2 中插入创建方式,一种是通过构造函数创建,另一种是定义一个 Class 类。我们更倾向于 Class 类名的形式创建。

也算是完成了一个小小的里程碑了!

看来 CKEditor5 的面纱正在被我一步步揭开,不容易啊!

之前本来打算在掘金更新的,写完之后复制过来,却意外发现 CSDN 的同学们比较热情,更加务实,专注于解决问题,并且不会吝啬自己的点赞。这也让我这种写博客的新手有动力更新。端午假期连续在家写了 4 篇内容。

相反,掘金很多文章都是凑字数、标题党,搞得太冗余繁杂了。而且大佬偏多,我发现大佬很少给别人点赞,除非那篇文章真的写得好。

后续我会继续研究 CKEditor5,并保持一颗学徒的心态,尽可能把这些经验总结出来。

http://www.xdnf.cn/news/762499.html

相关文章:

  • 72.编辑用户消息功能之前端实现
  • PCB制作入门
  • 开始通信之旅-----话题通信
  • 关于 java:4. 异常处理与调试
  • C#数字图像处理(二)
  • IO流1——体系介绍和字节输出流
  • 如何用利用deepseek的API能力来搭建属于自己的智能体-优雅草卓伊凡
  • 【AI面试秘籍】| 第25期:RAG的关键痛点及解决方案深度解析
  • OpenGL、GLUT、freeGLUT 与 GLFW 的区别
  • 【渲染】拆解《三国:谋定天下》场景渲染技术
  • C++实现汉诺塔游戏自动完成
  • [AD] CrownJewel-1 Logon 4799+vss-ShadowCopy+NTDS.dit/SYSTEM+$MFT
  • QT中子线程触发主线程弹窗并阻塞等待用户响应
  • Ⅰ.计算机二级选择题(C语言概述)
  • 第二章 机器学习基本概念
  • 【RocketMQ 生产者和消费者】- 生产者发送同步、异步、单向消息源码分析(1)
  • 利用IEEE 802.15.4z-IR UWB系统进行手势检测
  • Python中scapy库详细使用(强大的交互式数据包操作程序和库)
  • 基于 Three.js 的文本粒子解体效果技术原理剖析
  • 002 dart刷题
  • 车载控制器的“机电一体化”深度集成
  • 自编码器Auto-encoder(李宏毅)
  • Go语言实现高性能分布式爬虫系统 - 设计与实践
  • 在线音乐服务器测试报告
  • Codeforces 1027 Div3(ABCDEF)
  • 过滤攻击-隐私保护
  • 淘宝商品详情页有哪些常见的动态加载技术?
  • Python训练营---Day42
  • pikachu通关教程- over permission
  • 深入理解 C++11 中的 std::move —— 移动语义详解(小白友好版)