使用这个主要是为了Word文件的内容能够还原样式字体,其实也不太完善,但是大部分够用了,也可以自己二次修改
官网地址:hufe.club/canvas-edit…
1.官网打开,直接点击git地址打开,不用根据文档写,因为安装了什么都没有,直接是下载项目集成的
在git上下载整个项目,只需要src里面所有的文件
2.在项目中components目录新建CanvasEditor文件夹,将文件都复制进去
新增index.vue文件
<template>
<div class="container">
<div class="menu" editor-component="menu">
<div class="menu-item">
<div class="menu-item__undo">
<i></i>
</div>
<div class="menu-item__redo">
<i></i>
</div>
<div class="menu-item__painter" title="格式刷(双击可连续使用)">
<i></i>
</div>
<div class="menu-item__format" title="清除格式">
<i></i>
</div>
</div>
<div class="menu-divider"></div>
<div class="menu-item">
<div class="menu-item__font">
<span class="select" title="字体">微软雅黑</span>
<div class="options">
<ul>
<li data-family="Microsoft YaHei" style="font-family:'Microsoft YaHei';">微软雅黑</li>
<li data-family="华文宋体" style="font-family:'华文宋体';">华文宋体</li>
<li data-family="华文黑体" style="font-family:'华文黑体';">华文黑体</li>
<li data-family="华文仿宋" style="font-family:'华文仿宋';">华文仿宋</li>
<li data-family="华文楷体" style="font-family:'华文楷体';">华文楷体</li>
<li data-family="华文琥珀" style="font-family:'华文琥珀';">华文琥珀</li>
<li data-family="华文楷体" style="font-family:'华文楷体';">华文楷体</li>
<li data-family="华文隶书" style="font-family:'华文隶书';">华文隶书</li>
<li data-family="华文新魏" style="font-family:'华文新魏';">华文新魏</li>
<li data-family="华文行楷" style="font-family:'华文行楷';">华文行楷</li>
<li data-family="华文中宋" style="font-family:'华文中宋';">华文中宋</li>
<li data-family="华文彩云" style="font-family:'华文彩云';">华文彩云</li>
<li data-family="Arial" style="font-family:'Arial';">Arial</li>
<li data-family="Segoe UI" style="font-family:'Segoe UI';">Segoe UI</li>
<li data-family="Ink Free" style="font-family:'Ink Free';">Ink Free</li>
<li data-family="Fantasy" style="font-family:'Fantasy';">Fantasy</li>
</ul>
</div>
</div>
<div class="menu-item__size">
<span class="select" title="字体">小四</span>
<div class="options">
<ul>
<li data-size="56">初号</li>
<li data-size="48">小初</li>
<li data-size="34">一号</li>
<li data-size="32">小一</li>
<li data-size="29">二号</li>
<li data-size="24">小二</li>
<li data-size="21">三号</li>
<li data-size="20">小三</li>
<li data-size="18">四号</li>
<li data-size="16">小四</li>
<li data-size="14">五号</li>
<li data-size="12">小五</li>
<li data-size="10">六号</li>
<li data-size="8">小六</li>
<li data-size="7">七号</li>
<li data-size="6">八号</li>
</ul>
</div>
</div>
<div class="menu-item__size-add">
<i></i>
</div>
<div class="menu-item__size-minus">
<i></i>
</div>
<div class="menu-item__bold">
<i></i>
</div>
<div class="menu-item__italic">
<i></i>
</div>
<div class="menu-item__underline">
<i></i>
<span class="select"></span>
<div class="options">
<ul>
<li data-decoration-style='solid'>
<i></i>
</li>
<li data-decoration-style='double'>
<i></i>
</li>
<li data-decoration-style='dashed'>
<i></i>
</li>
<li data-decoration-style='dotted'>
<i></i>
</li>
<li data-decoration-style='wavy'>
<i></i>
</li>
</ul>
</div>
</div>
<div class="menu-item__strikeout" title="删除线(Ctrl+Shift+X)">
<i></i>
</div>
<div class="menu-item__superscript">
<i></i>
</div>
<div class="menu-item__subscript">
<i></i>
</div>
<div class="menu-item__color" title="字体颜色">
<i></i>
<span></span>
<input type="color" id="color" />
</div>
<div class="menu-item__highlight" title="高亮">
<i></i>
<span></span>
<input type="color" id="highlight">
</div>
</div>
<div class="menu-divider"></div>
<div class="menu-item">
<div class="menu-item__title">
<i></i>
<span class="select" title="切换标题">正文</span>
<div class="options">
<ul>
<li style="font-size:16px;">正文</li>
<li data-level="first" style="font-size:26px;">标题1</li>
<li data-level="second" style="font-size:24px;">标题2</li>
<li data-level="third" style="font-size:22px;">标题3</li>
<li data-level="fourth" style="font-size:20px;">标题4</li>
<li data-level="fifth" style="font-size:18px;">标题5</li>
<li data-level="sixth" style="font-size:16px;">标题6</li>
</ul>
</div>
</div>
<div class="menu-item__left">
<i></i>
</div>
<div class="menu-item__center">
<i></i>
</div>
<div class="menu-item__right">
<i></i>
</div>
<div class="menu-item__alignment">
<i></i>
</div>
<div class="menu-item__justify">
<i></i>
</div>
<div class="menu-item__row-margin">
<i title="行间距"></i>
<div class="options">
<ul>
<li data-rowmargin='1'>1</li>
<li data-rowmargin="1.25">1.25</li>
<li data-rowmargin="1.5">1.5</li>
<li data-rowmargin="1.75">1.75</li>
<li data-rowmargin="2">2</li>
<li data-rowmargin="2.5">2.5</li>
<li data-rowmargin="3">3</li>
</ul>
</div>
</div>
<div class="menu-item__list">
<i></i>
<div class="options">
<ul>
<li>
<label>取消列表</label>
</li>
<li data-list-type="ol" data-list-style='decimal'>
<label>有序列表:</label>
<ol>
<li>________</li>
</ol>
</li>
<li data-list-type="ul" data-list-style='checkbox'>
<label>复选框列表:</label>
<ul style="list-style-type: '☑️ ';">
<li>________</li>
</ul>
</li>
<li data-list-type="ul" data-list-style='disc'>
<label>实心圆点列表:</label>
<ul style="list-style-type: disc;">
<li>________</li>
</ul>
</li>
<li data-list-type="ul" data-list-style='circle'>
<label>空心圆点列表:</label>
<ul style="list-style-type: circle;">
<li>________</li>
</ul>
</li>
<li data-list-type="ul" data-list-style='square'>
<label>空心方块列表:</label>
<ul style="list-style-type: '☐ ';">
<li>________</li>
</ul>
</li>
</ul>
</div>
</div>
</div>
<div class="menu-divider"></div>
<div class="menu-item">
<div class="menu-item__table">
<i title="表格"></i>
</div>
<div class="menu-item__table__collapse">
<div class="table-close">×</div>
<div class="table-title">
<span class="table-select">插入</span>
<span>表格</span>
</div>
<div class="table-panel"></div>
</div>
<div class="menu-item__image">
<i title="图片"></i>
<input type="file" id="image" accept=".png, .jpg, .jpeg, .svg, .gif">
</div>
<div class="menu-item__hyperlink">
<i title="超链接"></i>
</div>
<div class="menu-item__separator">
<i title="分割线"></i>
<div class="options">
<ul>
<li data-separator='0,0'>
<i></i>
</li>
<li data-separator="1,1">
<i></i>
</li>
<li data-separator="3,1">
<i></i>
</li>
<li data-separator="4,4">
<i></i>
</li>
<li data-separator="7,3,3,3">
<i></i>
</li>
<li data-separator="6,2,2,2,2,2">
<i></i>
</li>
</ul>
</div>
</div>
<div class="menu-item__watermark">
<i title="水印(添加、删除)"></i>
<div class="options">
<ul>
<li data-menu="add">添加水印</li>
<li data-menu="delete">删除水印</li>
</ul>
</div>
</div>
<div class="menu-item__codeblock" title="代码块">
<i></i>
</div>
<div class="menu-item__page-break" title="分页符">
<i></i>
</div>
<div class="menu-item__control" style="display: none">
<i title="控件"></i>
<div class="options">
<ul>
<li data-control='text'>文本</li>
<li data-control="number">数值</li>
<li data-control="select">列举</li>
<li data-control="date">日期</li>
<li data-control="checkbox">复选框</li>
<li data-control="radio">单选框</li>
</ul>
</div>
</div>
<div class="menu-item__checkbox" title="复选框">
<i></i>
</div>
<div class="menu-item__radio" title="单选框">
<i></i>
</div>
<div class="menu-item__latex" title="LateX" style="display: none">
<i></i>
</div>
<div class="menu-item__date">
<i title="日期"></i>
<div class="options">
<ul>
<li data-format="yyyy-MM-dd"></li>
<li data-format="yyyy-MM-dd hh:mm:ss"></li>
</ul>
</div>
</div>
<div class="menu-item__block" title="内容块" style="display: none">
<i></i>
</div>
</div>
<div class="menu-divider"></div>
<div class="menu-item">
<div class="menu-item__search" data-menu="search">
<i></i>
</div>
<div class="menu-item__search__collapse" data-menu="search">
<div class="menu-item__search__collapse__search">
<input type="text" />
<label class="search-result"></label>
<div class="arrow-left">
<i></i>
</div>
<div class="arrow-right">
<i></i>
</div>
<span>×</span>
</div>
<div class="menu-item__search__collapse__replace">
<input type="text">
<button>替换</button>
</div>
</div>
<div class="menu-item__print" data-menu="print">
<i></i>
</div>
</div>
</div>
<div class="catalog" editor-component="catalog">
<div class="catalog__header">
<span>目录</span>
<div class="catalog__header__close">
<i></i>
</div>
</div>
<div class="catalog__main"></div>
</div>
<div class="editor"></div>
<div class="comment" editor-component="comment"></div>
<div class="footer" editor-component="footer">
<div>
<div class="catalog-mode" title="目录">
<i></i>
</div>
<div class="page-mode">
<i title="页面模式(分页、连页)"></i>
<div class="options">
<ul>
<li data-page-mode="paging" class="active">分页</li>
<li data-page-mode="continuity">连页</li>
</ul>
</div>
</div>
<span>可见页码:<span class="page-no-list">1</span></span>
<span>页面:<span class="page-no">1</span>/<span class="page-size">1</span></span>
<span>字数:<span class="word-count">0</span></span>
<span>行:<span class="row-no">0</span></span>
<span>列:<span class="col-no">0</span></span>
</div>
<div class="editor-mode" title="编辑模式(编辑、清洁、只读、表单、设计)">编辑模式</div>
<div>
<div class="page-scale-minus" title="缩小(Ctrl+-)">
<i></i>
</div>
<span class="page-scale-percentage" title="显示比例(点击可复原Ctrl+0)">100%</span>
<div class="page-scale-add" title="放大(Ctrl+=)">
<i></i>
</div>
<div class="paper-size">
<i title="纸张类型"></i>
<div class="options">
<ul>
<li data-paper-size="794*1123" class="active">A4</li>
<li data-paper-size="1593*2251">A2</li>
<li data-paper-size="1125*1593">A3</li>
<li data-paper-size="565*796">A5</li>
<li data-paper-size="412*488">5号信封</li>
<li data-paper-size="450*866">6号信封</li>
<li data-paper-size="609*862">7号信封</li>
<li data-paper-size="862*1221">9号信封</li>
<li data-paper-size="813*1266">法律用纸</li>
<li data-paper-size="813*1054">信纸</li>
</ul>
</div>
</div>
<div class="paper-direction">
<i title="纸张方向"></i>
<div class="options">
<ul>
<li data-paper-direction="vertical" class="active">纵向</li>
<li data-paper-direction="horizontal">横向</li>
</ul>
</div>
</div>
<div class="paper-margin" title="页边距">
<i></i>
</div>
<div class="fullscreen" title="全屏显示">
<i></i>
</div>
<div class="editor-option" title="编辑器设置">
<i></i>
</div>
</div>
</div>
</div>
</template>
<script setup>
import { ref, watch, defineExpose } from 'vue';
import Init from './canvas.ts';
import docxPlugin from "@hufe921/canvas-editor-plugin-docx";
const props = defineProps({
parentContent: {
type: Object,
default: () => null
}
});
const instance = ref(null);
watch(() => props.parentContent, (newVal) => {
if (newVal) {
instance.value = Init(JSON.parse(JSON.stringify(newVal)));
instance.value.instance.use(docxPlugin);
}
}, { deep: true });
const emit = defineEmits(['save-content']);
// Methods
const saveContent = () => {
let content = {
data: {}
};
content.data = instance.value.instance.command.getValue().data;
emit('save-content', content);
};
// 销毁编辑器实例
const destroyEditor = () => {
instance.value.instance.destroy();
};
defineExpose({
saveContent,
destroyEditor
})
</script>
<style>
@import url("./style.css");
@import url("./components/dialog/dialog.css");
</style>
新增parentTest.vue父级文件调用
<template>
<div class="parent-test">
<CanvasEditor
ref="canvasEditor"
:parentContent="parentContent"
:view="view"
:options="options"
@save-content="handleSaveCanvasEditorContent"
/>
<el-button class="btn-save" type="primary" size="large" @click="handleSaveContent">保 存</el-button>
</div>
</template>
<script setup lang="ts">
// @ts-ignore
import {ref, onMounted, reactive} from 'vue';
import CanvasEditor from './CanvasEditor/index.vue';
// 存放父组件传递的数据
const parentContent = ref<any>(undefined);
// 存放子组件数据
const content = ref<any>(undefined);
// 标识符
const view = ref<string | undefined>(undefined);
// 初始化数据
onMounted(() => {
console.log("模拟父组件向后端请求数据, 传递给子组件");
//getEditorContent();
view.value = "parent";
});
// 模拟后端获取数据的方法
const getEditorContent = () => {
parentContent.value = {
header: [],
main: [
{
value: "父类传递的数据 通过后端获取!\n",
size: 40,
bold: true,
},
{
value: "更好地方",
size: 24,
},
],
};
};
// 组件引用
const canvasEditor = ref<InstanceType<typeof CanvasEditor> | null>(null);
// 保存内容方法
const handleSaveContent = () => {
// 访问子组件的方法
(canvasEditor.value as any).saveContent();
};
// 处理子组件传递的数据
const handleSaveCanvasEditorContent = (data: any) => {
console.log("从子组件接收到的数据:", data);
};
const options=reactive({
mode: 'edit'
})
</script>
<style scoped>
.parent-test{
width: 100%;
}
.btn-save{
position: fixed;
right: 20px;
top: 10px;
z-index: 99;
}
</style>
修改main.ts名改为canvas.ts
import { commentList, data, options } from './mock'
import './style.css'
import prism from 'prismjs'
import Editor, {
BlockType,
Command,
ControlState,
ControlType,
EditorMode,
EditorZone,
ElementType,
IBlock,
ICatalogItem,
IElement,
KeyMap,
ListStyle,
ListType,
PageMode,
PaperDirection,
RowFlex,
TextDecorationStyle,
TitleLevel,
splitText
} from './editor'
import { Dialog } from './components/dialog/Dialog'
import { formatPrismToken } from './utils/prism'
import { Signature } from './components/signature/Signature'
import { debounce, nextTick, scrollIntoView } from './utils'
console.log(commentList, '批注')
console.log(data, '数据')
export default function Init (content:any) {
const isApple =
typeof navigator !== 'undefined' && /Mac OS X/.test(navigator.userAgent)
// 1. 初始化编辑器
const container = document.querySelector<HTMLDivElement>('.editor')!
const instance = new Editor(
container,
{
header: [
{
value: '第一人民医院',
size: 32,
rowFlex: RowFlex.CENTER
},
{
value: '\n门诊病历',
size: 18,
rowFlex: RowFlex.CENTER
},
{
value: '\n',
type: ElementType.SEPARATOR
}
],
main: <IElement[]>data,
footer: [
{
value: 'canvas-editor',
size: 12
}
]
},
options
)
console.log('实例: ', instance)
// cypress使用
Reflect.set(window, 'editor', instance)
// 菜单弹窗销毁
window.addEventListener(
'click',
evt => {
const visibleDom = document.querySelector('.visible')
if (!visibleDom || visibleDom.contains(<Node>evt.target)) return
visibleDom.classList.remove('visible')
},
{
capture: true
}
)
// 2. | 撤销 | 重做 | 格式刷 | 清除格式 |
const undoDom = document.querySelector<HTMLDivElement>('.menu-item__undo')!
undoDom.title = `撤销(${isApple ? '⌘' : 'Ctrl'}+Z)`
undoDom.onclick = function () {
console.log('undo')
instance.command.executeUndo()
}
const redoDom = document.querySelector<HTMLDivElement>('.menu-item__redo')!
redoDom.title = `重做(${isApple ? '⌘' : 'Ctrl'}+Y)`
redoDom.onclick = function () {
console.log('redo')
instance.command.executeRedo()
}
const painterDom = document.querySelector<HTMLDivElement>(
'.menu-item__painter'
)!
let isFirstClick = true
let painterTimeout: number
painterDom.onclick = function () {
if (isFirstClick) {
isFirstClick = false
painterTimeout = window.setTimeout(() => {
console.log('painter-click')
isFirstClick = true
instance.command.executePainter({
isDblclick: false
})
}, 200)
} else {
window.clearTimeout(painterTimeout)
}
}
painterDom.ondblclick = function () {
console.log('painter-dblclick')
isFirstClick = true
window.clearTimeout(painterTimeout)
instance.command.executePainter({
isDblclick: true
})
}
document.querySelector<HTMLDivElement>('.menu-item__format')!.onclick =
function () {
console.log('format')
instance.command.executeFormat()
}
// 3. | 字体 | 字体变大 | 字体变小 | 加粗 | 斜体 | 下划线 | 删除线 | 上标 | 下标 | 字体颜色 | 背景色 |
const fontDom = document.querySelector<HTMLDivElement>('.menu-item__font')!
const fontSelectDom = fontDom.querySelector<HTMLDivElement>('.select')!
const fontOptionDom = fontDom.querySelector<HTMLDivElement>('.options')!
fontDom.onclick = function () {
console.log('font')
fontOptionDom.classList.toggle('visible')
}
fontOptionDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
instance.command.executeFont(li.dataset.family!)
}
const sizeSetDom = document.querySelector<HTMLDivElement>('.menu-item__size')!
const sizeSelectDom = sizeSetDom.querySelector<HTMLDivElement>('.select')!
const sizeOptionDom = sizeSetDom.querySelector<HTMLDivElement>('.options')!
sizeSetDom.title = `设置字号`
sizeSetDom.onclick = function () {
console.log('size')
sizeOptionDom.classList.toggle('visible')
}
sizeOptionDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
instance.command.executeSize(Number(li.dataset.size!))
}
const sizeAddDom = document.querySelector<HTMLDivElement>(
'.menu-item__size-add'
)!
sizeAddDom.title = `增大字号(${isApple ? '⌘' : 'Ctrl'}+[)`
sizeAddDom.onclick = function () {
console.log('size-add')
instance.command.executeSizeAdd()
}
const sizeMinusDom = document.querySelector<HTMLDivElement>(
'.menu-item__size-minus'
)!
sizeMinusDom.title = `减小字号(${isApple ? '⌘' : 'Ctrl'}+])`
sizeMinusDom.onclick = function () {
console.log('size-minus')
instance.command.executeSizeMinus()
}
const boldDom = document.querySelector<HTMLDivElement>('.menu-item__bold')!
boldDom.title = `加粗(${isApple ? '⌘' : 'Ctrl'}+B)`
boldDom.onclick = function () {
console.log('bold')
instance.command.executeBold()
}
const italicDom =
document.querySelector<HTMLDivElement>('.menu-item__italic')!
italicDom.title = `斜体(${isApple ? '⌘' : 'Ctrl'}+I)`
italicDom.onclick = function () {
console.log('italic')
instance.command.executeItalic()
}
const underlineDom = document.querySelector<HTMLDivElement>(
'.menu-item__underline'
)!
underlineDom.title = `下划线(${isApple ? '⌘' : 'Ctrl'}+U)`
const underlineOptionDom =
underlineDom.querySelector<HTMLDivElement>('.options')!
underlineDom.querySelector<HTMLSpanElement>('.select')!.onclick =
function () {
underlineOptionDom.classList.toggle('visible')
}
underlineDom.querySelector<HTMLElement>('i')!.onclick = function () {
console.log('underline')
instance.command.executeUnderline()
underlineOptionDom.classList.remove('visible')
}
underlineDom.querySelector<HTMLUListElement>('ul')!.onmousedown = function (
evt
) {
const li = evt.target as HTMLLIElement
const decorationStyle = <TextDecorationStyle>li.dataset.decorationStyle
instance.command.executeUnderline({
style: decorationStyle
})
underlineOptionDom.classList.remove('visible')
}
const strikeoutDom = document.querySelector<HTMLDivElement>(
'.menu-item__strikeout'
)!
strikeoutDom.onclick = function () {
console.log('strikeout')
instance.command.executeStrikeout()
}
const superscriptDom = document.querySelector<HTMLDivElement>(
'.menu-item__superscript'
)!
superscriptDom.title = `上标(${isApple ? '⌘' : 'Ctrl'}+Shift+,)`
superscriptDom.onclick = function () {
console.log('superscript')
instance.command.executeSuperscript()
}
const subscriptDom = document.querySelector<HTMLDivElement>(
'.menu-item__subscript'
)!
subscriptDom.title = `下标(${isApple ? '⌘' : 'Ctrl'}+Shift+.)`
subscriptDom.onclick = function () {
console.log('subscript')
instance.command.executeSubscript()
}
const colorControlDom = document.querySelector<HTMLInputElement>('#color')!
colorControlDom.oninput = function () {
instance.command.executeColor(colorControlDom.value)
}
const colorDom = document.querySelector<HTMLDivElement>('.menu-item__color')!
const colorSpanDom = colorDom.querySelector('span')!
colorDom.onclick = function () {
console.log('color')
colorControlDom.click()
}
const highlightControlDom =
document.querySelector<HTMLInputElement>('#highlight')!
highlightControlDom.oninput = function () {
instance.command.executeHighlight(highlightControlDom.value)
}
const highlightDom = document.querySelector<HTMLDivElement>(
'.menu-item__highlight'
)!
const highlightSpanDom = highlightDom.querySelector('span')!
highlightDom.onclick = function () {
console.log('highlight')
highlightControlDom?.click()
}
const titleDom = document.querySelector<HTMLDivElement>('.menu-item__title')!
const titleSelectDom = titleDom.querySelector<HTMLDivElement>('.select')!
const titleOptionDom = titleDom.querySelector<HTMLDivElement>('.options')!
titleOptionDom.querySelectorAll('li').forEach((li, index) => {
li.title = `Ctrl+${isApple ? 'Option' : 'Alt'}+${index}`
})
titleDom.onclick = function () {
console.log('title')
titleOptionDom.classList.toggle('visible')
}
titleOptionDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
const level = <TitleLevel>li.dataset.level
instance.command.executeTitle(level || null)
}
const leftDom = document.querySelector<HTMLDivElement>('.menu-item__left')!
leftDom.title = `左对齐(${isApple ? '⌘' : 'Ctrl'}+L)`
leftDom.onclick = function () {
console.log('left')
instance.command.executeRowFlex(RowFlex.LEFT)
}
const centerDom =
document.querySelector<HTMLDivElement>('.menu-item__center')!
centerDom.title = `居中对齐(${isApple ? '⌘' : 'Ctrl'}+E)`
centerDom.onclick = function () {
console.log('center')
instance.command.executeRowFlex(RowFlex.CENTER)
}
const rightDom = document.querySelector<HTMLDivElement>('.menu-item__right')!
rightDom.title = `右对齐(${isApple ? '⌘' : 'Ctrl'}+R)`
rightDom.onclick = function () {
console.log('right')
instance.command.executeRowFlex(RowFlex.RIGHT)
}
const alignmentDom = document.querySelector<HTMLDivElement>(
'.menu-item__alignment'
)!
alignmentDom.title = `两端对齐(${isApple ? '⌘' : 'Ctrl'}+J)`
alignmentDom.onclick = function () {
console.log('alignment')
instance.command.executeRowFlex(RowFlex.ALIGNMENT)
}
const justifyDom = document.querySelector<HTMLDivElement>(
'.menu-item__justify'
)!
justifyDom.title = `分散对齐(${isApple ? '⌘' : 'Ctrl'}+Shift+J)`
justifyDom.onclick = function () {
console.log('justify')
instance.command.executeRowFlex(RowFlex.JUSTIFY)
}
const rowMarginDom = document.querySelector<HTMLDivElement>(
'.menu-item__row-margin'
)!
const rowOptionDom = rowMarginDom.querySelector<HTMLDivElement>('.options')!
rowMarginDom.onclick = function () {
console.log('row-margin')
rowOptionDom.classList.toggle('visible')
}
rowOptionDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
instance.command.executeRowMargin(Number(li.dataset.rowmargin!))
}
const listDom = document.querySelector<HTMLDivElement>('.menu-item__list')!
listDom.title = `列表(${isApple ? '⌘' : 'Ctrl'}+Shift+U)`
const listOptionDom = listDom.querySelector<HTMLDivElement>('.options')!
listDom.onclick = function () {
console.log('list')
listOptionDom.classList.toggle('visible')
}
listOptionDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
const listType = <ListType>li.dataset.listType || null
const listStyle = <ListStyle>(<unknown>li.dataset.listStyle)
instance.command.executeList(listType, listStyle)
}
// 4. | 表格 | 图片 | 超链接 | 分割线 | 水印 | 代码块 | 分隔符 | 控件 | 复选框 | LaTeX | 日期选择器
const tableDom = document.querySelector<HTMLDivElement>('.menu-item__table')!
const tablePanelContainer = document.querySelector<HTMLDivElement>(
'.menu-item__table__collapse'
)!
const tableClose = document.querySelector<HTMLDivElement>('.table-close')!
const tableTitle = document.querySelector<HTMLDivElement>('.table-select')!
const tablePanel = document.querySelector<HTMLDivElement>('.table-panel')!
// 绘制行列
const tableCellList: HTMLDivElement[][] = []
for (let i = 0; i < 10; i++) {
const tr = document.createElement('tr')
tr.classList.add('table-row')
const trCellList: HTMLDivElement[] = []
for (let j = 0; j < 10; j++) {
const td = document.createElement('td')
td.classList.add('table-cel')
tr.append(td)
trCellList.push(td)
}
tablePanel.append(tr)
tableCellList.push(trCellList)
}
let colIndex = 0
let rowIndex = 0
// 移除所有格选择
function removeAllTableCellSelect() {
tableCellList.forEach(tr => {
tr.forEach(td => td.classList.remove('active'))
})
}
// 设置标题内容
function setTableTitle(payload: string) {
tableTitle.innerText = payload
}
// 恢复初始状态
function recoveryTable() {
// 还原选择样式、标题、选择行列
removeAllTableCellSelect()
setTableTitle('插入')
colIndex = 0
rowIndex = 0
// 隐藏panel
tablePanelContainer.style.display = 'none'
}
tableDom.onclick = function () {
console.log('table')
tablePanelContainer!.style.display = 'block'
}
tablePanel.onmousemove = function (evt) {
const celSize = 16
const rowMarginTop = 10
const celMarginRight = 6
const { offsetX, offsetY } = evt
// 移除所有选择
removeAllTableCellSelect()
colIndex = Math.ceil(offsetX / (celSize + celMarginRight)) || 1
rowIndex = Math.ceil(offsetY / (celSize + rowMarginTop)) || 1
// 改变选择样式
tableCellList.forEach((tr, trIndex) => {
tr.forEach((td, tdIndex) => {
if (tdIndex < colIndex && trIndex < rowIndex) {
td.classList.add('active')
}
})
})
// 改变表格标题
setTableTitle(`${rowIndex}×${colIndex}`)
}
tableClose.onclick = function () {
recoveryTable()
}
tablePanel.onclick = function () {
// 应用选择
instance.command.executeInsertTable(rowIndex, colIndex)
recoveryTable()
}
const imageDom = document.querySelector<HTMLDivElement>('.menu-item__image')!
const imageFileDom = document.querySelector<HTMLInputElement>('#image')!
imageDom.onclick = function () {
imageFileDom.click()
}
imageFileDom.onchange = function () {
const file = imageFileDom.files![0]!
const fileReader = new FileReader()
fileReader.readAsDataURL(file)
fileReader.onload = function () {
// 计算宽高
const image = new Image()
const value = fileReader.result as string
image.src = value
image.onload = function () {
instance.command.executeImage({
value,
width: image.width,
height: image.height
})
imageFileDom.value = ''
}
}
}
const hyperlinkDom = document.querySelector<HTMLDivElement>(
'.menu-item__hyperlink'
)!
hyperlinkDom.onclick = function () {
console.log('hyperlink')
new Dialog({
title: '超链接',
data: [
{
type: 'text',
label: '文本',
name: 'name',
required: true,
placeholder: '请输入文本',
value: instance.command.getRangeText()
},
{
type: 'text',
label: '链接',
name: 'url',
required: true,
placeholder: '请输入链接'
}
],
onConfirm: payload => {
const name = payload.find(p => p.name === 'name')?.value
if (!name) return
const url = payload.find(p => p.name === 'url')?.value
if (!url) return
instance.command.executeHyperlink({
type: ElementType.HYPERLINK,
value: '',
url,
valueList: splitText(name).map(n => ({
value: n,
size: 16
}))
})
}
})
}
const separatorDom = document.querySelector<HTMLDivElement>(
'.menu-item__separator'
)!
const separatorOptionDom =
separatorDom.querySelector<HTMLDivElement>('.options')!
separatorDom.onclick = function () {
console.log('separator')
separatorOptionDom.classList.toggle('visible')
}
separatorOptionDom.onmousedown = function (evt) {
let payload: number[] = []
const li = evt.target as HTMLLIElement
const separatorDash = li.dataset.separator?.split(',').map(Number)
if (separatorDash) {
const isSingleLine = separatorDash.every(d => d === 0)
if (!isSingleLine) {
payload = separatorDash
}
}
instance.command.executeSeparator(payload)
}
const pageBreakDom = document.querySelector<HTMLDivElement>(
'.menu-item__page-break'
)!
pageBreakDom.onclick = function () {
console.log('pageBreak')
instance.command.executePageBreak()
}
const watermarkDom = document.querySelector<HTMLDivElement>(
'.menu-item__watermark'
)!
const watermarkOptionDom =
watermarkDom.querySelector<HTMLDivElement>('.options')!
watermarkDom.onclick = function () {
console.log('watermark')
watermarkOptionDom.classList.toggle('visible')
}
watermarkOptionDom.onmousedown = function (evt) {
const li = evt.target as HTMLLIElement
const menu = li.dataset.menu!
watermarkOptionDom.classList.toggle('visible')
if (menu === 'add') {
new Dialog({
title: '水印',
data: [
{
type: 'text',
label: '内容',
name: 'data',
required: true,
placeholder: '请输入内容'
},
{
type: 'color',
label: '颜色',
name: 'color',
required: true,
value: '#AEB5C0'
},
{
type: 'number',
label: '字体大小',
name: 'size',
required: true,
value: '120'
},
{
type: 'number',
label: '透明度',
name: 'opacity',
required: true,
value: '0.3'
},
{
type: 'select',
label: '重复',
name: 'repeat',
value: '0',
required: false,
options: [
{
label: '不重复',
value: '0'
},
{
label: '重复',
value: '1'
}
]
},
{
type: 'number',
label: '水平间隔',
name: 'horizontalGap',
required: false,
value: '10'
},
{
type: 'number',
label: '垂直间隔',
name: 'verticalGap',
required: false,
value: '10'
}
],
onConfirm: payload => {
const nullableIndex = payload.findIndex(p => !p.value)
if (~nullableIndex) return
const watermark = payload.reduce((pre, cur) => {
pre[cur.name] = cur.value
return pre
}, <any>{})
const repeat = watermark.repeat === '1'
instance.command.executeAddWatermark({
data: watermark.data,
color: watermark.color,
size: Number(watermark.size),
opacity: Number(watermark.opacity),
repeat,
gap:
repeat && watermark.horizontalGap && watermark.verticalGap
? [
Number(watermark.horizontalGap),
Number(watermark.verticalGap)
]
: undefined
})
}
})
} else {
instance.command.executeDeleteWatermark()
}
}
const codeblockDom = document.querySelector<HTMLDivElement>(
'.menu-item__codeblock'
)!
codeblockDom.onclick = function () {
console.log('codeblock')
new Dialog({
title: '代码块',
data: [
{
type: 'textarea',
name: 'codeblock',
placeholder: '请输入代码',
width: 500,
height: 300
}
],
onConfirm: payload => {
const codeblock = payload.find(p => p.name === 'codeblock')?.value
if (!codeblock) return
const tokenList = prism.tokenize(codeblock, prism.languages.javascript)
const formatTokenList = formatPrismToken(tokenList)
const elementList: IElement[] = []
for (let i = 0; i < formatTokenList.length; i++) {
const formatToken = formatTokenList[i]
const tokenStringList = splitText(formatToken.content)
for (let j = 0; j < tokenStringList.length; j++) {
const value = tokenStringList[j]
const element: IElement = {
value
}
if (formatToken.color) {
element.color = formatToken.color
}
if (formatToken.bold) {
element.bold = true
}
if (formatToken.italic) {
element.italic = true
}
elementList.push(element)
}
}
elementList.unshift({
value: '\n'
})
instance.command.executeInsertElementList(elementList)
}
})
}
const controlDom = document.querySelector<HTMLDivElement>(
'.menu-item__control'
)!
const controlOptionDom = controlDom.querySelector<HTMLDivElement>('.options')!
controlDom.onclick = function () {
console.log('control')
controlOptionDom.classList.toggle('visible')
}
controlOptionDom.onmousedown = function (evt) {
controlOptionDom.classList.toggle('visible')
const li = evt.target as HTMLLIElement
const type = <ControlType>li.dataset.control
switch (type) {
case ControlType.TEXT:
new Dialog({
title: '文本控件',
data: [
{
type: 'text',
label: '占位符',
name: 'placeholder',
required: true,
placeholder: '请输入占位符'
},
{
type: 'text',
label: '默认值',
name: 'value',
placeholder: '请输入默认值'
}
],
onConfirm: payload => {
const placeholder = payload.find(
p => p.name === 'placeholder'
)?.value
if (!placeholder) return
const value = payload.find(p => p.name === 'value')?.value || ''
instance.command.executeInsertControl({
type: ElementType.CONTROL,
value: '',
control: {
type,
value: value
? [
{
value
}
]
: null,
placeholder
}
})
}
})
break
case ControlType.SELECT:
new Dialog({
title: '列举控件',
data: [
{
type: 'text',
label: '占位符',
name: 'placeholder',
required: true,
placeholder: '请输入占位符'
},
{
type: 'text',
label: '默认值',
name: 'code',
placeholder: '请输入默认值'
},
{
type: 'textarea',
label: '值集',
name: 'valueSets',
required: true,
height: 100,
placeholder: `请输入值集JSON,例:\n[{\n"value":"有",\n"code":"98175"\n}]`
}
],
onConfirm: payload => {
const placeholder = payload.find(
p => p.name === 'placeholder'
)?.value
if (!placeholder) return
const valueSets = payload.find(p => p.name === 'valueSets')?.value
if (!valueSets) return
const code = payload.find(p => p.name === 'code')?.value
instance.command.executeInsertControl({
type: ElementType.CONTROL,
value: '',
control: {
type,
code,
value: null,
placeholder,
valueSets: JSON.parse(valueSets)
}
})
}
})
break
case ControlType.CHECKBOX:
new Dialog({
title: '复选框控件',
data: [
{
type: 'text',
label: '默认值',
name: 'code',
placeholder: '请输入默认值,多个值以英文逗号分割'
},
{
type: 'textarea',
label: '值集',
name: 'valueSets',
required: true,
height: 100,
placeholder: `请输入值集JSON,例:\n[{\n"value":"有",\n"code":"98175"\n}]`
}
],
onConfirm: payload => {
const valueSets = payload.find(p => p.name === 'valueSets')?.value
if (!valueSets) return
const code = payload.find(p => p.name === 'code')?.value
instance.command.executeInsertControl({
type: ElementType.CONTROL,
value: '',
control: {
type,
code,
value: null,
valueSets: JSON.parse(valueSets)
}
})
}
})
break
case ControlType.RADIO:
new Dialog({
title: '单选框控件',
data: [
{
type: 'text',
label: '默认值',
name: 'code',
placeholder: '请输入默认值'
},
{
type: 'textarea',
label: '值集',
name: 'valueSets',
required: true,
height: 100,
placeholder: `请输入值集JSON,例:\n[{\n"value":"有",\n"code":"98175"\n}]`
}
],
onConfirm: payload => {
const valueSets = payload.find(p => p.name === 'valueSets')?.value
if (!valueSets) return
const code = payload.find(p => p.name === 'code')?.value
instance.command.executeInsertControl({
type: ElementType.CONTROL,
value: '',
control: {
type,
code,
value: null,
valueSets: JSON.parse(valueSets)
}
})
}
})
break
case ControlType.DATE:
new Dialog({
title: '日期控件',
data: [
{
type: 'text',
label: '占位符',
name: 'placeholder',
required: true,
placeholder: '请输入占位符'
},
{
type: 'text',
label: '默认值',
name: 'value',
placeholder: '请输入默认值'
},
{
type: 'select',
label: '日期格式',
name: 'dateFormat',
value: 'yyyy-MM-dd hh:mm:ss',
required: true,
options: [
{
label: 'yyyy-MM-dd hh:mm:ss',
value: 'yyyy-MM-dd hh:mm:ss'
},
{
label: 'yyyy-MM-dd',
value: 'yyyy-MM-dd'
}
]
}
],
onConfirm: payload => {
const placeholder = payload.find(
p => p.name === 'placeholder'
)?.value
if (!placeholder) return
const value = payload.find(p => p.name === 'value')?.value || ''
const dateFormat =
payload.find(p => p.name === 'dateFormat')?.value || ''
instance.command.executeInsertControl({
type: ElementType.CONTROL,
value: '',
control: {
type,
dateFormat,
value: value
? [
{
value
}
]
: null,
placeholder
}
})
}
})
break
case ControlType.NUMBER:
new Dialog({
title: '数值控件',
data: [
{
type: 'text',
label: '占位符',
name: 'placeholder',
required: true,
placeholder: '请输入占位符'
},
{
type: 'text',
label: '默认值',
name: 'value',
placeholder: '请输入默认值'
}
],
onConfirm: payload => {
const placeholder = payload.find(
p => p.name === 'placeholder'
)?.value
if (!placeholder) return
const value = payload.find(p => p.name === 'value')?.value || ''
instance.command.executeInsertControl({
type: ElementType.CONTROL,
value: '',
control: {
type,
value: value
? [
{
value
}
]
: null,
placeholder
}
})
}
})
break
default:
break
}
}
const checkboxDom = document.querySelector<HTMLDivElement>(
'.menu-item__checkbox'
)!
checkboxDom.onclick = function () {
console.log('checkbox')
instance.command.executeInsertElementList([
{
type: ElementType.CHECKBOX,
checkbox: {
value: false
},
value: ''
}
])
}
const radioDom = document.querySelector<HTMLDivElement>('.menu-item__radio')!
radioDom.onclick = function () {
console.log('radio')
instance.command.executeInsertElementList([
{
type: ElementType.RADIO,
checkbox: {
value: false
},
value: ''
}
])
}
const latexDom = document.querySelector<HTMLDivElement>('.menu-item__latex')!
latexDom.onclick = function () {
console.log('LaTeX')
new Dialog({
title: 'LaTeX',
data: [
{
type: 'textarea',
height: 100,
name: 'value',
placeholder: '请输入LaTeX文本'
}
],
onConfirm: payload => {
const value = payload.find(p => p.name === 'value')?.value
if (!value) return
instance.command.executeInsertElementList([
{
type: ElementType.LATEX,
value
}
])
}
})
}
const dateDom = document.querySelector<HTMLDivElement>('.menu-item__date')!
const dateDomOptionDom = dateDom.querySelector<HTMLDivElement>('.options')!
dateDom.onclick = function () {
console.log('date')
dateDomOptionDom.classList.toggle('visible')
// 定位调整
const bodyRect = document.body.getBoundingClientRect()
const dateDomOptionRect = dateDomOptionDom.getBoundingClientRect()
if (dateDomOptionRect.left + dateDomOptionRect.width > bodyRect.width) {
dateDomOptionDom.style.right = '0px'
dateDomOptionDom.style.left = 'unset'
} else {
dateDomOptionDom.style.right = 'unset'
dateDomOptionDom.style.left = '0px'
}
// 当前日期
const date = new Date()
const year = date.getFullYear().toString()
const month = (date.getMonth() + 1).toString().padStart(2, '0')
const day = date.getDate().toString().padStart(2, '0')
const hour = date.getHours().toString().padStart(2, '0')
const minute = date.getMinutes().toString().padStart(2, '0')
const second = date.getSeconds().toString().padStart(2, '0')
const dateString = `${year}-${month}-${day}`
const dateTimeString = `${dateString} ${hour}:${minute}:${second}`
dateDomOptionDom.querySelector<HTMLLIElement>('li:first-child')!.innerText =
dateString
dateDomOptionDom.querySelector<HTMLLIElement>('li:last-child')!.innerText =
dateTimeString
}
dateDomOptionDom.onmousedown = function (evt) {
const li = evt.target as HTMLLIElement
const dateFormat = li.dataset.format!
dateDomOptionDom.classList.toggle('visible')
instance.command.executeInsertElementList([
{
type: ElementType.DATE,
value: '',
dateFormat,
valueList: [
{
value: li.innerText.trim()
}
]
}
])
}
const blockDom = document.querySelector<HTMLDivElement>('.menu-item__block')!
blockDom.onclick = function () {
console.log('block')
new Dialog({
title: '内容块',
data: [
{
type: 'select',
label: '类型',
name: 'type',
value: 'iframe',
required: true,
options: [
{
label: '网址',
value: 'iframe'
},
{
label: '视频',
value: 'video'
}
]
},
{
type: 'number',
label: '宽度',
name: 'width',
placeholder: '请输入宽度(默认页面内宽度)'
},
{
type: 'number',
label: '高度',
name: 'height',
required: true,
placeholder: '请输入高度'
},
{
type: 'input',
label: '地址',
name: 'src',
required: false,
placeholder: '请输入地址'
},
{
type: 'textarea',
label: 'HTML',
height: 100,
name: 'srcdoc',
required: false,
placeholder: '请输入HTML代码(仅网址类型有效)'
}
],
onConfirm: payload => {
const type = payload.find(p => p.name === 'type')?.value
if (!type) return
const width = payload.find(p => p.name === 'width')?.value
const height = payload.find(p => p.name === 'height')?.value
if (!height) return
// 地址或HTML代码至少存在一项
const src = payload.find(p => p.name === 'src')?.value
const srcdoc = payload.find(p => p.name === 'srcdoc')?.value
const block: IBlock = {
type: <BlockType>type
}
if (block.type === BlockType.IFRAME) {
if (!src && !srcdoc) return
block.iframeBlock = {
src,
srcdoc
}
} else if (block.type === BlockType.VIDEO) {
if (!src) return
block.videoBlock = {
src
}
}
const blockElement: IElement = {
type: ElementType.BLOCK,
value: '',
height: Number(height),
block
}
if (width) {
blockElement.width = Number(width)
}
instance.command.executeInsertElementList([blockElement])
}
})
}
// 5. | 搜索&替换 | 打印 |
const searchCollapseDom = document.querySelector<HTMLDivElement>(
'.menu-item__search__collapse'
)!
const searchInputDom = document.querySelector<HTMLInputElement>(
'.menu-item__search__collapse__search input'
)!
const replaceInputDom = document.querySelector<HTMLInputElement>(
'.menu-item__search__collapse__replace input'
)!
const searchDom =
document.querySelector<HTMLDivElement>('.menu-item__search')!
searchDom.title = `搜索与替换(${isApple ? '⌘' : 'Ctrl'}+F)`
const searchResultDom =
searchCollapseDom.querySelector<HTMLLabelElement>('.search-result')!
function setSearchResult() {
const result = instance.command.getSearchNavigateInfo()
if (result) {
const { index, count } = result
searchResultDom.innerText = `${index}/${count}`
} else {
searchResultDom.innerText = ''
}
}
searchDom.onclick = function () {
console.log('search')
searchCollapseDom.style.display = 'block'
const bodyRect = document.body.getBoundingClientRect()
const searchRect = searchDom.getBoundingClientRect()
const searchCollapseRect = searchCollapseDom.getBoundingClientRect()
if (searchRect.left + searchCollapseRect.width > bodyRect.width) {
searchCollapseDom.style.right = '0px'
searchCollapseDom.style.left = 'unset'
} else {
searchCollapseDom.style.right = 'unset'
}
searchInputDom.focus()
}
searchCollapseDom.querySelector<HTMLSpanElement>('span')!.onclick =
function () {
searchCollapseDom.style.display = 'none'
searchInputDom.value = ''
replaceInputDom.value = ''
instance.command.executeSearch(null)
setSearchResult()
}
searchInputDom.oninput = function () {
instance.command.executeSearch(searchInputDom.value || null)
setSearchResult()
}
searchInputDom.onkeydown = function (evt) {
if (evt.key === 'Enter') {
instance.command.executeSearch(searchInputDom.value || null)
setSearchResult()
}
}
searchCollapseDom.querySelector<HTMLButtonElement>('button')!.onclick =
function () {
const searchValue = searchInputDom.value
const replaceValue = replaceInputDom.value
if (searchValue && searchValue !== replaceValue) {
instance.command.executeReplace(replaceValue)
}
}
searchCollapseDom.querySelector<HTMLDivElement>('.arrow-left')!.onclick =
function () {
instance.command.executeSearchNavigatePre()
setSearchResult()
}
searchCollapseDom.querySelector<HTMLDivElement>('.arrow-right')!.onclick =
function () {
instance.command.executeSearchNavigateNext()
setSearchResult()
}
const printDom = document.querySelector<HTMLDivElement>('.menu-item__print')!
printDom.title = `打印(${isApple ? '⌘' : 'Ctrl'}+P)`
printDom.onclick = function () {
console.log('print')
instance.command.executePrint()
}
// 6. 目录显隐 | 页面模式 | 纸张缩放 | 纸张大小 | 纸张方向 | 页边距 | 全屏 | 设置
const editorOptionDom =
document.querySelector<HTMLDivElement>('.editor-option')!
editorOptionDom.onclick = function () {
const options = instance.command.getOptions()
new Dialog({
title: '编辑器配置',
data: [
{
type: 'textarea',
name: 'option',
width: 350,
height: 300,
required: true,
value: JSON.stringify(options, null, 2),
placeholder: '请输入编辑器配置'
}
],
onConfirm: payload => {
const newOptionValue = payload.find(p => p.name === 'option')?.value
if (!newOptionValue) return
const newOption = JSON.parse(newOptionValue)
instance.command.executeUpdateOptions(newOption)
}
})
}
async function updateCatalog() {
const catalog = await instance.command.getCatalog()
const catalogMainDom =
document.querySelector<HTMLDivElement>('.catalog__main')!
catalogMainDom.innerHTML = ''
if (catalog) {
const appendCatalog = (
parent: HTMLDivElement,
catalogItems: ICatalogItem[]
) => {
for (let c = 0; c < catalogItems.length; c++) {
const catalogItem = catalogItems[c]
const catalogItemDom = document.createElement('div')
catalogItemDom.classList.add('catalog-item')
// 渲染
const catalogItemContentDom = document.createElement('div')
catalogItemContentDom.classList.add('catalog-item__content')
const catalogItemContentSpanDom = document.createElement('span')
catalogItemContentSpanDom.innerText = catalogItem.name
catalogItemContentDom.append(catalogItemContentSpanDom)
// 定位
catalogItemContentDom.onclick = () => {
instance.command.executeLocationCatalog(catalogItem.id)
}
catalogItemDom.append(catalogItemContentDom)
if (catalogItem.subCatalog && catalogItem.subCatalog.length) {
appendCatalog(catalogItemDom, catalogItem.subCatalog)
}
// 追加
parent.append(catalogItemDom)
}
}
appendCatalog(catalogMainDom, catalog)
}
}
let isCatalogShow = true
const catalogDom = document.querySelector<HTMLElement>('.catalog')!
const catalogModeDom =
document.querySelector<HTMLDivElement>('.catalog-mode')!
const catalogHeaderCloseDom = document.querySelector<HTMLDivElement>(
'.catalog__header__close'
)!
const switchCatalog = () => {
isCatalogShow = !isCatalogShow
if (isCatalogShow) {
catalogDom.style.display = 'block'
updateCatalog()
} else {
catalogDom.style.display = 'none'
}
}
catalogModeDom.onclick = switchCatalog
catalogHeaderCloseDom.onclick = switchCatalog
const pageModeDom = document.querySelector<HTMLDivElement>('.page-mode')!
const pageModeOptionsDom =
pageModeDom.querySelector<HTMLDivElement>('.options')!
pageModeDom.onclick = function () {
pageModeOptionsDom.classList.toggle('visible')
}
pageModeOptionsDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
instance.command.executePageMode(<PageMode>li.dataset.pageMode!)
}
document.querySelector<HTMLDivElement>('.page-scale-percentage')!.onclick =
function () {
console.log('page-scale-recovery')
instance.command.executePageScaleRecovery()
}
document.querySelector<HTMLDivElement>('.page-scale-minus')!.onclick =
function () {
console.log('page-scale-minus')
instance.command.executePageScaleMinus()
}
document.querySelector<HTMLDivElement>('.page-scale-add')!.onclick =
function () {
console.log('page-scale-add')
instance.command.executePageScaleAdd()
}
// 纸张大小
const paperSizeDom = document.querySelector<HTMLDivElement>('.paper-size')!
const paperSizeDomOptionsDom =
paperSizeDom.querySelector<HTMLDivElement>('.options')!
paperSizeDom.onclick = function () {
paperSizeDomOptionsDom.classList.toggle('visible')
}
paperSizeDomOptionsDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
const paperType = li.dataset.paperSize!
const [width, height] = paperType.split('*').map(Number)
instance.command.executePaperSize(width, height)
// 纸张状态回显
paperSizeDomOptionsDom
.querySelectorAll('li')
.forEach(child => child.classList.remove('active'))
li.classList.add('active')
}
// 纸张方向
const paperDirectionDom =
document.querySelector<HTMLDivElement>('.paper-direction')!
const paperDirectionDomOptionsDom =
paperDirectionDom.querySelector<HTMLDivElement>('.options')!
paperDirectionDom.onclick = function () {
paperDirectionDomOptionsDom.classList.toggle('visible')
}
paperDirectionDomOptionsDom.onclick = function (evt) {
const li = evt.target as HTMLLIElement
const paperDirection = li.dataset.paperDirection!
instance.command.executePaperDirection(<PaperDirection>paperDirection)
// 纸张方向状态回显
paperDirectionDomOptionsDom
.querySelectorAll('li')
.forEach(child => child.classList.remove('active'))
li.classList.add('active')
}
// 页面边距
const paperMarginDom =
document.querySelector<HTMLDivElement>('.paper-margin')!
paperMarginDom.onclick = function () {
const [topMargin, rightMargin, bottomMargin, leftMargin] =
instance.command.getPaperMargin()
new Dialog({
title: '页边距',
data: [
{
type: 'text',
label: '上边距',
name: 'top',
required: true,
value: `${topMargin}`,
placeholder: '请输入上边距'
},
{
type: 'text',
label: '下边距',
name: 'bottom',
required: true,
value: `${bottomMargin}`,
placeholder: '请输入下边距'
},
{
type: 'text',
label: '左边距',
name: 'left',
required: true,
value: `${leftMargin}`,
placeholder: '请输入左边距'
},
{
type: 'text',
label: '右边距',
name: 'right',
required: true,
value: `${rightMargin}`,
placeholder: '请输入右边距'
}
],
onConfirm: payload => {
const top = payload.find(p => p.name === 'top')?.value
if (!top) return
const bottom = payload.find(p => p.name === 'bottom')?.value
if (!bottom) return
const left = payload.find(p => p.name === 'left')?.value
if (!left) return
const right = payload.find(p => p.name === 'right')?.value
if (!right) return
instance.command.executeSetPaperMargin([
Number(top),
Number(right),
Number(bottom),
Number(left)
])
}
})
}
// 全屏
const fullscreenDom = document.querySelector<HTMLDivElement>('.fullscreen')!
fullscreenDom.onclick = toggleFullscreen
window.addEventListener('keydown', evt => {
if (evt.key === 'F11') {
toggleFullscreen()
evt.preventDefault()
}
})
document.addEventListener('fullscreenchange', () => {
fullscreenDom.classList.toggle('exist')
})
function toggleFullscreen() {
console.log('fullscreen')
if (!document.fullscreenElement) {
document.documentElement.requestFullscreen()
} else {
document.exitFullscreen()
}
}
// 7. 编辑器使用模式
let modeIndex = 0
const modeList = [
{
mode: EditorMode.EDIT,
name: '编辑模式'
},
{
mode: EditorMode.CLEAN,
name: '清洁模式'
},
{
mode: EditorMode.READONLY,
name: '只读模式'
},
{
mode: EditorMode.FORM,
name: '表单模式'
},
{
mode: EditorMode.PRINT,
name: '打印模式'
},
{
mode: EditorMode.DESIGN,
name: '设计模式'
}
]
const modeElement = document.querySelector<HTMLDivElement>('.editor-mode')!
modeElement.onclick = function () {
// 模式选择循环
modeIndex === modeList.length - 1 ? (modeIndex = 0) : modeIndex++
// 设置模式
const { name, mode } = modeList[modeIndex]
modeElement.innerText = name
instance.command.executeMode(mode)
// 设置菜单栏权限视觉反馈
const isReadonly = mode === EditorMode.READONLY
const enableMenuList = ['search', 'print']
document.querySelectorAll<HTMLDivElement>('.menu-item>div').forEach(dom => {
const menu = dom.dataset.menu
isReadonly && (!menu || !enableMenuList.includes(menu))
? dom.classList.add('disable')
: dom.classList.remove('disable')
})
}
// 模拟批注
const commentDom = document.querySelector<HTMLDivElement>('.comment')!
async function updateComment() {
const groupIds = await instance.command.getGroupIds()
for (const comment of commentList) {
const activeCommentDom = commentDom.querySelector<HTMLDivElement>(
`.comment-item[data-id='${comment.id}']`
)
// 编辑器是否存在对应成组id
if (groupIds.includes(comment.id)) {
// 当前dom是否存在-不存在则追加
if (!activeCommentDom) {
const commentItem = document.createElement('div')
commentItem.classList.add('comment-item')
commentItem.setAttribute('data-id', comment.id)
commentItem.onclick = () => {
instance.command.executeLocationGroup(comment.id)
}
commentDom.append(commentItem)
// 选区信息
const commentItemTitle = document.createElement('div')
commentItemTitle.classList.add('comment-item__title')
commentItemTitle.append(document.createElement('span'))
const commentItemTitleContent = document.createElement('span')
commentItemTitleContent.innerText = comment.rangeText
commentItemTitle.append(commentItemTitleContent)
const closeDom = document.createElement('i')
closeDom.onclick = () => {
instance.command.executeDeleteGroup(comment.id)
}
commentItemTitle.append(closeDom)
commentItem.append(commentItemTitle)
// 基础信息
const commentItemInfo = document.createElement('div')
commentItemInfo.classList.add('comment-item__info')
const commentItemInfoName = document.createElement('span')
commentItemInfoName.innerText = comment.userName
const commentItemInfoDate = document.createElement('span')
commentItemInfoDate.innerText = comment.createdDate
commentItemInfo.append(commentItemInfoName)
commentItemInfo.append(commentItemInfoDate)
commentItem.append(commentItemInfo)
// 详细评论
const commentItemContent = document.createElement('div')
commentItemContent.classList.add('comment-item__content')
commentItemContent.innerText = comment.content
commentItem.append(commentItemContent)
commentDom.append(commentItem)
}
} else {
// 编辑器内不存在对应成组id则dom则移除
activeCommentDom?.remove()
}
}
}
// 8. 内部事件监听
instance.listener.rangeStyleChange = function (payload) {
// 控件类型
payload.type === ElementType.SUBSCRIPT
? subscriptDom.classList.add('active')
: subscriptDom.classList.remove('active')
payload.type === ElementType.SUPERSCRIPT
? superscriptDom.classList.add('active')
: superscriptDom.classList.remove('active')
payload.type === ElementType.SEPARATOR
? separatorDom.classList.add('active')
: separatorDom.classList.remove('active')
separatorOptionDom
.querySelectorAll('li')
.forEach(li => li.classList.remove('active'))
if (payload.type === ElementType.SEPARATOR) {
const separator = payload.dashArray.join(',') || '0,0'
const curSeparatorDom = separatorOptionDom.querySelector<HTMLLIElement>(
`[data-separator='${separator}']`
)!
if (curSeparatorDom) {
curSeparatorDom.classList.add('active')
}
}
// 富文本
fontOptionDom
.querySelectorAll<HTMLLIElement>('li')
.forEach(li => li.classList.remove('active'))
const curFontDom = fontOptionDom.querySelector<HTMLLIElement>(
`[data-family='${payload.font}']`
)
if (curFontDom) {
fontSelectDom.innerText = curFontDom.innerText
fontSelectDom.style.fontFamily = payload.font
curFontDom.classList.add('active')
}
sizeOptionDom
.querySelectorAll<HTMLLIElement>('li')
.forEach(li => li.classList.remove('active'))
const curSizeDom = sizeOptionDom.querySelector<HTMLLIElement>(
`[data-size='${payload.size}']`
)
if (curSizeDom) {
sizeSelectDom.innerText = curSizeDom.innerText
curSizeDom.classList.add('active')
} else {
sizeSelectDom.innerText = `${payload.size}`
}
payload.bold
? boldDom.classList.add('active')
: boldDom.classList.remove('active')
payload.italic
? italicDom.classList.add('active')
: italicDom.classList.remove('active')
payload.underline
? underlineDom.classList.add('active')
: underlineDom.classList.remove('active')
payload.strikeout
? strikeoutDom.classList.add('active')
: strikeoutDom.classList.remove('active')
if (payload.color) {
colorDom.classList.add('active')
colorControlDom.value = payload.color
colorSpanDom.style.backgroundColor = payload.color
} else {
colorDom.classList.remove('active')
colorControlDom.value = '#000000'
colorSpanDom.style.backgroundColor = '#000000'
}
if (payload.highlight) {
highlightDom.classList.add('active')
highlightControlDom.value = payload.highlight
highlightSpanDom.style.backgroundColor = payload.highlight
} else {
highlightDom.classList.remove('active')
highlightControlDom.value = '#ffff00'
highlightSpanDom.style.backgroundColor = '#ffff00'
}
// 行布局
leftDom.classList.remove('active')
centerDom.classList.remove('active')
rightDom.classList.remove('active')
alignmentDom.classList.remove('active')
justifyDom.classList.remove('active')
if (payload.rowFlex && payload.rowFlex === 'right') {
rightDom.classList.add('active')
} else if (payload.rowFlex && payload.rowFlex === 'center') {
centerDom.classList.add('active')
} else if (payload.rowFlex && payload.rowFlex === 'alignment') {
alignmentDom.classList.add('active')
} else if (payload.rowFlex && payload.rowFlex === 'justify') {
justifyDom.classList.add('active')
} else {
leftDom.classList.add('active')
}
// 行间距
rowOptionDom
.querySelectorAll<HTMLLIElement>('li')
.forEach(li => li.classList.remove('active'))
const curRowMarginDom = rowOptionDom.querySelector<HTMLLIElement>(
`[data-rowmargin='${payload.rowMargin}']`
)!
curRowMarginDom.classList.add('active')
// 功能
payload.undo
? undoDom.classList.remove('no-allow')
: undoDom.classList.add('no-allow')
payload.redo
? redoDom.classList.remove('no-allow')
: redoDom.classList.add('no-allow')
payload.painter
? painterDom.classList.add('active')
: painterDom.classList.remove('active')
// 标题
titleOptionDom
.querySelectorAll<HTMLLIElement>('li')
.forEach(li => li.classList.remove('active'))
if (payload.level) {
const curTitleDom = titleOptionDom.querySelector<HTMLLIElement>(
`[data-level='${payload.level}']`
)!
titleSelectDom.innerText = curTitleDom.innerText
curTitleDom.classList.add('active')
} else {
titleSelectDom.innerText = '正文'
titleOptionDom.querySelector('li:first-child')!.classList.add('active')
}
// 列表
listOptionDom
.querySelectorAll<HTMLLIElement>('li')
.forEach(li => li.classList.remove('active'))
if (payload.listType) {
listDom.classList.add('active')
const listType = payload.listType
const listStyle =
payload.listType === ListType.OL ? ListStyle.DECIMAL : payload.listType
const curListDom = listOptionDom.querySelector<HTMLLIElement>(
`[data-list-type='${listType}'][data-list-style='${listStyle}']`
)
if (curListDom) {
curListDom.classList.add('active')
}
} else {
listDom.classList.remove('active')
}
// 批注
commentDom
.querySelectorAll<HTMLDivElement>('.comment-item')
.forEach(commentItemDom => {
commentItemDom.classList.remove('active')
})
if (payload.groupIds) {
const [id] = payload.groupIds
const activeCommentDom = commentDom.querySelector<HTMLDivElement>(
`.comment-item[data-id='${id}']`
)
if (activeCommentDom) {
activeCommentDom.classList.add('active')
scrollIntoView(commentDom, activeCommentDom)
}
}
// 行列信息
const rangeContext = instance.command.getRangeContext()
if (rangeContext) {
document.querySelector<HTMLSpanElement>('.row-no')!.innerText = `${
rangeContext.startRowNo + 1
}`
document.querySelector<HTMLSpanElement>('.col-no')!.innerText = `${
rangeContext.startColNo + 1
}`
}
}
instance.listener.visiblePageNoListChange = function (payload) {
const text = payload.map(i => i + 1).join('、')
document.querySelector<HTMLSpanElement>('.page-no-list')!.innerText = text
}
instance.listener.pageSizeChange = function (payload) {
document.querySelector<HTMLSpanElement>(
'.page-size'
)!.innerText = `${payload}`
}
instance.listener.intersectionPageNoChange = function (payload) {
document.querySelector<HTMLSpanElement>('.page-no')!.innerText = `${
payload + 1
}`
}
instance.listener.pageScaleChange = function (payload) {
document.querySelector<HTMLSpanElement>(
'.page-scale-percentage'
)!.innerText = `${Math.floor(payload * 10 * 10)}%`
}
instance.listener.controlChange = function (payload) {
const disableMenusInControlContext = [
'table',
'hyperlink',
'separator',
'page-break',
'control'
]
// 菜单操作权限
disableMenusInControlContext.forEach(menu => {
const menuDom = document.querySelector<HTMLDivElement>(
`.menu-item__${menu}`
)!
payload.state === ControlState.ACTIVE
? menuDom.classList.add('disable')
: menuDom.classList.remove('disable')
})
}
instance.listener.pageModeChange = function (payload) {
const activeMode = pageModeOptionsDom.querySelector<HTMLLIElement>(
`[data-page-mode='${payload}']`
)!
pageModeOptionsDom
.querySelectorAll('li')
.forEach(li => li.classList.remove('active'))
activeMode.classList.add('active')
}
const handleContentChange = async function () {
// 字数
const wordCount = await instance.command.getWordCount()
document.querySelector<HTMLSpanElement>('.word-count')!.innerText = `${
wordCount || 0
}`
// 目录
if (isCatalogShow) {
nextTick(() => {
updateCatalog()
})
}
// 批注
nextTick(() => {
updateComment()
})
}
instance.listener.contentChange = debounce(handleContentChange, 200)
handleContentChange()
instance.listener.saved = function (payload) {
console.log('elementList: ', payload)
}
// 9. 右键菜单注册
instance.register.contextMenuList([
{
name: '批注',
when: payload => {
return (
!payload.isReadonly &&
payload.editorHasSelection &&
payload.zone === EditorZone.MAIN
)
},
callback: (command: Command) => {
new Dialog({
title: '批注',
data: [
{
type: 'textarea',
label: '批注',
height: 100,
name: 'value',
required: true,
placeholder: '请输入批注'
}
],
onConfirm: payload => {
const value = payload.find(p => p.name === 'value')?.value
if (!value) return
const groupId = command.executeSetGroup()
if (!groupId) return
commentList.push({
id: groupId,
content: value,
userName: 'Hufe',
rangeText: command.getRangeText(),
createdDate: new Date().toLocaleString()
})
}
})
}
},
{
name: '签名',
icon: 'signature',
when: payload => {
return !payload.isReadonly && payload.editorTextFocus
},
callback: (command: Command) => {
new Signature({
onConfirm(payload) {
if (!payload) return
const { value, width, height } = payload
if (!value || !width || !height) return
command.executeInsertElementList([
{
value,
width,
height,
type: ElementType.IMAGE
}
])
}
})
}
},
{
name: '格式整理',
icon: 'word-tool',
when: payload => {
return !payload.isReadonly
},
callback: (command: Command) => {
command.executeWordTool()
}
}
])
// 10. 快捷键注册
instance.register.shortcutList([
{
key: KeyMap.P,
mod: true,
isGlobal: true,
callback: (command: Command) => {
command.executePrint()
}
},
{
key: KeyMap.F,
mod: true,
isGlobal: true,
callback: (command: Command) => {
const text = command.getRangeText()
searchDom.click()
if (text) {
searchInputDom.value = text
instance.command.executeSearch(text)
setSearchResult()
}
}
},
{
key: KeyMap.MINUS,
ctrl: true,
isGlobal: true,
callback: (command: Command) => {
command.executePageScaleMinus()
}
},
{
key: KeyMap.EQUAL,
ctrl: true,
isGlobal: true,
callback: (command: Command) => {
command.executePageScaleAdd()
}
},
{
key: KeyMap.ZERO,
ctrl: true,
isGlobal: true,
callback: (command: Command) => {
command.executePageScaleRecovery()
}
}
])
return {instance};
}
运行显示这样就是可以了
3.修改功能
1.修改canvas.ts的Init函数,用于传入内容,选项,批注列表,隐藏引用的mock数据
export default function Init (content:any,options:any={},comments:any=[]) {
2.源码中是没有方法获取到最新批注的,删除的时候需要自己手动清除
const commentList=JSON.parse(JSON.stringify(comments));
// 模拟批注
const commentDom = document.querySelector<HTMLDivElement>('.comment')!
async function updateComment() {
const groupIds = await instance.command.getGroupIds()
//移除传入数据的值
const removeComment=(commentId:string)=>{
const index= comments.findIndex((item:any) => item.id === commentId)
if (index > -1) {
comments.splice(index, 1)
}
}
for (const comment of commentList) {
const activeCommentDom = commentDom.querySelector<HTMLDivElement>(
`.comment-item[data-id='${comment.id}']`
)
// 编辑器是否存在对应成组id
if (groupIds.includes(comment.id)) {
// 当前dom是否存在-不存在则追加
if (!activeCommentDom) {
const commentItem = document.createElement('div')
commentItem.classList.add('comment-item')
commentItem.setAttribute('data-id', comment.id)
commentItem.onclick = () => {
instance.command.executeLocationGroup(comment.id)
}
commentDom.append(commentItem)
// 选区信息
const commentItemTitle = document.createElement('div')
commentItemTitle.classList.add('comment-item__title')
commentItemTitle.append(document.createElement('span'))
const commentItemTitleContent = document.createElement('span')
commentItemTitleContent.innerText = comment.rangeText
commentItemTitle.append(commentItemTitleContent)
const closeDom = document.createElement('i')
closeDom.onclick = () => {
instance.command.executeDeleteGroup(comment.id)
removeComment(comment.id)
}
commentItemTitle.append(closeDom)
commentItem.append(commentItemTitle)
// 基础信息
const commentItemInfo = document.createElement('div')
commentItemInfo.classList.add('comment-item__info')
const commentItemInfoName = document.createElement('span')
commentItemInfoName.innerText = comment.userName
const commentItemInfoDate = document.createElement('span')
commentItemInfoDate.innerText = comment.createdDate
commentItemInfo.append(commentItemInfoName)
commentItemInfo.append(commentItemInfoDate)
commentItem.append(commentItemInfo)
// 详细评论
const commentItemContent = document.createElement('div')
commentItemContent.classList.add('comment-item__content')
commentItemContent.innerText = comment.content
commentItem.append(commentItemContent)
commentDom.append(commentItem)
}
} else {
// 编辑器内不存在对应成组id则dom则移除
activeCommentDom?.remove()
removeComment(comment.id)
}
}
}
3.图片粘贴带入File文件,我们这用的是id保存图片数据的,所以上传或者复制的图片都要带原始File文件上传
- 修改canvas.ts的图片上传,将file加入
const imageDom = document.querySelector<HTMLDivElement>('.menu-item__image')!
const imageFileDom = document.querySelector<HTMLInputElement>('#image')!
imageDom.onclick = function () {
imageFileDom.click()
}
imageFileDom.onchange = function () {
const file = imageFileDom.files![0]!
const fileReader = new FileReader()
fileReader.readAsDataURL(file)
fileReader.onload = function () {
// 计算宽高
const image = new Image()
const value = fileReader.result as string
image.src = value
image.onload = function () {
instance.command.executeImage({
value,
width: image.width,
height: image.height,
file
})
imageFileDom.value = ''
}
}
}
- 修改/CanvasEditor/editor/core/event/handlers/paste.ts的图片粘贴,将file加入
export function pasteImage(host: CanvasEvent, file: File | Blob) {
const draw = host.getDraw()
if (draw.isReadonly() || draw.isDisabled()) return
const rangeManager = draw.getRange()
const { startIndex } = rangeManager.getRange()
const elementList = draw.getElementList()
// 创建文件读取器
const fileReader = new FileReader()
fileReader.readAsDataURL(file)
fileReader.onload = () => {
// 计算宽高
const image = new Image()
const value = fileReader.result as string
image.src = value
image.onload = () => {
const imageElement: IElement = {
value,
type: ElementType.IMAGE,
width: image.width,
height: image.height,
file:file
}
if (~startIndex) {
formatElementContext(elementList, [imageElement], startIndex, {
editorOptions: draw.getOptions()
})
}
draw.insertElementList([imageElement])
}
}
}
- 修改/CanvasEditor/editor/dataset/constant/Element.ts的类型
export const EDITOR_ELEMENT_ZIP_ATTR: Array<keyof IElement> = [
'type',
'font',
'size',
'bold',
'color',
'italic',
'highlight',
'underline',
'strikeout',
'rowFlex',
'rowMargin',
'dashArray',
'trList',
'tableToolDisabled',
'borderType',
'borderColor',
'width',
'height',
'url',
'colgroup',
'valueList',
'control',
'checkbox',
'radio',
'dateFormat',
'block',
'level',
'title',
'listType',
'listStyle',
'listWrap',
'groupIds',
'conceptId',
'imgDisplay',
'imgFloatPosition',
'imgToolDisabled',
'textDecoration',
'extension',
'externalId',
'areaId',
'area',
'hide',
'file'
]
- 修改/CanvasEditor/editor/interface/Element.ts
export interface IImageBasic {
imgDisplay?: ImageDisplay
imgFloatPosition?: {
x: number
y: number
pageNo?: number
},
file?:File | Blob
}
4.图片预览,由于层级有问题,修改css无用,直接改css类名
修改/CanvasEditor/editor/assets/css/previewer/previewer.css的".ce-image-previewer"改为".ce-image-previewer1"
修改/CanvasEditor/editor/core/draw/particle/previewer/Previewer.ts的_drawPreviewer方法
private _drawPreviewer() {
const previewerContainer = document.createElement('div')
previewerContainer.classList.add(`${EDITOR_PREFIX}-image-previewer1`)
5.编辑器实例销毁,对外暴露方法,父组件调用
// 销毁编辑器实例
const destroyEditor = () => {
instance.value.instance.destroy();
};
6.右键导入Word文档到编辑器,导出样式错乱,不建议使用
安装npm i @hufe921/canvas-editor-plugin-docx
const registerContextMenu=()=>{
instance.value.instance.register.contextMenuList([
{
name: "导入文档",
when: payload => !payload.isReadonly,
callback: (command) => {
const docxFileInput = document.querySelector("#file-docx") as any;
docxFileInput.click();
docxFileInput.onchange = () => {
const file = docxFileInput?.files?.[0];
if (!file) return;
const reader = new FileReader();
reader.onload = (event) => {
const buffer = event?.target?.result;
if (buffer instanceof ArrayBuffer) {
instance.value.instance.command.executeImportDocx({
arrayBuffer: buffer,
});
}
docxFileInput.value = "";
};
reader.readAsArrayBuffer(file);
};
}
},
// {
// name: "导出文档",
// when: (payload) => true,
// callback: (command) => {
// instance.value.instance.command.executeExportDocx({
// fileName: "canvas-editor",
// });
// },
// },
])
}
onMounted(()=>{
instance.value = Init(JSON.parse(JSON.stringify(props.content)),props.options,props.commentList);
instance.value.instance.use(docxPlugin);
registerContextMenu();
})