Project Icon

zhlint

智能中文文本校对与格式化工具

zhlint 是一款专门针对中文文本的智能校对与格式化工具。它可自动优化中英文间距、转换标点符号全半角、统一中文标点简繁体等。该工具支持 Markdown 和 Hexo 标签语法,既可通过命令行使用,也可作为 Node.js 包集成。zhlint 提供多种自定义规则,使用者可根据实际需求灵活调整校对标准。

logo zhlint

一个用于中文文本内容的检查工具。

如何安装

你可以通过 npmyarn 轻松安装 zhlint

# 通过 npm 安装
npm install zhlint -g

# 或通过 yarn 安装
yarn global add zhlint

# 或通过 pnpm 安装
pnpm add zhlint -g

使用方法

作为命令行工具

# 使用通配符匹配文件,对其进行检查,并打印验证报告,
# 如果发现任何错误,则以代码 `1` 退出。
zhlint <文件模式>

# 使用通配符匹配文件并修复所有可能发现的错误。
zhlint <文件模式> --fix

# 检查文件并将修复后的内容输出到另一个文件
zhlint <输入文件路径> --output=<输出文件路径>

# 打印使用说明
zhlint --help

验证报告可能看起来像这样:

高级用法

zhlint 还支持 rc 和 ignore 配置文件来自定义规则:

# 默认为 .zhlintrc
zhlint --config <文件路径>

# 默认为 .zhlintignore
zhlint --ignore <文件路径>
zhlint --file-ignore <文件路径>

# 默认为 .zhlintcaseignore
zhlint --case-ignore <文件路径>

# 默认为当前目录
zhlint --dir <路径>

在 rc 配置文件中,你可以编写如下 JSON:

{
  "preset": "default",
  "rules": {
    "adjustedFullwidthPunctuation": ""
  }
}

更多详细信息,请参阅支持的规则

在 file-ignore 文件中,你可以使用 .gitignore 语法编写一些行来忽略文件:

在 case-ignore 文件中,你可以编写一些被忽略的情况,如:

( , )

更多详细信息,请参阅设置忽略的情况

作为 Node.js 包

const { run, report } = require('zhlint')

const value = '自动在中文和English之间加入空格'
const options = { rules: { preset: 'default' } }
const output = run(value, options)

// 打印 '自动在中文和 English 之间加入空格''
console.log(output.result)

// 打印验证报告
report([output])

验证报告的格式更像这样:

1:6 - 此处中英文内容之间需要一个空格

自动在中文和English之间加入空格
      ^

1:13 - 此处中英文内容之间需要一个空格

自动在中文和English之间加入空格
             ^
无效文件:
- foo.md

发现 2 个错误。

高级用法

zhlint 还支持 rc 和 ignore 配置文件来自定义规则:

const { readRc, runWithConfig } = require('zhlint')

const value = '自动在中文和English之间加入空格'

const dir = '...' // 目标目录路径
const configPath = '...' // 配置文件路径
const fileIgnorePath = '...' // file-ignore 文件路径
const caseIgnorePath = '...' // case-ignore 文件路径

const config = readRc(dir, configPath, fileIgnorePath, caseIgnorePath)
const output = runWithConfig(value, config)

// ... 进一步操作

作为独立包

你可以在 dist/zhlint.js 中找到一个独立版本的 JavaScript 文件。例如,你可以直接将其作为 <script> 标签添加到你的浏览器中。然后就会有一个全局变量 zhlint 供你使用。

API

  • run(str: string, options?: Options): Result:检查特定内容。
    • 参数:
      • str:你想要检查的文本内容。
      • options:一些配置选项。
    • 返回:
      • 单个输入字符串的结果。它包含修复后的文本内容作为 value 以及所有 validations 的信息。
  • report(results: Result[], logger?: Console): void:打印每个文件的验证报告。
    • 参数:
      • results:所有已检查结果的数组。
      • logger:日志记录器实例,默认在 Node.js/浏览器中是 console
  • readRc: (dir: string, config: string, fileIgnore: string, caseIgnore: string, logger?: Console) => Config:从 rc 和 ignore 文件读取配置。
  • runWithConfig(str: string, config: Config): Result:使用 rc 配置检查特定内容。

选项

自定义你自己的检查配置和其他高级选项。

