中文打字练习 & 跟打器 — 支持 码长 / 击键 / 速度 / 键准 专业统计,Linux Wayland 原生可用
Chinese typing practice tool with keystroke statistics (码长/击键/键准), native Linux Wayland support via evdev
TypeType 是一个基于 PySide6 + QML 的中文打字练习跟打器,提供打字圈常用的专业统计指标:
- 码长(击/字)— 每个中文字符平均消耗的按键次数
- 击键(击/秒)— 每秒物理按键次数
- 速度(字/分钟)— 每分钟输入中文字符数
- 键准(%)— 有效按键比例,反映按键效率(错键率)
- 错字 / 回改 / 退格 — 错误修正统计
核心优势: 通过 Linux evdev 直接读取内核键盘事件,完全绕过 Wayland text-input-v3 协议对浏览器/应用层按键事件的屏蔽,实现 Wayland 下真实的物理击键统计。这是目前少数能在 Wayland 上准确统计码长和击键的打字工具之一。
- 📊 实时 码长 / 击键 / 速度 / 键准 统计,配合 错字 / 回改 / 退格 分析
- 📈 字符级统计(SQLite 持久化)与薄弱字分析
- 🏆 服务端排行榜与成绩提交(支持分片模式聚合成绩)
- 📝 本地文本与网络文本统一载文
- 🪟 跨平台支持(详见 💻 支持平台)
| 平台 | 支持状态 | 键盘监听方式 |
|---|---|---|
| Linux Wayland | ✅ 原生支持 | evdev 直接读取内核事件 |
| Linux X11 | ✅ 支持 | X11 事件 |
| macOS | ✅ 支持 | Quartz CGEventTap 全局监听 |
| Windows | ✅ 支持 | QML Text 变化统计 |
💻 详细权限配置:见下方 Linux Wayland 权限 和 macOS 输入监控权限 章节。
不了解 Wayland/X11 的区别可以先阅读:《细说 Wayland 和 X11》。
浏览器(Firefox / Chromium)在 Wayland 下使用 text-input-v3 协议与输入法通信。拼音输入时,每个按键走 IME composition 流程,浏览器的 keydown/keyup 事件不为拼音按键触发,只在 compositionend 时提交最终中文字符。因此:
- ❌ 网页版跟打器在 Wayland 上无法统计真实击键数和码长
- ✅ TypeType 使用 evdev 直接读取
/dev/input/event*,在物理键盘层面计数,不依赖任何显示协议
原理:evdev 是 Linux 内核的输入设备接口,在 Wayland 合成器处理键盘事件之前就能拿到原始按键数据。
uv sync
uv run python main.py联网功能说明: 排行榜、载文等联网功能依赖 typetype-server 服务端,默认配置指向
127.0.0.1:8080。当前服务端还存在不少安全性问题,所以 IP 暂不便公开。想体验在线服务可以联系whynusn@qq.com,或者参考服务端仓库自行本地部署后修改config/config.example.json中的base_url。仅使用本地打字功能则无需服务端。
全局键盘监听需要读取 /dev/input/event*,通常需要将用户加入 input 组:
sudo usermod -aG input $USER重新登录后生效。即使没有该权限,程序也会优雅降级,不影响基础打字功能。
macOS 下准确统计中文输入法的物理击键需要 Quartz 全局键盘监听。首次运行时,如果系统未授权,程序会降级到 QML 文本变化统计,基础打字仍可用,但码长/击键会按上屏字符估算。
如需准确击键统计,请在系统设置中授予运行 TypeType 的终端或打包应用以下权限后重启程序:
- 隐私与安全性 → 输入监控
- 隐私与安全性 → 辅助功能
# 运行
uv run python main.py
# 测试
uv run pytest
# 代码检查
uv run ruff check .
uv run ruff format --check .
# 自动格式化
uv run ruff format .| 层 | 技术 |
|---|---|
| 桌面 UI | PySide6 + QML + RinUI |
| 后端语言 | Python 3.12+ |
| 架构 | Clean Architecture + Ports & Adapters |
| 本地持久化 | SQLite |
| 网络请求 | httpx |
| 包管理 | uv |
| 击键监听 | evdev(Linux)/ Quartz CGEventTap(macOS) |
typetype/
├── main.py
├── docs/
├── config/
├── resources/
├── src/
│ ├── backend/
│ └── qml/
├── tests/
└── RinUI/
presentation/ # Bridge + Adapters,负责 QML/Qt 适配
application/ # UseCases + Gateways,负责编排与边界整合
domain/ # 纯业务逻辑(TypingService 计算码长/击键/键准)
ports/ # 抽象协议(KeyListener、TextProvider 等 Port 定义)
integration/ # Port 实现(GlobalKeyListener 读 evdev、ApiClient 等)
infrastructure/ # 通用基础设施(网络异常模型、加密等)
models/ # Entity / DTO
workers/ # 后台任务(异步加载文本/排行榜/薄弱字等)
config/ # 运行时配置(RuntimeConfig、文本来源配置)
security/ # 加密与安全存储(crypt、secure_storage)
utils/ # 工具类(Logger、text_id)
物理按键 → evdev (/dev/input/event*) → GlobalKeyListener (KeyListener Port 实现)
→ keyPressed 信号 → TypingAdapter.handlePressed()
→ TypingService.accumulate_key() → SessionStat.key_stroke_count
→ 码长 = key_stroke_count / char_count
→ 击键 = key_stroke_count / time
→ 键准 = (key_stroke_count - backspace - correction × 码长) / key_stroke_count
当前默认配置中包含:
- 网络来源:
jisubei - 本地来源:
builtin_demo、fst_500、mid_500、lst_500、essential_single_char
配置文件示例见:config/config.example.json
默认只输出 warning 及以上日志,可通过环境变量调整:
TYPETYPE_DEBUG=1TYPETYPE_LOG_LEVEL=debug|info|warning|error|none
示例:
TYPETYPE_DEBUG=1 uv run python main.pyuv run python -m ensurepip --upgrade
uv pip install --upgrade nuitka --index-url https://pypi.org/simple
uv run python -m nuitka main.py \
--follow-imports \
--enable-plugin=pyside6 \
--include-qt-plugins=qml \
--include-package=RinUI \
--include-data-dir=RinUI=RinUI \
--include-data-dir=config=config \
--output-dir=deployment \
--quiet \
--noinclude-qt-translations \
--standalone \
--noinclude-dlls=libQt6WebEngine* \
--include-data-dir=src/qml=src/qml \
--include-data-dir=resources/texts=resources/texts \
--include-data-files=resources/images/TypeTypeLogo.png=resources/images/TypeTypeLogo.png \
--include-data-files=resources/fonts/HarmonyOS_Sans_SC_Regular-subset.ttf=resources/fonts/HarmonyOS_Sans_SC_Regular-subset.ttf \
--include-data-files=resources/fonts/LXGWWenKai-Regular-subset.ttf=resources/fonts/LXGWWenKai-Regular-subset.ttf \
--include-data-dir=resources/trainer=resources/trainer \
--include-data-dir=resources/ziti=resources/zitiWindows 建议追加:
--assume-yes-for-downloads --windows-console-mode=disable --include-windows-runtime-dlls=yes --noinclude-dlls=Qt6WebEngine*
| 文档 | 面向 | 职责 | 何时读 |
|---|---|---|---|
| ARCHITECTURE.md | 所有人 | 唯一事实来源:架构设计、代码分层、数据流、陷阱集锦 | 第一次了解项目 |
| AGENTS.md | AI 开发者 | 开发约束与陷阱:编码规范、已知坑位、验证要求 | 修改代码前 |
| docs/reference/ | 开发者/AI | 速查表:配置字段、QML 页面、Bridge API、端点清单 | 需要快速查找 |
| docs/guides/ | 开发者/AI | 操作手册:开发、测试、部署等任务步骤 | 执行具体工作流 |
| docs/examples/ | 开发者/AI | 代码示例:独立可运行示例和最佳实践 | 学习扩展模式 |
| docs/history/ | 参考 | 历史归档:完成的设计文档、bug 记录、演进记录 | 理解过去决策 |
| docs/meta/README.md | AI/维护者 | 文档规范:文档职责定义、权威矩阵、同步规则 | 维护文档或不确定信息来源 |
| CHANGELOG.md | 用户/维护者 | 发布历史:版本变更、用户可见改动 | 查看版本演进 |
README.md (快速入门)
→ ARCHITECTURE.md § 快速开始
→ ARCHITECTURE.md § 一句话先理解项目
ARCHITECTURE.md § 分层架构(理解分层)
→ AGENTS.md § 当前架构(快速索引)
→ AGENTS.md § 已知陷阱(查阅相关坑位)
→ 对应源文件位置
AGENTS.md § 已知陷阱(关键词搜索)
→ 对应 ARCHITECTURE.md 相关部分
→ docs/history/ (查历史 bug 记录)
docs/reference/README.md (速查入口)
→ 对应文件(config.md / qml-pages.md / bridge-slots.md / api-endpoints.md / typing-metrics.md)
docs/guides/README.md
→ 对应指南
→ docs/reference/ 或 docs/examples/ 按需查证
中文打字练习, 跟打器, 码长统计, 击键统计, 打字速度测试, Wayland 打字工具, evdev 键盘监听, Linux 中文输入练习, Chinese typing practice, typing tutor, keystroke statistics, code length, typing speed test
在对话中输入以下关键词,AI 会自动执行对应操作:
| 你说 | AI 会做什么 |
|---|---|
| "项目概览" | 汇总 README + ARCHITECTURE 的核心内容 |
| "同步文档" | 检查代码变更,同步更新受影响的文档 |
| "检查文档" | 运行 scripts/verify-framework.sh 验证完整性 |
| "记录决策" | 引导创建新的架构决策记录 |
| "更新 CHANGELOG" | 根据近期变更补充发布历史 |
- RinUI — Fluent Design 风格 QML 组件库(MIT License © 2025 RinLit)
- 晴发文 / TypeSunny(a810439322) — macOS 适配贡献者(Quartz CGEventTap 键盘监听),同时维护 TypeSunny(晴跟打) 开源跟打器项目
MIT © 2026 whynusn