一个通过canvas将长文本分成若干页渲染的js工具,处理排版禁则,支持辅助平面Unicode字符。目前仅工作于浏览器环境。
在浏览器中,尽管css有分栏布局,但却很难实现“分页”。浏览器没有提供计算一个盒子可以显示多少文本的API,只有Canvas的上下文提供了一个measureText()
方法,但它只能简单的计算一段文本的渲染宽度,当文本不止一行时就无法计算了。由于不同字体不同字符有不同的宽度,再加上各浏览器对排版禁则的处理不一致,基本杜绝了计算浏览器原生DOM元素能够显示多少文本的可能。
于是我参考W3C对中文排版的草案中的排版禁则,通过CanvasRenderingContext2D.measureText()方法计算字符宽度,对文本进行分页。
名称 | 符号 |
---|---|
点号 | 、,,..。::;;!!?? |
结束引号 | '"」』”’ |
结束括号 | )]})】〗〕]} |
结束乙式书名号 | 》〉 |
连接号 | –~~— |
间隔号 | ·.‧•・ |
分隔号 | // |
名称 | 符号 |
---|---|
开始引号 | 「『“‘ |
开始括号 | ([{(【〖〔[〔 |
开始单双书名号 | 《〈 |
分隔号 | // |
名称 | 符号 |
---|---|
破折号 | ── |
省略号 | …… |
本项目默认按照上述禁则处理排版禁则,但可以通过选项控制排版禁则。 草案中规定,当碰到行首为(行首禁则中的)标点时,应遵守「先挤进,后推出」原则,即先尝试压缩当前行的标点,无法挤压再取前一行的最后一个字至下一行。考虑到标点符号压缩的复杂性,本项目暂未实现该特性,而是直接尝试取上一行的最后一字到下一行,如果直到上一行首都没找到允许出现在行首的字符,则采取不处理的方式。
本项目定义了文本片段(text fragment)的概念。通过在选项中指定一个文本片段数组,可以对每一个片段设置不同的样式,如字体、颜色、对齐方式等,具体请参考配置项。一般来说,一个文本片段即一个文本段落,虽然文本片段内也支持换行,但不会有段间距和行缩进。在本项目文档中文本片段和文本段落等价,都是指FragmentOptions
定义的文本片段。
本项目支持以下特性:
- 每个段落单独设置字体、缩进、边距、字体颜色等;
- 支持两端对齐;
- 支持自定义排版禁则;
- 可配置段落之间边距是否折叠;
- 支持rtl模式;
这是一个在线例子。
本项目仅支持浏览器环境,且依赖于Canvas 2D上下文,请参考canvas兼容性。
// 计算分页
function computeTextFrames(options: FrameOptions): TextFrame[];
// 渲染分页到canvas
function renderFrame(context: CanvasRenderingContext2D, frame: TextFrame, clear: boolean): void;
npm install @mattuy/text-frame --save
// cjs
// const { computeTextFrames, renderFrame } = require('@mattuy/text-frame');
// or esm
import { computeTextFrames, renderFrame } from '@mattuy/text-frame/esm';
const frames = computeTextFrames({
viewWidth: 320,
viewHeight: 640,
fontSize: 16,
margin: 8,
color: '#000',
fragments: [
{
color: 'green',
fontSize: 20,
margin: 32,
textAlign: 'center',
text: "标题"
},
{
textIndent: 32,
fontFamily: 'serif',
margin: { bottom: 32 },
marginCollapse: true,
textAlign: 'justify',
text: "这是一个多行文本。"
}
]
});
const canvas = document.createElement('canvas');
canvas.style.width = '320px';
canvas.style.height = '640px';
document.body.append(canvas);
renderFrame(canvas.getContext('2d'), frames[0]);
<script src="text-frame/dist/umd/text-frame-min.js"></script>
<script>
const frames = TextFrame.computeTextFrames({
// ...options
});
console.log(frames)
// ...render
</script>
配置说明请参考配置项。
-
克隆仓库
git clone https://github.com/mattuylee/text-frame.git
-
安装依赖
cd text-frame
npm install
npm run dist
npm run example
执行npm run debug
,然后rollup会监听src下文件的变化自动重新编译。可浏览器直接打开example/index.html
测试。
MIT