type Options = {
  rules?: RuleOptions
  hyperParse?: string[]
  ignoredCases?: IgnoredCase[]
  logger?: Console
}
  • rules:自定义检查配置。它可以是 undefined,表示不进行任何检查。它可以是 { preset: 'default' },表示使用默认配置。有关 RuleOptions 的更多详细信息,请参阅支持的规则
  • hyperParse:通过名称自定义超级解析器。它可以是 undefined,表示仅使用默认的忽略情况解析器Markdown 解析器Hexo 标签解析器
  • ignoredCases:提供你想要跳过的例外情况。
  • logger:与 report(...) 中的参数相同。

RC 配置

  • presetstring(可选)
  • rules:不包含 preset 字段的 RuleOptions。(可选)
  • hyperParsersstring[](可选)
  • caseIgnoresstring[],优先级低于 .zhlintcaseignore。(可选)

输出

type Result = {
  // 文件的基本信息和可用性
  file?: string
  disabled: boolean

  // 文件的原始内容
  origin: string

  // 所有错误消息
  validations: Validation[]
}

type Validation = {
  message: string
  index: number
  length: number
}
  • Result
    • file:文件名。这是一个可选字段,仅在命令行界面中使用。
    • origin:原始文本内容。
    • result:最终修正后的文本内容。
    • validations:所有验证信息。
  • Validation
    • index:目标标记在输入字符串中的索引。
    • length:目标标记在输入字符串中的长度。
    • message:以自然语言描述的验证信息。

功能

支持 Markdown 语法

我们默认支持对 Markdown 语法的文本内容进行检查。例如:

run('自动在_中文_和**English**之间加入空格', options)

它会先分析 Markdown 语法并提取纯文本内容进行检查。之后,修正后的纯文本内容会被替换回原始的 Markdown 字符串,并作为结果中的 value 输出。

支持 Hexo 标签语法

特别地,我们支持 Hexo 标签语法,因为在使用 Hexo 构建 Vue.js 网站时,markdown 源文件多多少少会包含这些特殊标签,从而导致意外的结果。

因此,我们默认额外跳过 Hexo 风格的标签。例如:

run(
  '现在过滤器只能用在插入文本中 (`{% raw %}{{ }}{% endraw %}` tags)。',
  options
)

设置忽略情况

在某些实际情况下,我们有特殊的文本内容因故意不遵循规则。因此,我们可以使用 ignoredCases 选项来配置。例如,我们想保留括号内的空格,这在默认情况下是无效的。那么我们可以在文件中的任何位置添加一行 HTML 注释:

<!-- 正确的情况 -->

前面的文本 (里面的文本) 后面的文本

<!-- 错误的情况 -->

vm.$on( event, callback )

<!-- 然后我们可以在下面写这行来使其生效 -->
<!-- zhlint ignore: ( , ) -->

或者直接作为选项传递:

run(str, { ignoredCases: { textStart: '( ', textEnd: ' )' } })

如果你想忽略整个文件,你也可以添加这个 HTML 注释:

<!-- zhlint disabled -->

支持的预处理器(超级解析器)

  • ignore:通过 HTML 注释 <!-- zhlint ignore: ... --> 查找所有被忽略的部分。
  • hexo:查找所有 Hexo 标签以避免它们被解析。
  • markdown:解析 markdown 语法并查找所有块级文本和内联级标记。

支持的规则

大多数规则来自于过去在 W3C 中文文字排版需求W3C HTML 中文兴趣小组Vue.js 中文文档站点 的翻译经验。

... 这部分可能会有争议。因此,如果你对某些地方感到不适,我们非常乐意了解并改进。随时欢迎提出 issue。然后我们可以讨论可能的更好选择或决定。

