很多人第一次接触 Textual,是从官方文档的 Hello World 开始的。三行代码,终端里弹出一个带边框的窗口——酷。然后呢?然后就懵了。
App 是什么?Screen 又是什么?compose() 里返回的 ComposeResult 到底是个啥类型?这些问题不搞清楚,写出来的代码就是一堆"能跑但不知道为什么能跑"的玩意儿。
我在用 Textual 开发一个本地文件管理工具的时候,前两周基本上就是在和这套结构死磕。踩了不少坑,也慢慢摸出了门道。今天就把这些东西掰开揉碎讲一讲——不是翻译文档,是真正从"为什么这么设计"的角度来聊。
先说 App。
简单粗暴地理解:App 就是你整个终端应用的容器。它负责启动事件循环、管理屏幕栈、处理全局按键绑定、控制主题切换……一句话,它是老板。
python1from textual.app import App, ComposeResult
2from textual.widgets import Header, Footer, Label
3
4class MyApp(App):
5"""一个最简单的 Textual 应用"""
6
7CSS = """
8 Label {
9 content-align: center middle;
10 height: 1fr;
11 }
12 """
13
14def compose(self) -> ComposeResult:
15yield Header()
16yield Label("你好,Textual!")
17yield Footer()
18
19if __name__ == "__main__":
20 app = MyApp()
21 app.run()
这段代码能跑。但你注意到没有——compose() 出现在了 App 里,而不是 Screen 里。这是初学者最容易混淆的地方。
App 本身也是一个隐式的 Screen。 准确说,当你在 App 里直接写 compose(),Textual 会自动把它包成一个默认的 Screen 推入屏幕栈。这是个"便捷通道",适合写简单的单屏应用。但一旦你的应用有多个界面,这条路就走不通了。
TITLE 和 SUB_TITLE 控制顶部 Header 显示的内容。CSS 或 CSS_PATH 用来注入样式。BINDINGS 定义全局快捷键。这三个是使用频率最高的类变量,几乎每个项目都要用到。
python1class MyApp(App):
2TITLE = "文件管理器"
3SUB_TITLE = "v1.0.0 - Windows 专属版"
4CSS_PATH = "style.tcss"
5
6BINDINGS = [
7 ("q", "quit", "退出"),
8 ("d", "toggle_dark", "切换暗色模式"),
9 ]action_toggle_dark() 这个方法不用自己写,App 基类已经内置了。这是 Textual 的一个设计哲学——把常见操作都内置进去,让你专注业务逻辑。
Screen 才是你日常打交道最多的东西。
如果说 App 是整个剧场,那 Screen 就是舞台上的一幕。你可以有"主菜单"这一幕,有"设置界面"这一幕,有"确认对话框"这一幕——它们各自独立,各自有自己的组件树、自己的样式、自己的事件处理逻辑。
python1from textual.app import App, ComposeResult
2from textual.screen import Screen
3from textual.widgets import Header, Footer, Button, Label
4from textual.containers import Container
5
6class MainScreen(Screen):
7"""主界面"""
8
9def compose(self) -> ComposeResult:
10yield Header()
11with Container():
12yield Label("欢迎使用文件管理器", id="welcome")
13yield Button("打开设置", id="btn-settings", variant="primary")
14yield Button("退出", id="btn-quit", variant="error")
15yield Footer()
16
17def on_button_pressed(self, event: Button.Pressed) -> None:
18if event.button.id == "btn-settings":
19 self.app.push_screen(SettingsScreen())
20elif event.button.id == "btn-quit":
21 self.app.exit()
22
23
24class SettingsScreen(Screen):
25"""设置界面"""
26
27def compose(self) -> ComposeResult:
28yield Header()
29yield Label("这里是设置页面")
30yield Button("返回", id="btn-back")
31yield Footer()
32
33def on_button_pressed(self, event: Button.Pressed) -> None:
34if event.button.id == "btn-back":
35 self.app.pop_screen()
36
37
38class MyApp(App):
39SCREENS = {"main": MainScreen, "settings": SettingsScreen}
40
41def on_mount(self) -> None:
42 self.push_screen("main")
注意几个细节。
SCREENS 字典是可选的,用来提前注册屏幕——好处是可以用字符串名称来引用,而不是每次都实例化一个新对象。push_screen() 把新屏幕压入栈顶,pop_screen() 弹出当前屏幕回到上一层。这是个栈结构,不是替换,是叠加。
这个设计有个很实用的场景:弹出确认框。你不需要销毁当前界面再重建,直接 push_screen(ConfirmDialog()) 就行,用户确认或取消后 pop_screen() 回来,原来的界面状态完好无损。
on_mount() 在屏幕被推入栈并完成渲染后触发,适合做数据加载。on_unmount() 在屏幕被弹出时触发,适合做清理工作。
python1class DataScreen(Screen):
2def compose(self) -> ComposeResult:
3yield Header()
4yield Label("加载中...", id="status")
5yield Footer()
6
7async def on_mount(self) -> None:
8# 模拟异步数据加载
9await self.load_data()
10
11async def load_data(self) -> None:
12import asyncio
13await asyncio.sleep(1) # 实际项目里换成真实的 IO 操作
14 self.query_one("#status", Label).update("数据加载完成!")Textual 基于 asyncio,这意味着你可以在 on_mount 里直接写异步代码,不用担心阻塞 UI。这在 Windows 下开发尤其重要——很多文件 IO 操作如果同步执行,界面会直接卡死。
ComposeResult 是个类型别名,本质上是 Iterable[Widget]。
但它不只是一个列表。它是一个生成器协议——Textual 会在渲染时逐一消费你 yield 出来的组件,按顺序构建 DOM 树。这个设计让你可以用 Python 最自然的方式来描述界面结构。
python1from textual.app import ComposeResult
2from textual.widgets import Label, Button, Input
3from textual.containers import Vertical, Horizontal
4
5def compose(self) -> ComposeResult:
6# 最简单的用法:直接 yield 组件
7yield Label("用户名")
8yield Input(placeholder="请输入用户名", id="username")
9
10# 用容器组织布局
11with Horizontal():
12yield Button("确认", variant="success")
13yield Button("取消", variant="default")with 语法是个语法糖。Horizontal() 这类容器实现了 __enter__ 和 __exit__,在 with 块里 yield 的组件会自动成为该容器的子节点。这比手动维护父子关系优雅多了。
这是很多人没意识到的一点——compose() 是普通的 Python 函数,你可以在里面写逻辑。
python1def compose(self) -> ComposeResult:
2yield Header()
3
4# 根据条件动态生成组件
5 items = self.load_menu_items() # 从配置或数据库读取
6
7with Vertical(id="menu"):
8for item in items:
9yield Button(
10 label=item["label"],
11 id=f"menu-{item['id']}",
12 variant="primary" if item.get("highlight") else "default"
13 )
14
15yield Footer()这种动态性在实际项目里非常有用。比如根据用户权限显示不同的菜单项,或者根据配置文件渲染不同数量的输入框——都不需要什么特殊 API,就是普通的 Python 控制流。
光说不练假把式。来看一个稍微完整一点的例子——一个简单的任务管理器,有任务列表和添加任务两个界面。
python1from textual.app import App, ComposeResult
2from textual.screen import Screen
3from textual.widgets import (
4Header, Footer, Button,
5Input, Label, ListView, ListItem
6)
7from textual.containers import Vertical, Horizontal
8
9
10# 添加任务界面
11class AddTaskScreen(Screen):
12"""输入新任务的弹出界面"""
13
14def compose(self) -> ComposeResult:
15yield Header()
16with Vertical(id="form"):
17yield Label("新任务名称:")
18yield Input(placeholder="例如:完成技术文章", id="task-input")
19with Horizontal(id="buttons"):
20yield Button("添加", id="btn-add", variant="success")
21yield Button("取消", id="btn-cancel", variant="default")
22yield Footer()
23
24def on_button_pressed(self, event: Button.Pressed) -> None:
25if event.button.id == "btn-add":
26 task_name = self.query_one("#task-input", Input).value.strip()
27if task_name:
28# 通过 dismiss 把数据传回上一个 Screen self.dismiss(task_name)
29elif event.button.id == "btn-cancel":
30 self.dismiss(None)
31
32# 主界面
33class TaskListScreen(Screen):
34"""任务列表主界面"""
35
36BINDINGS = [("a", "add_task", "添加任务")]
37
38def __init__(self):
39super().__init__()
40 self._tasks: list[str] = ["写周报", "Code Review"]
41
42def compose(self) -> ComposeResult:
43yield Header()
44with Vertical():
45yield ListView(
46 *[ListItem(Label(t)) for t in self._tasks],
47 id="task-list"
48 )
49yield Button("+ 添加任务", id="btn-add", variant="primary")
50yield Footer()
51
52def action_add_task(self) -> None:
53"""快捷键 'a' 触发"""
54 self.app.push_screen(AddTaskScreen(), self._on_task_added)
55
56def on_button_pressed(self, event: Button.Pressed) -> None:
57if event.button.id == "btn-add":
58 self.action_add_task()
59
60def _on_task_added(self, task_name: str | None) -> None:
61"""AddTaskScreen dismiss 后的回调"""
62if task_name:
63 self._tasks.append(task_name)
64 list_view = self.query_one("#task-list", ListView)
65 list_view.append(ListItem(Label(task_name)))
66
67# 应用入口
68class TaskApp(App):
69TITLE = "任务管理器"
70CSS = """
71 #form { margin: 2 4; padding: 1 2; border: solid $primary; } #buttons { margin-top: 1; align: right middle; } #task-list { height: 1fr; } """
72
73def on_mount(self) -> None:
74 self.push_screen(TaskListScreen())
75
76
77if __name__ == "__main__":
78TaskApp().run()
这个例子里有几个值得单独说的点。
push_screen() 的第二个参数是回调函数——当被推入的 Screen 调用 self.dismiss(value) 时,这个回调会被触发,并接收 value 作为参数。这是 Textual 在 Screen 间传递数据的标准方式,比全局变量干净多了。
BINDINGS 定义在 Screen 里的快捷键只在该 Screen 处于栈顶时有效。这个作用域隔离非常重要——不同界面可以绑定同一个按键到不同操作,互不干扰。
在 Windows 上跑 Textual,有几个坑是 macOS/Linux 用户不会遇到的。
终端选择很关键。Windows 自带的 cmd.exe 对 ANSI 转义码的支持历来是个问题,虽然 Windows 10 之后改善了很多,但还是推荐用 Windows Terminal。PowerShell 7+ 也可以,体验比 5.x 好一大截。
字体问题。Textual 大量使用 Unicode 字符来绘制边框和图标。如果你发现界面里有奇怪的方块或者乱码,多半是字体不支持。换成 Cascadia Code 或者 Nerd Font 系列基本就能解决。
asyncio 事件循环。Windows 下 Python 3.10 之前默认使用 SelectorEventLoop,某些异步操作会有兼容性问题。如果遇到莫名其妙的错误,可以在入口处加一行:
python1import asyncio
2import sys
3
4if sys.platform == "win32":
5 asyncio.set_event_loop_policy(asyncio.WindowsProactorEventLoopPolicy())Textual 的这套 App → Screen → Widget 层级,其实和 Web 开发的思维非常接近。App 像是浏览器,Screen 像是页面,Widget 像是 DOM 元素。如果你有前端背景,上手会特别快。
ComposeResult 的生成器设计,则让界面描述变成了纯粹的 Python——没有模板语言,没有 DSL,就是函数和 yield。这种设计降低了心智负担,但也意味着你需要对 Python 的生成器机制有基本了解,否则某些行为会让你困惑。
从"能跑"到"搞明白",这中间的距离,往往就是把这几个核心概念真正理解透彻的距离。
欢迎在评论区聊聊你在使用 Textual 过程中遇到的问题,或者你觉得这套结构设计还有哪些值得深挖的地方。