Download the PHP package avzer/cherry-markdown without Composer
On this page you can find all versions of the php package avzer/cherry-markdown. It is possible to download/install these versions without Composer. Possible dependencies are resolved automatically.
Download avzer/cherry-markdown
More information about avzer/cherry-markdown
Files in avzer/cherry-markdown
Package cherry-markdown
Short Description A Markdown Editor
License
Homepage https://github.com/Tencent/cherry-markdown
Informations about the package cherry-markdown
Cherry Markdown Editor
介绍
Cherry Markdown Editor 是一款 Javascript Markdown 编辑器,具有开箱即用、轻量简洁、易于扩展等特点. 它可以运行在浏览器或服务端(NodeJs).
开箱即用
开发者可以使用非常简单的方式调用并实例化 Cherry Markdown 编辑器,实例化的编辑器默认支持大部分常用的 markdown 语法(如标题、目录、流程图、公式等)。
易于拓展
当 Cherry Markdown 编辑器支持的语法不满足开发者需求时,可以快速的进行二次开发或功能扩展。同时,CherryMarkdown 编辑器应该由纯 JavaScript 实现,不应该依赖 angular、vue、react 等框架技术,框架只提供容器环境即可。
特性
语法特性
- 图片缩放、对齐、引用
- 根据表格内容生成图表
- 字体颜色、字体大小
- 字体背景颜色、上标、下标
- checklist
- 音视频
多种模式
- 双栏编辑预览模式(支持同步滚动)
- 纯预览模式
- 无工具栏模式(极简编辑模式)
- 移动端预览模式
功能特性
- 复制 Html 粘贴成 MD 语法
- 经典换行&常规换行
- 多光标编辑
- 图片尺寸
- 导出长图、pdf
- float toolbar 在新行行首支持快速工具栏
- bubble toolbar 选中文字时联想出快速工具栏
性能特性
- 局部渲染
- 局部更新
安全
Cherry Markdown 有内置的安全 Hook,通过过滤白名单以及 DomPurify 进行扫描过滤.
样式主题
Cherry Markdown 有多种样式主题选择
特性展示
这里可以看到特性的简短的演示 screenshot
在线体验
安装
通过 yarn
通过 npm
如果需要开启 mermaid
画图、表格自动转图表功能,需要同时添加mermaid
与echarts
包。
目前Cherry推荐的插件版本为[email protected]
、[email protected]
Quick start
Browser
UMD
ESM
Node
使用轻量版本
因 mermaid 库尺寸非常大,Cherry 构建产物中包含了不内置 mermaid 的核心构建包,可按以下方式引入核心构建。
完整模式 (图形界面)
引擎模式 (语法编译)
⚠️ 关于 mermaid
核心构建包不包含 mermaid 依赖,需要手动引入相关插件。
异步加载
recommend 使用异步引入,以下为 webpack 异步引入样例。
配置
默认配置
代码块语法来彻底取代该语法
- 但在后续的沟通中,开发团队发现在某些场景下该语法有更好的显示效果
- 因此开发团队在6.X版本中才引入了该语法
- 已经引用6.x以下版本的业务如果想做到用户无感知升级,可以去掉该语法:
- indentedCodeBlock:false */ indentedCodeBlock: true, }, emoji: { useUnicode: true, // 是否使用unicode进行渲染 }, fontEmphasis: { allowWhitespace: false, // 是否允许首尾空格 }, strikethrough: { needWhitespace: false, // 是否必须有首位空格 }, mathBlock: { engine: 'MathJax', // katex或MathJax src: '', plugins: true, // 默认加载插件 }, inlineMath: { engine: 'MathJax', // katex或MathJax src: '', }, toc: { /* 默认只渲染一个目录 / allowMultiToc: false, }, header: { /**
- 标题的样式:
-
- default 默认样式,标题前面有锚点
-
- autonumber 标题前面有自增序号锚点
-
- none 标题没有锚点
*/
anchorStyle: 'default',
},
},
},
editor: {
codemirror: {
// depend on codemirror theme name: https://codemirror.net/demo/theme.html
// 自行导入主题文件:
import 'codemirror/theme/<theme-name>.css';
theme: 'default', }, // 编辑器的高度,默认100%,如果挂载点存在内联设置的height则以内联样式为主 height: '100%', // defaultModel 编辑器初始化后的默认模式,一共有三种模式:1、双栏编辑预览模式;2、纯编辑模式;3、预览模式 // edit&preview: 双栏编辑预览模式 // editOnly: 纯编辑模式(没有预览,可通过toolbar切换成双栏或预览模式) // previewOnly: 预览模式(没有编辑框,toolbar只显示“返回编辑”按钮,可通过toolbar切换成编辑模式) defaultModel: 'edit&preview', // 粘贴时是否自动将html转成markdown convertWhenPaste: true, }, toolbars: { theme: 'dark', // light or dark showToolbar: true, // false:不展示顶部工具栏; true:展示工具栏; toolbars.showToolbar=false 与 toolbars.toolbar=false 等效 toolbar: [ 'bold', 'italic', 'strikethrough', '|', 'color', 'header', '|', 'list', { insert: [ 'image', 'audio', 'video', 'link', 'hr', 'br', 'code', 'formula', 'toc', 'table', 'line-table', 'bar-table', 'pdf', 'word', ], }, 'graph', 'settings', ], sidebar: [], bubble: ['bold', 'italic', 'underline', 'strikethrough', 'sub', 'sup', 'quote', '|', 'size', 'color'], // array or false float: ['h1', 'h2', 'h3', '|', 'checklist', 'quote', 'quickTable', 'code'], // array or false }, fileUpload: callbacks.fileUpload, callback: { afterChange: callbacks.afterChange, afterInit: callbacks.afterInit, beforeImageMounted: callbacks.beforeImageMounted, }, previewer: { dom: false, className: 'cherry-markdown', // 是否启用预览区域编辑能力(目前支持编辑图片尺寸、编辑表格内容) enablePreviewerBubble: true, }, // 预览页面不需要绑定事件 isPreviewOnly: false, // 预览区域跟随编辑器光标自动滚动 autoScrollByCursor: true, // 外层容器不存在时,是否强制输出到body上 forceAppend: true, } javascript toolbars: { ... // other config bubble: false, // array or false float: false, // array or false },
- none 标题没有锚点
*/
anchorStyle: 'default',
},
},
},
editor: {
codemirror: {
// depend on codemirror theme name: https://codemirror.net/demo/theme.html
// 自行导入主题文件:
javascript /*
- 生成一个屏蔽敏感词汇的hook
- 名字叫blockSensitiveWords
- 匹配规则会挂载到实例的RULE属性上 */ let BlockSensitiveWordsHook = Cherry.createSyntaxHook('blockSensitiveWords', Cherry.constants.HOOKS_TYPE_LIST.SEN, { makeHtml(str) { return str.replace(this.RULE.reg, '**'); }, rule(str) { return { reg: /敏感词汇/g, }; }, }); new Cherry({ id: 'markdown-container', value: '# welcome to cherry editor!', engine: { customSyntax: { // 注入编辑器的自定义语法中 BlockSensitiveWordsHook: { syntaxClass: BlockSensitiveWordsHook, // 有同名hook则强制覆盖 force: true, // 在处理图片的hook之前调用 // before: 'image', }, }, }, }); javascript /
- 生成一个屏蔽敏感词汇的hook
- 名字叫blockSensitiveWords
- 匹配规则会挂载到实例的RULE属性上
*/
let BlockSensitiveWordsHook = Cherry.createSyntaxHook('blockSensitiveWords', Cherry.constants.HOOKS_TYPE_LIST.PAR, {
// 开启缓存,用于保护内容
needCache: true,
// 预处理文本,避免受影响
beforeMakeHtml(str) {
return str.replace(this.RULE.reg, (match, code) => {
const lineCount = (match.match(/\n/g) || []).length;
const sign = this.$engine.md5(match);
const html =
<div data-sign="${sign}" data-lines="${lineCount + 1}" >***</div>
; return this.pushCache(html, sign, lineCount); }); }, makeHtml(str, sentenceMakeFunc) { return str; }, rule(str) { return { reg: /sensitive words/g, }; }, }); new Cherry({ id: 'markdown-container', value: '# welcome to cherry editor!', engine: { customSyntax: { // 注入编辑器的自定义语法中 BlockSensitiveWordsHook: { syntaxClass: BlockSensitiveWordsHook, // 有同名hook则强制覆盖 force: true, // 在处理图片的hook之前调用 // before: 'image', }, }, }, }); javascript /* - 生成一个添加前缀模板的hook
- 名字叫AddPrefixTemplate
- 图标类名icon-add-prefix
*/
let AddPrefixTemplate = Cherry.createMenuHook('AddPrefixTemplate', 'icon-add-prefix', {
onClick(selection) {
return 'Prefix-' + selection;
},
});
new Cherry({
id: 'markdown-container',
value: '# welcome to cherry editor!',
toolbars: {
toolbar: [
'bold',
// ...其他菜单项
// @see src/Cherry.config.js
'settings',
'addPrefix', // 在顶部菜单栏的尾部添加自定义菜单项
],
bubble: [
'bold' /* ...其他菜单项 /,
,
'color',
'addPrefix', // 在气泡菜单(选中文本时出现)的尾部添加自定义菜单项
], // array or false
float: [
'h1' /* ...其他菜单项 /,
,
'code',
'addPrefix', // 在浮动菜单(在新的空行出现)的尾部添加自定义菜单项
], // array or false
customMenu: {
// 注入编辑器的菜单中
// 对象 key 可以作为菜单项的名字(需要保证唯一),在上方的配置中使用
addPrefix: AddPrefixTemplate,
},
},
});
json
{
"markdown": " \tfoo\tbaz\t\tbim\n",
"html": "
\n", "example": 2, "start_line": 363, "end_line": 368, "section": "Tabs" },foo\tbaz\t\tbim\n
对于这个测试点,Jest 会比对 Cherry.makeHtml(" \tfoo\tbaz\t\tbim\n")
生成的 html 与用例中的预期结果 "<pre><code>foo\tbaz\t\tbim\n</code></pre>\n"
。Cherry Markdown 的匹配器已忽略 data-line
等私有属性。
CommonMark 规范及测试用例可参考:https://spec.commonmark.org/ 。
快照测试
调用 yarn run test:snapshot
运行快照测试,书写用例可参考 test/core/hooks/List.spec.ts
,新用例第一次运行会自动生成快照,第二次会将生成内容与快照进行比对。如果需要重新生成快照,将 test/core/hooks/__snapshots__
下对应的旧快照删除再运行一次即可。
快照测试运行速度较慢,仅在易出错且包含 Cherry 特色语法的 Hook 上使用。
Contributing
欢迎加入我们打造强大的 Markdown 编辑器。当然你也可以给我们提交特性需求的 issue。 在写特性功能之前,你需要了解 commit_convention.
License
Apache-2.0