type RuleOptions = {
  /* 预设 */

  // 自定义预设,目前仅支持:
  // - `'default'`
  preset?: string

  /* 标点符号 */

  // 将这些标点符号转换为半角。
  // 默认预设:`()`
  // 例如:`(文字)` -> `(文字)`
  halfwidthPunctuation?: string

  // 将这些标点符号转换为全角。
  // 默认预设:`,。:;?!""''`
  // 例如:`文字,文字.` -> `文字,文字。`
  fullwidthPunctuation?: string

  // 在处理它们周围的空格问题时,将这些全角标点符号视为半全角标点符号。
  // 因为像引号这样的东西在现代中文字体中只以半角形式呈现。
  // 默认预设:`""''`
  adjustedFullwidthPunctuation?: string

  // 将繁体中文标点符号转换为简体或反之。
  // 默认预设:`simplified`
  // 例如:`「文字」` -> `"文字"`
  //
  // 除此之外,我们还统一了一些常见的标点符号:
  //
  // // U+2047 双问号,U+203C 双叹号
  // // U+2048 问号叹号,U+2049 叹号问号
  // '??': ['⁇'],
  // '!!': ['‼'],
  // '?!': ['⁈'],
  // '!?': ['⁉'],
  //
  // // U+002F 斜线,U+FF0F 全角斜线
  // '/': ['/', '/'],
  //
  // // U+FF5E 全角波浪线
  // '~': ['~', '~'],
  //
  // // U+2026 水平省略号,U+22EF 中线水平省略号
  // '…': ['…', '⋯'],
  //
  // // U+25CF 黑圆圈,U+2022 项目符号,U+00B7 中点,
  // // U+2027 连字点,U+30FB 片假名中点
  // '·': ['●', '•', '·', '‧', '・'],
  //
  // 高级用法:你也可以指定一个更详细的映射,如:
  //
  // ```
  // {
  //   default: true, // 遵循所有默认预设
  //   '「': ['"', '【'], // 将 `"` 或 `【` 转换为 `「`
  //   '」': ['"', '】'], // 将 `"` 或 `】` 转换为 `」`
  //  '…': true, // 遵循此字符的默认预设
  //  '·': false, // 不统一这些字符中的任何一个
  // }
  // ```
  unifiedPunctuation?:
    | 'traditional'
    | 'simplified'
    | (Record<string, boolean | string[]> & { default: boolean })

  // 特殊情况:对缩写跳过 `fullWidthPunctuation`。
  // 默认预设:
  // `['Mr.','Mrs.','Dr.','Jr.','Sr.','vs.','etc.','i.e.','e.g.','a.k.a']`
  skipAbbrs?: string[]

  /* 字母周围的空格 */

  // 默认预设:`true`
  // - `true`:一个空格
  // - `undefined`:不做任何处理
  // 例如:`foo  bar` -> `foo bar`
  spaceBetweenHalfwidthContent?: boolean

  // 默认预设:`true`
  // - `true`:零个空格
  // - `undefined`:不做任何处理
  // 例如:`文 字` -> `文字`
  noSpaceBetweenFullwidthContent?: boolean

  // 默认预设:`true`
  // - `true`:一个空格
  // - `false`:零个空格
  // - `undefined`:不做任何处理
  // 例如:`文字 foo文字` -> `文字 foo 文字`(`true`)
  // 例如:`文字foo 文字` -> `文字foo文字`(`false`)
  spaceBetweenMixedwidthContent?: boolean
// 特殊情况:跳过数字与中文单位之间的`spaceBetweenMixedWidthContent`
// 默认预设:`年月日天号时分秒`
skipZhUnits?: string

/* 标点符号周围的空格 */

// 默认预设:`true`
// - `true`:零空格
// - `undefined`:不做处理
// 例如:`文字 ,文字` -> `文字,文字`
noSpaceBeforePauseOrStop?: boolean

// 默认预设:`true`
// - `true`:一个空格
// - `false`:零空格
// - `undefined`:不做处理
// 例如:`文字,文字` -> `文字, 文字`(`true`)
// 例如:`文字, 文字` -> `文字,文字`(`false`)
spaceAfterHalfwidthPauseOrStop?: boolean

// 默认预设:`true`
// - `true`:零空格
// - `undefined`:不做处理
// 例如:`文字, 文字` -> `文字,文字`
noSpaceAfterFullwidthPauseOrStop?: boolean

/* 引号周围的空格 */

// 默认预设:`true`
// - `true`:一个空格
// - `false`:零空格
// - `undefined`:不做处理
// 例如:`文字 "文字"文字` -> `文字 "文字" 文字`(`true`)
// 例如:`文字"文字" 文字` -> `文字"文字"文字`(`false`)
spaceOutsideHalfwidthQuotation?: boolean

// 默认预设:`true`
// - `true`:零空格
// - `undefined`:不做处理
// 例如:`文字 "文字" 文字` -> `文字"文字"文字`
noSpaceOutsideFullwidthQuotation?: boolean

// 默认预设:`true`
// - `true`:零空格
// - `undefined`:不做处理
// 例如:`文字" 文字 "文字` -> `文字"文字"文字`
noSpaceInsideQuotation?: boolean

/* 括号周围的空格 */

// 默认预设:`true`
// - `true`:一个空格
// - `false`:零空格
// - `undefined`:不做处理
spaceOutsideHalfwidthBracket?: boolean

// 默认预设:`true`
// - `true`:零空格
// - `undefined`:不做处理
noSpaceOutsideFullwidthBracket?: boolean

// 默认预设:`true`
// - `true`:零空格
// - `undefined`:不做处理
noSpaceInsideBracket?: boolean

/* 代码周围的空格 */

// 默认预设:`true`
// - `true`:一个空格
// - `false`:零空格
// - `undefined`:不做处理
// 例如:'文字 `code`文字' -> '文字 `code` 文字'('true')
// 例如:'文字`code` 文字' -> '文字`code`文字'('false')
spaceOutsideCode?: boolean

/* Markdown/HTML包裹符周围的空格 */

// 默认为`true`
// - `true`:零空格
// - `undefined`:不做处理
// 例如:`文字** foo **文字` -> `文字 **foo** 文字`
noSpaceInsideHyperMark?: boolean

/* 开头/结尾的空格 */

// 默认为`true`
// 例如:` 文字 ` -> `文字`
trimSpace?: boolean

/* 跳过纯西文句子 */

// 默认为`true`
skipPureWestern?: boolean
}

