在线尝试示例:传送门
🔖 MdPreview Props
这是预览组件MdPreview
的Props
,它们同样也是MdEditor
的:
📃 modelValue
-
类型:
string
-
默认值:
''
编辑的内容。
vue<MdEditor v-model="xxx" />
🛍 theme
-
类型:
'light' | 'dark'
-
默认值:
'light'
编辑器主题。
vue<MdEditor theme="dark" />
🎀 class
-
类型:
string
-
默认值:
''
...
🔤 language
-
类型:
string
-
默认值:
'zh-CN'
内置中英文(
'zh-CN'
,'en-US'
),可自行扩展其他语言,同时可覆盖内置的中英文。你也可以使用现成的扩展语言:md-editor-extension。使用及贡献方式见扩展库文档~
🎲 editorId
-
类型:
string
-
默认值:
'md-editor-v-\d'
已过时。5.x 版本开始使用 id 替换。
编辑器唯一标识,使用默认前缀和
useId
拼接。当使用服务端渲染时,请务必设置该属性为固定值,防止产生服务端与客户端渲染内容不一致错误提示。,5.0 开始没有该限制。
🎲 id
-
类型:
string
-
默认值:
'md-editor-v-\d'
编辑器唯一标识,使用默认前缀和
useId
拼接。
🔢 showCodeRowNumber
-
类型:
boolean
-
默认值:
true
代码块是否显示行号。
🔦 previewTheme
-
类型:
'default' | 'github' | 'vuepress' | 'mk-cute' | 'smart-blue' | 'cyanosis'
-
默认值:
'default'
预览内容主题,支持自定义。
主题自定义方式:
- 编辑 css
css.xxx-theme { color: red; }
- 设置
previewTheme
vue<MdEditor previewTheme="xxx" />
参考markdown-theme项目。
🎅🏻 style
-
类型:
string | CSSProperties
-
默认值:
''
编辑器内联样式。
☝️ noMermaid
-
类型:
boolean
-
默认值:
false
如果你不希望使用图表展示内容,可以设置关闭。
vue<MdEditor noMermaid />
❌ noKatex
-
类型:
boolean
-
默认值:
false
如果你不希望使用数学公式展示内容,可以设置关闭。
vue<MdEditor noKatex />
🦉 codeTheme
-
类型:
'atom'|'a11y'|'github'|'gradient'|'kimbie'|'paraiso'|'qtcreator'|'stackoverflow'
-
默认值:
'atom'
代码块高亮样式名称。
你可以添加自己的样式,把该属性设置为你想要的即可,方式如下:
- 配置样式链接
jsimport { config } from 'md-editor-v3'; config({ editorExtensions: { highlight: { css: { atom: { light: 'https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.5.1/styles/atom-one-light.min.css', dark: 'https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.5.1/styles/atom-one-dark.min.css', }, xxx: { light: 'https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.5.1/styles/xxx-light.css', dark: 'https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.5.1/styles/xxx-dark.css', }, }, }, }, });
- 设置
codeTheme
vue<MdEditor codeTheme="xxx" />
🎱 mdHeadingId
-
类型:
(text: string, level: number, index: number) => string
-
默认值:
(text) => text
构造标题
ID
的生成方式。vue<template> <MdEditor :mdHeadingId="mdHeadingId" /> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const mdHeadingId = (_text, _level, index) => `heading-${index}`; </script>
🐣 sanitize
-
类型:
(html: string) => string
-
默认值:
(html) => html
通过该属性修改编译后的 html 内容
该属性为保留属性
基本的危险代码处理方案在 3.x 以后已内置,例如
<script>alert(123)</script>
,4.11.3 之前建议使用该属性来清理更复杂的内容以防止 XSS。在 4.11.3 以后实现了更完善的处理方案,参考
使用
sanitize-html
演示vue<template> <MdEditor :sanitize="sanitize" /> </template> <script setup> import sanitizeHtml from 'sanitize-html'; import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const sanitize = (html) => sanitizeHtml(html); </script>
💅 formatCopiedText
-
类型:
(text: string) => string
-
默认值:
(text) => text
格式化复制代码
vue<template> <MdEditor :formatCopiedText="formatCopiedText" /> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const formatCopiedText = (text) => { return `${text} - from md-editor-v3`; }; </script>
🛁 codeStyleReverse
-
类型:
boolean
-
默认值:
true
某些预览主题的代码模块背景是暗色系,将这个属性设置为 true,会自动在该主题下的 light 模式下使用暗色系的代码风格。
🧼 codeStyleReverseList
-
类型:
Array
-
默认值:
['default', 'mk-cute']
需要自动调整的预览主题,已默认包含 default、mk-cute。
🕊 noHighlight
-
类型:
boolean
-
默认值:
false
不高亮代码,也不会加载相应的扩展库
🕊 noImgZoomIn
-
类型:
boolean
-
默认值:
false
是否关闭编辑器默认的放大功能
vue<MdEditor noImgZoomIn />
在
4.15.4
以后,也可以设置.not-zoom
来禁用它markdown<img class="not-zoom">
😬 customIcon
-
类型:
CustomIcon
-
默认值:
{}
自定义的图标
类型提示
copy、collapse-tips 对应的图标只能是字符串,其他的都可以是组件或者字符串
vue<template> <MdEditor :customIcon="customIcon" /> </template> <script 😬setup lang="ts"> import type { CustomIcon } from 'md-editor-v3'; import { MdEditor, StrIcon } from 'md-editor-v3'; // 假设你使用了三方图标库或者自定义了图标组件 import { IconFont } from 'tdesign-icons-vue-next'; import 'md-editor-v3/lib/style.css'; const customIcon: CustomIcon = { bold: { component: 'A', }, // 演示使用默认图标复制内容 copy: StrIcon('copy', {}), // copy: '<i class="fa fa-car"></i>', // 'collapse-tips': '<i class="fa fa-car"></i>', preview: { component: '<i class="fa fa-car"></i>', }, github: { component: IconFont, props: { name: 'sneer', }, }, }; </script>
类型
CustomIcon
tstype IconName = | 'bold' | 'underline' | 'italic' | 'strike-through' | 'title' | 'sub' | 'sup' | 'quote' | 'unordered-list' | 'ordered-list' | 'task' | 'code-row' | 'code' | 'link' | 'image' | 'table' | 'revoke' | 'next' | 'save' | 'prettier' | 'minimize' | 'maximize' | 'fullscreen-exit' | 'fullscreen' | 'preview-only' | 'preview' | 'preview-html' | 'catalog' | 'github' | 'mermaid' | 'formula' | 'close' | 'delete' | 'upload'; type CustomIcon = { [key in IconName]?: { component: VNode; props: { [key: string | number | symbol]: any; }; }; } & { copy?: string; 'collapse-tips': string; };
💅 sanitizeMermaid
-
类型:
(h: string) => Promise<string>
-
默认值:
(h: string) => Promise.resolve(h)
转换生成的 mermaid 代码
🕹 codeFoldable
-
类型:
boolean
-
默认值:
true
是否开启折叠代码功能
⏲ autoFoldThreshold
-
类型:
number
-
默认值:
30
触发自动折叠代码的行数阈值
🔩 MdEditor Props
除去和MdPreivew
相同的以外:
💻 pageFullscreen
-
类型:
boolean
-
默认值:
false
页面内全屏。
📱 preview
-
类型:
boolean
-
默认值:
true
是否显示预览。
📀 htmlPreview
-
类型:
boolean
-
默认值:
false
是否显示 html 预览。当设置为
true
时,需要将preview
设置为false
jsx<MdEditor htmlPreview preview={false} />
🧱 toolbars
-
类型:
Array
-
默认值:
[all]
选择性展示工具栏,可选内容见下方。
你可以随意排序工具栏,通过
'-'
分割两个工具,通过'='
实现左右放置!从 v1.10.0 开始,你可以自定义工具栏,将
defToolbars
中自定义工具项的下标穿插在toolbars
实现展示(这并不规范)[all]
js[ 'bold', 'underline', 'italic', '-', 'title', 'strikeThrough', 'sub', 'sup', 'quote', 'unorderedList', 'orderedList', 'task', '-', 'codeRow', 'code', 'link', 'image', 'table', 'mermaid', 'katex', '-', 'revoke', 'next', 'save', '=', 'pageFullscreen', 'fullscreen', 'preview', 'previewOnly', 'htmlPreview', 'catalog', 'github', ];
🧱 toolbarsExclude
-
类型:
Array
-
默认值:
[]
选择性不展示工具栏,内容同上。
🪒 noPrettier
-
类型:
boolean
-
默认值:
false
是否启用 prettier 优化 md 内容。
🤏 tabWidth
-
类型:
number
-
默认值:
2
编辑器一个 TAB 键等于空格数。
📅 tableShape
-
类型:
[number, number] \| [number, number, number, number]
-
默认值:
[6, 4]
标题栏添加表格时,预设待选表格大小,第一个代表最大列数,第二个代表最大行数,第三个代表扩展最大列数,第四个代表扩展最大行数。
vue<template> <MdEditor :tableShape="tableShape" /> </tempale> <script setup> const tableShape = [8, 4]; </script>
🪧 placeholder
-
类型:
string
-
默认值:
''
啊这-_-!
🦶 footers
-
类型:
Array<'markdownTotal' | '=' | 'scrollSwitch' | number>
-
默认值:
['markdownTotal', '=', 'scrollSwitch']
页脚显示内容,
'='
左右分割,设置为[]
不显示页脚。
⛵️ scrollAuto
-
类型:
boolean
-
默认值:
true
默认左右同步滚动状态。
🥹 noUploadImg
-
类型:
boolean
-
默认值:
false
工具栏不显示上传图片入口。
vue<template> <MdEditor noUploadImg /> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; </script>
🔬 autoFocus
-
类型:
boolean
-
默认值:
false
原生属性,文本区域自动获得焦点。
🔩 disabled
-
类型:
boolean
-
默认值:
false
原生属性,禁用文本区域。
🔒 readOnly
-
类型:
boolean
-
默认值:
false
原生属性,文本区域为只读。
📏 maxLength
-
类型:
number
-
默认值:``
原生属性,文本区域允许的最大字符数。
📥 autoDetectCode
-
类型:
boolean
-
默认值:
false
是否启用自动识别粘贴代码类别,目前仅支持从
vscode
复制的内容。
📝 completions
-
类型:
Array<CompletionSource>
-
默认值:
[]
添加额外的输入自动完成来源。
vue<template> <MdEditor v-model="text" :completions="completions" /> </template> <script setup lang="ts"> import { ref } from 'vue'; import { CompletionSource } from '@codemirror/autocomplete'; import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const text = ref(''); const completions = ref<Array<CompletionSource>>([ (context) => { const word = context.matchBefore(/@\w*/); if (word === null || (word.from == word.to && context.explicit)) { return null; } return { from: word.from, options: [ { label: '@imzbf', type: 'text', }, ], }; }, ]); </script>
📥 showToolbarName
-
类型:
boolean
-
默认值:
false
是否在工具栏下面显示对应的文字名称
📥 inputBoxWidth
-
类型:
string
-
默认值:
50%
输入框默认的宽度
🪒 transformImgUrl
-
类型:
(imgUrl: string) => string | Promise<string>
-
默认值:
t => t
转换图片链接
🎍 插槽
🪶 defToolbars
自定义工具栏插槽,通过使用内置的NormalToolbar
普通点击触发事件组件,DropdownToolbar
下拉点击触发事件组件和ModalToolbar
弹窗触发事件组件进行扩展。将defToolbars
插槽中的组件下标穿插在toolbars
实现展示(这并不规范)。
-
Setup 模板
vue<template> <MdEditor :toolbars="toolbars"> <template #defToolbars> <NormalToolbar title="mark" @onClick="handler"> <template #trigger> <svg class="md-editor-icon" aria-hidden="true"> <use xlink:href="#icon-mark"></use> </svg> </template> </NormalToolbar> </template> </MdEditor> </template> <script setup> import { MdEditor, NormalToolbar } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const toolbars = ['bold', '-', 0, '=', 'github']; const handler = () => { console.log('NormalToolbar clicked!'); }; </script>
-
Jsx 模板
jsximport { defineComponent } from 'vue'; import { MdEditor, NormalToolbar } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const toolbars = ['bold', '-', 0, '=', 'github']; export default defineComponent({ setup() { return () => ( <MdEditor toolbars={toolbars} defToolbars={ <> <NormalToolbar trigger={ <svg class="md-editor-icon" aria-hidden="true"> <use xlinkHref="#icon-strike-through" /> </svg> } /> </> } /> ); }, });
扩展组件属性参考内置组件,使用示例参见文档分支,提供标记、表情和弹窗预览扩展组件。
🦿 defFooters
自定义扩展页脚
-
Setup 模板
vue<template> <MdEditor :footers="footers"> <template #defFooters> <span>¥_¥</span> <span>^_^</span> </template> </MdEditor> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; // 将插槽中的组件下标放到对应的位置即可显示 const footers = ['markdownTotal', 0, '=', 1, 'scrollSwitch']; </script>
-
Jsx 模板
jsximport { defineComponent } from 'vue'; import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const footers = ['markdownTotal', 0, '=', 1, 'scrollSwitch']; export default defineComponent({ setup() { return () => ( <MdEditor footers={footers} defFooters={ <> <span>¥_¥</span> <span>^_^</span> </> } /> ); }, });
🧵 MdPreview 绑定事件
🚁 onHtmlChanged
-
类型:
(h: string) => void
html 变化回调事件,用于获取预览 html 代码。
🗒 onGetCatalog
-
类型:
(list: HeadList[]) => void
动态获取
markdown
目录。
🪢 MdEditor 绑定事件
除去和MdPreivew
相同的以外:
📞 onChange
-
类型:
(v: string) => void
内容变化事件(当前与
textarea
的oninput
事件绑定,每输入一个单字即会触发)。
💾 onSave
-
类型:
(v: string, h: Promise<string>) => void
保存事件,快捷键与保存按钮均会触发。
vue<template> <MdEditor @onSave="onSave" /> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const onSave = (v, h) => { console.log(v); h.then((html) => { console.log(html); }); }; </script>
📸 onUploadImg
-
类型:
files: Array<File>, callback: (urls: string[] | { url: string; alt: string; title: string }[]) => void
上传图片事件,弹窗会等待上传结果,务必将上传后的 urls 作为 callback 入参回传。
vue<template> <MdEditor v-model="text" @onUploadImg="onUploadImg" /> </template> <script setup> import { ref } from 'vue'; import axios from 'axios'; import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const text = ref('# Hello Editor'); const onUploadImg = async (files, callback) => { const res = await Promise.all( files.map((file) => { return new Promise((rev, rej) => { const form = new FormData(); form.append('file', file); axios .post('/api/img/upload', form, { headers: { 'Content-Type': 'multipart/form-data', }, }) .then((res) => rev(res)) .catch((error) => rej(error)); }); }) ); // 方式一 callback(res.map((item) => item.data.url)); // 方式二 // callback( // res.map((item: any) => ({ // url: item.data.url, // alt: 'alt', // title: 'title' // })) // ); }; </script>
💀 onError
-
类型:
(err: { name: 'Cropper' \| 'fullscreen' \| 'prettier' \| 'overlength'; message: string }) => void
捕获执行错误事件,目前支持
Cropper
、fullscreen
、prettier
实例未加载完成操作,以及输入内容超出限制长度的错误。vue<template> <MdEditor @onError="onError" /> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const onError = (err) => { alert(err.message); }; </script>
🐾 onBlur
-
类型:
(event: FocusEvent) => void
输入框失去焦点时触发事件。
vue<template> <MdEditor @onBlur="onBlur" /> </template> <script setup> import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const onBlur = (e) => { console.log('onBlur', e); }; </script>
🔖 onFocus
-
类型:
(event: FocusEvent) => void
输入框获得焦点时触发事件。
🔖 onInput
-
类型:
(event: Event) => void
输入框键入内容事件。
🔖 onDrop
-
类型:
(event: DragEvent) => void
拖放内容事件。
vue<template> <MdEditor v-model="text" @onDrop="onDrop" /> </template> <script setup lang="ts"> import { ref } from 'vue'; import { MdEditor } from 'md-editor-v3'; import 'md-editor-v3/lib/style.css'; const text = ref(''); const onDrop = (e: DragEvent) => { e.stopPropagation(); console.log('ee', e.dataTransfer?.files[0]); }; </script>
🔖 onInputBoxWidthChange
-
类型:
(width: string) => void
调整输入框宽度事件
🤱🏼 实例暴露
2.5.0 版本之后,编辑器暴露了若干方法在组件实例上,用来快捷监听编辑器内部状态或对调整内部状态。
vue
<template>
<MdEditor ref="editorRef" />
</template>
<script setup lang="ts">
import { ref, onMounted } from 'vue';
import { MdEditor } from 'md-editor-v3';
import type { ExposeParam } from 'md-editor-v3';
//
// import type { ExposePreviewParam } from 'md-editor-v3';
import 'md-editor-v3/lib/style.css';
const editorRef = ref<ExposeParam>();
onMounted(() => {
editorRef.value?.on('catalog', console.log);
});
</script>
Name | MdEditor | MdPreview |
---|---|---|
on | √ | × |
togglePageFullscreen | √ | × |
toggleFullscreen | √ | × |
togglePreview | √ | × |
togglePreviewOnly | √ | × |
toggleHtmlPreview | √ | × |
toggleCatalog | √ | × |
triggerSave | √ | × |
insert | √ | × |
focus | √ | × |
rerender | √ | √ |
getSelectedText | √ | × |
resetHistory | √ | × |
👂🏼 on
监听编辑器内部状态,包括:屏幕全屏、浏览器全屏、预览文本、预览 html、目录等。
-
pageFullscreen
jseditorRef.value?.on('pageFullscreen', (status) => console.log(status));
-
fullscreen
jseditorRef.value?.on('fullscreen', (status) => console.log(status));
-
preview
jseditorRef.value?.on('preview', (status) => console.log(status));
-
previewOnly
jseditorRef.value?.on('previewOnly', (status) => console.log(status));
-
htmlPreview
jseditorRef.value?.on('htmlPreview', (status) => console.log(status));
-
catalog
jseditorRef.value?.on('catalog', (status) => console.log(status));
💻 togglePageFullscreen
切换页面内全屏。
js
editorRef.value?.togglePageFullscreen(true);
不设置入参切换为相反状态
🖥 toggleFullscreen
切换屏幕全屏。
js
editorRef.value?.toggleFullscreen(true);
不设置入参切换为相反状态
📖 togglePreview
切换是否显示预览。
js
editorRef.value?.togglePreview(true);
不设置入参切换为相反状态
📖 togglePreviewOnly
切换仅预览状态。
js
editorRef.value?.togglePreviewOnly(true);
不设置入参切换为相反状态
📼 toggleHtmlPreview
切换是否显示 html 预览。
js
editorRef.value?.toggleHtmlPreview(true);
不设置入参切换为相反状态
🧬 toggleCatalog
切换是否显示目录。
js
editorRef.value?.toggleCatalog(true);
不设置入参切换为相反状态
💾 triggerSave
触发保存。
js
editorRef.value?.triggerSave();
💉 insert
手动向文本框插入内容。
js
/**
* @params selectedText 选中的内容
*/
editorRef.value?.insert((selectedText) => {
/**
* @return targetValue 待插入内容
* @return select 插入后是否自动选中内容,默认:true
* @return deviationStart 插入后选中内容鼠标开始位置,默认:0
* @return deviationEnd 插入后选中内容鼠标结束位置,默认:0
*/
return {
targetValue: `${selectedText}`,
select: true,
deviationStart: 0,
deviationEnd: 0,
};
});
🎯 focus
手动聚焦输入框。
ts
import type { FocusOption } from 'md-editor-v3';
const option: FocusOption | undefined = 'start';
// 聚焦时光标的位置,不提供默认上次失焦时的位置
editorRef.value?.focus(option);
ts
type FocusOption =
| 'start'
| 'end'
| {
// 选中的开始位置,默认光标位置
rangeAnchor?: number;
// 选中的结束位置,默认光标位置
rangeHead?: number;
// 光标的位置
cursorPos: number;
};
✒️ rerender
手动重新渲染内容。
js
editorRef.value?.rerender();
🔍 getSelectedText
获取当前选中的文字。
js
console.log(editorRef.value?.getSelectedText());
🗑 resetHistory
清除当前的历史记录。
🎛 domEventHandlers
支持监听全部的 dom 事件。
js
editorRef.value?.domEventHandlers({
compositionstart: () => {
console.log('compositionstart');
},
});
🎛 execCommand
通过触发器向编辑器插入内容。
js
editorRef.value?.execCommand('bold');
🔖 getEditorView
获取 codemirror 实例。
💴 配置编辑器
使用config(option: ConfigOption)
方法,可以对构建实例进行定制。
我们建议你在项目入口配置,例如 vite 创建的项目中的 main.js。不要在组件中去调用 config
!
🦪 codeMirrorExtensions
根据主题和内部默认的 codeMirror 扩展自定义新的扩展。
ts
type CodeMirrorExtensions = (
theme: Themes,
extensions: Array<Extension>,
keyBindings: Array<KeyBinding>,
options: {
editorId: string;
}
) => Array<Extension>;
使用示例:编辑器默认不显示输入框的行号,需要手动添加扩展
js
import { config } from 'md-editor-v3';
import { lineNumbers } from '@codemirror/view';
config({
codeMirrorExtensions(_theme, extensions) {
return [...extensions, lineNumbers()];
},
});
🍤 markdownItConfig
自定义 markdown-it 核心库扩展、属性等。
ts
type MarkdownItConfig = (
md: markdownit,
options: {
editorId: string;
}
) => void;
使用示例:配置使用markdown-it-anchor
并在标题右侧显示一个超链接符号
js
import { config } from 'md-editor-v3';
import ancher from 'markdown-it-anchor';
config({
markdownItConfig(mdit) {
mdit.use(ancher, {
permalink: true,
});
},
});
🍤 markdownItPlugins
挑选、新增 markdown-it 核心库已预设的扩展。
ts
type MarkdownItPlugins = (
plugins: Array<MarkdownItConfigPlugin>,
options: {
editorId: string;
}
) => Array<MarkdownItConfigPlugin>;
使用示例:修改图片的类名
js
import { config } from 'md-editor-v3';
config({
markdownItPlugins(plugins) {
return plugins.map((p) => {
if (p.type === 'image') {
return {
...p,
options: {
...p.options,
classes: 'my-class',
},
};
}
return p;
});
},
});
🍙 editorConfig
编辑器常规配置,语言、mermaid
默认模板、渲染延迟:
🍚 languageUserDefined
js
import { config } from 'md-editor-v3';
config({
editorConfig: {
// 语言
languageUserDefined: {
'zh-CN': {
toolbarTips: {
bold: '加粗',
underline: '下划线',
italic: '斜体',
strikeThrough: '删除线',
title: '标题',
sub: '下标',
sup: '上标',
quote: '引用',
unorderedList: '无序列表',
orderedList: '有序列表',
task: '任务列表',
codeRow: '行内代码',
code: '块级代码',
link: '链接',
image: '图片',
table: '表格',
mermaid: 'mermaid图',
katex: 'katex公式',
revoke: '后退',
next: '前进',
save: '保存',
prettier: '美化',
pageFullscreen: '浏览器全屏',
fullscreen: '屏幕全屏',
preview: '预览',
previewOnly: '仅预览',
htmlPreview: 'html代码预览',
catalog: '目录',
github: '源码地址',
},
titleItem: {
h1: '一级标题',
h2: '二级标题',
h3: '三级标题',
h4: '四级标题',
h5: '五级标题',
h6: '六级标题',
},
imgTitleItem: {
link: '添加链接',
upload: '上传图片',
clip2upload: '裁剪上传',
},
linkModalTips: {
linkTitle: '添加链接',
imageTitle: '添加图片',
descLabel: '链接描述:',
descLabelPlaceHolder: '请输入描述...',
urlLabel: '链接地址:',
urlLabelPlaceHolder: '请输入链接...',
buttonOK: '确定',
},
clipModalTips: {
title: '裁剪图片上传',
buttonUpload: '上传',
},
copyCode: {
text: '复制代码',
successTips: '已复制!',
failTips: '复制失败!',
},
mermaid: {
flow: '流程图',
sequence: '时序图',
gantt: '甘特图',
class: '类图',
state: '状态图',
pie: '饼图',
relationship: '关系图',
journey: '旅程图',
},
katex: {
inline: '行内公式',
block: '块级公式',
},
footer: {
markdownTotal: '字数',
scrollAuto: '同步滚动',
},
},
},
},
});
🍘 mermaidTemplate
js
import { config } from 'md-editor-v3';
config({
editorConfig: {
// mermaid模板
mermaidTemplate: {
// 流程图
flow: `flow tempalte`,
// 时序图
sequence: `sequence template`,
// 甘特图
gantt: `gantt template`,
// 类图
class: `class template`,
// 状态图
state: `state template`,
// 饼图
pie: `pie template`,
// 关系图
relationship: `relationship template`,
// 旅程图
journey: `journey template`,
},
},
});
🍥 renderDelay
js
import { config } from 'md-editor-v3';
config({
editorConfig: {
// 输入渲染延迟(ms)
renderDelay: 0,
},
});
🍥 zIndex
js
import { config } from 'md-editor-v3';
config({
editorConfig: {
// 内部弹窗的zIndex
zIndex: 2000,
},
});
🥠 editorExtensions
类型如下,用于配置编辑器内部的扩展
typescript
import { config } from 'md-editor-v3';
config({
editorExtensions: { highlight: { js: 'https://xxx.cc' } },
});
ts
export interface EditorExtensions {
highlight?: {
instance?: any;
js?: string;
css?: {
[key: string]: {
light: string;
dark: string;
};
};
};
prettier?: {
// >= 2.2.0
prettierInstance?: any;
parserMarkdownInstance?: any;
standaloneJs?: string;
parserMarkdownJs?: string;
};
cropper?: {
instance?: any;
js?: string;
css?: string;
};
screenfull?: {
instance?: any;
js?: string;
};
mermaid?: {
instance?: any;
js?: string;
};
katex?: {
instance?: any;
js?: string;
css?: string;
};
}
🥠 editorExtensionsAttrs
同步添加 CDN 链接标签的上属性,类型与editorExtensions
一直,值类型是HTMLElementTagNameMap<script|link>
内部提供所有链接的integrity
值,使用方式如下:
js
import { config } from 'md-editor-v3';
config({
editorExtensionsAttrs: {
highlight: {
js: {
className: 'hglh-js',
},
css: {
atom: {
light: {
className: 'atom-light-css',
},
dark: {
className: 'atom-dark-css',
},
},
},
},
},
});
使用内置的基础属性配置示例:
js
import { config, editorExtensionsAttrs } from 'md-editor-v3';
config({
editorExtensionsAttrs,
});
提醒
不要尝试在 editorExtensionsAttrs 定义 script 的 src\onload\id,link 的 rel\href\id 它们会被默认值覆盖
🎨 mermaidConfig
mermaid 配置项,配置详情
js
import { config } from 'md-editor-v3';
config({
mermaidConfig(base: any) {
return {
...base,
logLevel: 'error',
};
},
});
🔧 katexConfig
katex 配置项,配置详情
js
import { config } from 'md-editor-v3';
config({
katexConfig(base: any) {
return {
...base,
strict: false,
};
},
});
🪡 快捷键
主要以CTRL
搭配对应功能英文单词首字母,冲突项添加SHIFT
,再冲突替换为ALT
。
注意事项
快捷键仅在输入框获取到焦点时可用!
键位 | 功能 | 说明 |
---|---|---|
TAB | 空格 | 通过tabWidth 属性预设 TAB 键位新增空格长度,默认 2,支持多行 |
SHIFT + TAB | 取消空格 | 同上,一次取消两个空格,支持多行 |
CTRL + C | 复制 | 选中时复制选中内容,未选中时复制当前行内容 |
CTRL + X | 剪切 | 选中时剪切选中内容,未选中时剪切当前行 |
CTRL + D | 删除 | 选中时删除选中内容,未选中时删除当前行 |
CTRL + S | 保存 | 触发编辑器的onSave 回调 |
CTRL + B | 加粗 | **加粗** |
CTRL + U | 下划线 | <u>下划线</u> |
CTRL + I | 斜体 | *斜体* |
CTRL + 1-6 | 1-6 级标题 | # 标题 |
CTRL + ↑ | 上角标 | <sup>上角标</sup> |
CTRL + ↓ | 下角标 | <sub>下角标</sub> |
CTRL + O | 有序列表 | 1. 有序列表 |
CTRL + L | 链接 | [链接](https://github.com/imzbf) |
CTRL + Z | 撤回 | 触发编辑器内内容撤回,与系统无关 |
CTRL + F | 查找替换 | |
CTRL + SHIFT + S | 删除线 | ~删除线~ |
CTRL + SHIFT + U | 无序列表 | - 无序列表 |
CTRL + SHIFT + C | 块级代码 | 多行代码块 |
CTRL + SHIFT + I | 图片链接 | ![图片](https://github.com/imzbf) |
CTRL + SHIFT + Z | 前进一步 | 触发编辑器内内容前进,与系统无关 |
CTRL + SHIFT + F | 美化内容 | |
CTRL + ALT + C | 行内代码 | 行内代码块 |
CTRL + SHIFT + ALT + T | 表格 | |表格| |
🪤 内置组件
按需引用编辑器的扩展组件,例如:import { DropdownToolbar } from 'md-editor-v3'
。
内置属性提示
为了帮助开发者快速插入和使用编辑器的属性,编辑器组件已经默认向头部工具栏和尾部工具栏中的扩展组件添加了下面的属性的值(如果你也提供了,那么会优先使用你提供的内容),更详细的参考示例:ExportPDF
名称 | defToolbars | defFooters |
---|---|---|
insert | √ | × |
theme | √ | √ |
previewtheme | √ | × |
codeTheme | √ | × |
language | √ | √ |
disabled | √ | √ |
例子:
vue
<!-- HeaderTool.vue -->
<template>
<NormalToolbar>触发</NormalToolbar>
</template>
<script setup>
const props = defineProps({
theme: {
type: String,
},
insert: {
type: Function,
},
...
});
console.log('==', props);
// == { insert: (...)=> {...}, theme: 'light', ... }
</script>
<!-- MyEditor1.vue -->
<template>
<MdEditor :toolbars="toolbars">
<template #defToolbars>
<HeaderTool key="key" />
</template>
</MdEditor>
</template>
<script setup>
const toolbars = [0];
</script>
<!-- =================================== -->
<!-- FooterTool.vue -->
<template>
<NormalFooterToolbar>触发</NormalFooterToolbar>
</template>
<script setup>
const props = defineProps({
theme: {
type: String,
},
language: {
type: String,
},
disabled: {
type: Boolean,
},
});
console.log('==', props);
// == { theme: 'light', disabled: false, language: 'zh-CN' }
</script>
<!-- MyEditor2.vue -->
<template>
<MdEditor :footers="footers">
<template #defFooters>
<HeaderTool key="key" />
</template>
</MdEditor>
</template>
<script setup>
const footers = [0];
</script>
🐣 NormalToolbar
-
props
title
:string
,非必须,作为工具栏上的 hover 提示。
-
events
onClick
:(e: MouseEvent) => void
,必须,点击事件。
-
slots
default
:any
,非必须,通常是个图标,用来展示在工具栏上。:trigger
string | VNode
,非必须,已废弃,同上。
vue
<template>
<NormalToolbar title="mark" @onClick="handler">
<svg class="md-editor-icon" aria-hidden="true">
<use xlink:href="#icon-mark"></use>
</svg>
</NormalToolbar>
</template>
<script setup lang="ts">
import { PropType } from 'vue';
import { NormalToolbar } from 'md-editor-v3';
import type { Insert } from 'md-editor-v3';
const props = defineProps({
/**
* `insert`方法会由编辑器自动向组件的组件注入。
*/
insert: {
type: Function as PropType<Insert>,
default: () => {
//
},
},
});
const handler = () => {
props.insert((selectedText) => {
/**
* targetValue 待插入内容
* select 插入后是否自动选中内容,默认:true
* deviationStart 插入后选中内容鼠标开始位置,默认:0
* deviationEnd 插入后选中内容鼠标结束位置,默认:0
*/
return {
targetValue: `==${selectedText}==`,
select: true,
deviationStart: 0,
deviationEnd: 0,
};
});
};
</script>
vue
<template>
<MdEditor v-model="text" :toolbars="toolbars">
<template #defToolbars>
<MyToolbar />
</template>
</MdEditor>
</template>
<script setup lang="ts">
import { ref } from 'vue';
import { MdEditor } from 'md-editor-v3';
import MyToolbar from './MyToolbar.vue';
import 'md-editor-v3/lib/style.css';
const text = ref('');
const toolbars = ['bold', 0, 'github'];
</script>
🐼 DropdownToolbar
-
props
title
:string
,非必须,作为工具栏上的 hover 提示。visible
:boolean
,必须,下拉状态。
-
events
onChange
:(visible: boolean) => void
,必须,状态变化事件。
-
slots
default
:any
,非必须,通常是个图标,用来展示在工具栏上。:trigger
string | VNode
,非必须,已废弃,同上。overlay
:string | VNode
,必须,下拉框中的内容。
vue
<template>
<DropdownToolbar title="emoji" :visible="visible" :onChange="onChange">
<template #overlay>
<div class="emoji-container">
<ol class="emojis">
<li
v-for="(emoji, index) of emojis"
:key="`emoji-${index}`"
@click="handler(emoji)"
v-text="emoji"
></li>
</ol>
</div>
</template>
<svg class="md-editor-icon" aria-hidden="true">
<use xlink:href="#icon-emoji"></use>
</svg>
</DropdownToolbar>
</template>
<script setup lang="ts">
import { PropType, ref } from 'vue';
import { DropdownToolbar } from 'md-editor-v3';
import type { Insert } from 'md-editor-v3';
const emojis = ['😀', '😃'];
const props = defineProps({
/**
* `insert`方法会由编辑器自动向组件的组件注入。
*/
insert: {
type: Function as PropType<Insert>,
default: () => {
//
},
},
});
const visible = ref(false);
const onChange = (_visible: boolean) => {
visible.value = _visible;
};
const handler = (emoji: any) => {
props.insert(() => {
/**
* targetValue 待插入内容
* select 插入后是否自动选中内容,默认:true
* deviationStart 插入后选中内容鼠标开始位置,默认:0
* deviationEnd 插入后选中内容鼠标结束位置,默认:0
*/
return {
targetValue: emoji,
select: true,
deviationStart: 0,
deviationEnd: 0,
};
});
};
</script>
vue
<template>
<MdEditor v-model="text" :toolbars="toolbars">
<template #defToolbars>
<MyToolbar />
</template>
</MdEditor>
</template>
<script setup lang="ts">
import { ref } from 'vue';
import { MdEditor } from 'md-editor-v3';
import MyToolbar from './MyToolbar.vue';
import 'md-editor-v3/lib/style.css';
const text = ref('');
const toolbars = ['bold', 0, 'github'];
</script>
🦉 ModalToolbar
-
props
title
:string
,非必须,作为工具栏上的 hover 提示。visible
:boolean
,必须,弹窗显示状态。width
:string
,非必须,弹窗宽度,默认auto
。height
:string
,同width
。showAdjust
:boolean
,非必须,是否显示弹窗全屏按钮。isFullscreen
:boolean
,显示全屏按钮时必须,弹窗全屏状态。class
:string
,^4.16.8
,非必须,类名。style
:CSSProperties | string
,^4.16.8
,非必须,样式。showMask
:boolean
,^4.16.8
,非必须,是否展示遮罩层,默认 true。
-
events
onClick
:() => void
,必须,工具栏点击事件。onClose
:() => void
,必须,弹窗点击关闭事件。onAdjust
:(val: boolean) => void
,弹窗全屏按钮点击事件。
-
slots
modalTitle
:string | VNode
,非必须,弹窗标题栏。trigger
:string | VNode
,必须,通常是个图标,用来展示在工具栏上。default
:any
,非必须,弹窗中的内容。
vue
<template>
<ModalToolbar
:visible="data.modalVisible"
:isFullscreen="data.modalFullscreen"
showAdjust
title="Preview"
modalTitle="Page Preview"
width="870px"
height="600px"
@onClick="data.modalVisible = true"
@onClose="data.modalVisible = false"
@onAdjust="data.modalFullscreen = !data.modalFullscreen"
>
<button @click="handler">Click me</button>
<template #trigger>
<svg class="md-editor-icon" aria-hidden="true">
<use xlink:href="#icon-read"></use>
</svg>
</template>
</ModalToolbar>
</template>
<script setup lang="ts">
import { PropType, reactive } from 'vue';
import { ModalToolbar } from 'md-editor-v3';
import type { Insert } from 'md-editor-v3';
const data = reactive({
modalVisible: false,
modalFullscreen: false,
});
const props = defineProps({
/**
* `insert`方法会由编辑器自动向组件的组件注入。
*/
insert: {
type: Function as PropType<Insert>,
default: () => {
//
},
},
});
const handler = () => {
props.insert((selectedText) => {
/**
* targetValue 待插入内容
* select 插入后是否自动选中内容,默认:true
* deviationStart 插入后选中内容鼠标开始位置,默认:0
* deviationEnd 插入后选中内容鼠标结束位置,默认:0
*/
return {
targetValue: `==${selectedText}==`,
select: true,
deviationStart: 0,
deviationEnd: 0,
};
});
};
</script>
vue
<template>
<MdEditor v-model="text" :toolbars="toolbars">
<template #defToolbars>
<MyToolbar />
</template>
</MdEditor>
</template>
<script setup lang="ts">
import { ref } from 'vue';
import { MdEditor } from 'md-editor-v3';
import MyToolbar from './MyToolbar.vue';
import 'md-editor-v3/lib/style.css';
const text = ref('');
const toolbars = ['bold', 0, 'github'];
</script>
🐻 MdCatalog
-
props
editorId
:string
,必须,对应编辑器的id
,在内部注册目录变化监听事件。class
:string
,非必须,目录组件最外层类名。mdHeadingId
:mdHeadingId
,非必须,特殊化编辑器标题的算法,与编辑器相同。scrollElement
:string | HTMLElement
,非必须,为字符时应是一个元素选择器。仅预览模式中,整页滚动时,设置为document.documentElement
。theme
:'light' | 'dark'
,非必须,当需要切换主题时提供,同编辑器的theme
。offsetTop
:number
,非必须,标题距离顶部该像素时高亮当前目录项,默认 20 像素。scrollElementOffsetTop
:number
,非必须,滚动区域的固定顶部高度,默认 0。
-
events
onClick
:(e: MouseEvent, t: TocItem) => void
,非必须,导航点击事件。onActive
:(heading: HeadList | undefined) => void
,非必须,高亮的标题变化事件。
scrollElement
说明:仅预览下,该元素必须已定位的并且支持滚动。
vue
<template>
<!-- 保证editorId相同 -->
<MdPreview :id="state.id" :modelValue="state.text" :theme="state.theme" />
<MdCatalog
:editorId="state.id"
:scrollElement="scrollElement"
:theme="state.theme"
/>
</template>
<script setup>
import { reactive } from 'vue';
import { MdPreview, MdCatalog } from 'md-editor-v3';
import 'md-editor-v3/lib/style.css';
const state = reactive({
theme: 'dark',
text: '标题',
id: 'my-editor',
});
const scrollElement = document.documentElement;
</script>
🛸 MdModal
编辑器内部的弹窗组件,它通常配合下拉工具栏组件使用。
-
props
visible
:boolean
,必须,弹窗显示状态。width
:string
,非必须,弹窗宽度,默认auto
。height
:string
,同width
。showAdjust
:boolean
,非必须,是否显示弹窗全屏按钮。isFullscreen
:boolean
,显示全屏按钮时必须,弹窗全屏状态。class
:string
,非必须,类名。style
:CSSProperties | string
,非必须,样式。showMask
:boolean
,^4.16.8
,非必须,是否展示遮罩层,默认 true。
-
events
onClose
:() => void
,必须,弹窗点击关闭事件。onAdjust
:(val: boolean) => void
,弹窗全屏按钮点击事件。
-
slots
title
:string | VNode
,非必须,弹窗标题栏。default
:any
,非必须,弹窗中的内容。
vue
<template>
<DropdownToolbar title="emoji" :visible="state.visible" :onChange="onChange">
<template #overlay>
<ul>
<li @click="state.mVisible = true">option 1</li>
<li>option 2</li>
</ul>
</template>
<template #trigger>
<svg class="md-editor-icon" aria-hidden="true">
<use xlink:href="#icon-emoji"></use>
</svg>
</template>
<template #default>
<MdModal title="title" :visible="state.mVisible" @onClose="onClose">
Content, Content
</MdModal>
</template>
</DropdownToolbar>
</template>
<script setup lang="ts">
import { reactive } from 'vue';
import { DropdownToolbar, MdModal } from 'md-editor-v3';
const state = reactive({
visible: false,
mVisible: false,
});
const onClose = () => {
state.mVisible = false;
};
const onChange = (_visible: boolean) => {
visible.value = _visible;
};
</script>
🛸 NormalFooterToolbar
通用的页脚工具组件
-
events
onClick
:(e: MouseEvent) => void
,非必须,点击事件。
-
slots
default
:any
,必须,内容。
vue
<!-- FooterTool.vue -->
<template>
<NormalFooterToolbar>触发</NormalFooterToolbar>
</template>
<script>
import { MdEditor, NormalFooterToolbar } from 'md-editor-v3';
</script>
<!-- MyEditor.vue -->
<template>
<MdEditor :footers="footers">
<template #defFooters>
<FooterTool key="key" />
</template>
</MdEditor>
</template>
<script setup>
import { MdEditor, NormalFooterToolbar } from 'md-editor-v3';
const footers = [0];
</script>
🪤 内部配置
js
import {
allToolbar,
allFooter,
zh_CN,
en_US,
editorExtensionsAttrs,
} from 'md-editor-v3';
console.log(allToolbar, allFooter, zh_CN, en_US, editorExtensionsAttrs);
📦 内部工具
🧹 clearSideEffects
>=5.0.0
清空组件带来的副作用,例如
- 使用 CDN 嵌入的链接,为了保证多个组件能够正常使用,组件在卸载时不会主动移除
js
import { clearSideEffects } from 'md-editor-v3';
clearSideEffects();
这会删除掉页面中的 CDN 引用,如果当前页面还存在该组件实例,请不要这样做!
XSSPlugin
>=5.0.0
使用示例:添加 xss 扩展