更多信息

zhlint现已在GitHub上开源,欢迎提交问题

项目侧边栏1项目侧边栏2
推荐项目
Project Cover

豆包MarsCode

豆包 MarsCode 是一款革命性的编程助手,通过AI技术提供代码补全、单测生成、代码解释和智能问答等功能,支持100+编程语言,与主流编辑器无缝集成,显著提升开发效率和代码质量。

Project Cover

AI写歌

Suno AI是一个革命性的AI音乐创作平台,能在短短30秒内帮助用户创作出一首完整的歌曲。无论是寻找创作灵感还是需要快速制作音乐,Suno AI都是音乐爱好者和专业人士的理想选择。

Project Cover

白日梦AI

白日梦AI提供专注于AI视频生成的多样化功能,包括文生视频、动态画面和形象生成等,帮助用户快速上手,创造专业级内容。

Project Cover

有言AI

有言平台提供一站式AIGC视频创作解决方案,通过智能技术简化视频制作流程。无论是企业宣传还是个人分享,有言都能帮助用户快速、轻松地制作出专业级别的视频内容。

Project Cover

Kimi

Kimi AI助手提供多语言对话支持,能够阅读和理解用户上传的文件内容,解析网页信息,并结合搜索结果为用户提供详尽的答案。无论是日常咨询还是专业问题,Kimi都能以友好、专业的方式提供帮助。

Project Cover

讯飞绘镜

讯飞绘镜是一个支持从创意到完整视频创作的智能平台,用户可以快速生成视频素材并创作独特的音乐视频和故事。平台提供多样化的主题和精选作品,帮助用户探索创意灵感。

Project Cover

讯飞文书

讯飞文书依托讯飞星火大模型,为文书写作者提供从素材筹备到稿件撰写及审稿的全程支持。通过录音智记和以稿写稿等功能,满足事务性工作的高频需求,帮助撰稿人节省精力,提高效率,优化工作与生活。

Project Cover

阿里绘蛙

绘蛙是阿里巴巴集团推出的革命性AI电商营销平台。利用尖端人工智能技术,为商家提供一键生成商品图和营销文案的服务,显著提升内容创作效率和营销效果。适用于淘宝、天猫等电商平台,让商品第一时间被种草。

Project Cover

AIWritePaper论文写作

AIWritePaper论文写作是一站式AI论文写作辅助工具,简化了选题、文献检索至论文撰写的整个过程。通过简单设定,平台可快速生成高质量论文大纲和全文,配合图表、参考文献等一应俱全,同时提供开题报告和答辩PPT等增值服务,保障数据安全,有效提升写作效率和论文质量。

投诉举报邮箱: service@vectorlightyear.com
@2024 懂AI·鲁ICP备2024100362号-6·鲁公网安备37021002001498号