mirror of
https://github.com/LCTT/TranslateProject.git
synced 2025-01-10 22:21:11 +08:00
949e20f629
感谢诸位一直以来的无私奉献,愿倒下一个 Linux 中国,站起来无数开源新苗。
846 lines
34 KiB
Markdown
846 lines
34 KiB
Markdown
[#]: subject: "Crash Course On Using Textual"
|
||
[#]: via: "https://fedoramagazine.org/crash-course-on-using-textual/"
|
||
[#]: author: "Jose Nunez https://fedoramagazine.org/author/josevnz/"
|
||
[#]: collector: "lujun9972/lctt-scripts-1700446145"
|
||
[#]: translator: "ChatGPT"
|
||
[#]: reviewer: "wxy"
|
||
[#]: publisher: "wxy"
|
||
[#]: url: "https://linux.cn/article-16557-1.html"
|
||
|
||
Textual:为 Python 增加漂亮的文本用户界面(TUI)
|
||
======
|
||
|
||
![][0]
|
||
|
||
### 快速入门使用 Textual
|
||
|
||
Python 在 Linux 上有像 [TkInter][4] 这样的优秀 GUI(图形用户界面)开发库,但如果你不能运行图形应用程序怎么办?
|
||
|
||
文本终端,并非只在 Linux 上有,而且 BSD 和其它的出色的类 Unix 操作系统上也有。如果你的代码是用 Python 编写的,你应该使用 [Textual][5] 来帮助你编写 TUI(文本用户界面)。在这个快速介绍中,我将向你展示两个你可以用 Textual 做的示例,并且介绍它未来可能的应用方向。
|
||
|
||
所以 Textual 是什么?
|
||
|
||
> Textual 是一个为 Python 构建的快速应用程序开发框架,由 Textualize.io 构建。它可以让你用简单的 Python API 构建复杂的用户界面,并运行在终端或网络浏览器上!
|
||
|
||
#### 你需要的跟进这个教程的工具
|
||
|
||
你需要有以下条件:
|
||
|
||
1. 具备基础的编程经验,最好熟练使用 Python。
|
||
2. 理解基础的面向对象概念,比如类和继承。
|
||
3. 一台安装了 Linux 与 Python 3.9+ 的机器
|
||
4. 一款好的编辑器(Vim 或者 PyCharm 是不错的选择)
|
||
|
||
我尽可能简单化代码,以便你能轻松理解。此外,我强烈建议你下载代码,或至少按照接下来的说明安装相关程序。
|
||
|
||
#### 安装步骤
|
||
|
||
首先创建一个虚拟环境:
|
||
|
||
```
|
||
python3 -m venv ~/virtualenv/Textualize
|
||
```
|
||
|
||
现在,你可以克隆 Git 仓库并创建一个可以编辑的发布版本:
|
||
|
||
```
|
||
. ~/virtualenv/Textualize/bin/activate
|
||
pip install --upgrade pip
|
||
pip install --upgrade wheel
|
||
pip install --upgrade build
|
||
pip install --editable .
|
||
```
|
||
|
||
或者直接从 Pypi.org 安装:
|
||
|
||
```
|
||
. ~/virtualenv/Textualize/bin/activate
|
||
pip install --upgrade KodegeekTextualize
|
||
```
|
||
|
||
### 我们的首个程序:日志浏览器
|
||
|
||
![][6]
|
||
|
||
这个 _日志浏览器_ 就是一款简单的应用,能执行用户 _[PATH][7]_ 路径上的一系列 UNIX 命令,并在任务执行完毕后捕获输出。
|
||
|
||
以下是该应用的代码:
|
||
|
||
```
|
||
import shutil
|
||
from textual import on
|
||
from textual.app import ComposeResult, App
|
||
from textual.widgets import Footer, Header, Button, SelectionList
|
||
from textual.widgets.selection_list import Selection
|
||
from textual.screen import ModalScreen
|
||
# Operating system commands are hardcoded
|
||
OS_COMMANDS = {
|
||
"LSHW": ["lshw", "-json", "-sanitize", "-notime", "-quiet"],
|
||
"LSCPU": ["lscpu", "--all", "--extended", "--json"],
|
||
"LSMEM": ["lsmem", "--json", "--all", "--output-all"],
|
||
"NUMASTAT": ["numastat", "-z"]
|
||
}
|
||
|
||
class LogScreen(ModalScreen):
|
||
# ... Code of the full separate screen omitted, will be explained next
|
||
def __init__(self, name = None, ident = None, classes = None, selections = None):
|
||
super().__init__(name, ident, classes)
|
||
pass
|
||
|
||
class OsApp(App):
|
||
BINDINGS = [
|
||
("q", "quit_app", "Quit"),
|
||
]
|
||
CSS_PATH = "os_app.tcss"
|
||
ENABLE_COMMAND_PALETTE = False # Do not need the command palette
|
||
|
||
def action_quit_app(self):
|
||
self.exit(0)
|
||
|
||
def compose(self) -> ComposeResult:
|
||
# Create a list of commands, valid commands are assumed to be on the PATH variable.
|
||
selections = [Selection(name.title(), ' '.join(cmd), True) for name, cmd in OS_COMMANDS.items() if shutil.which(cmd[0].strip())]
|
||
yield Header(show_clock=False)
|
||
sel_list = SelectionList(*selections, id='cmds')
|
||
sel_list.tooltip = "Select one more more command to execute"
|
||
yield sel_list
|
||
yield Button(f"Execute {len(selections)} commands", id="exec", variant="primary")
|
||
yield Footer()
|
||
|
||
@on(SelectionList.SelectedChanged)
|
||
def on_selection(self, event: SelectionList.SelectedChanged) -> None:
|
||
button = self.query_one("#exec", Button)
|
||
selections = len(event.selection_list.selected)
|
||
if selections:
|
||
button.disabled = False
|
||
else:
|
||
button.disabled = True
|
||
button.label = f"Execute {selections} commands"
|
||
|
||
@on(Button.Pressed)
|
||
def on_button_click(self):
|
||
selection_list = self.query_one('#cmds', SelectionList)
|
||
selections = selection_list.selected
|
||
log_screen = LogScreen(selections=selections)
|
||
self.push_screen(log_screen)
|
||
|
||
def main():
|
||
app = OsApp()
|
||
app.title = f"Output of multiple well known UNIX commands".title()
|
||
app.sub_title = f"{len(OS_COMMANDS)} commands available"
|
||
app.run()
|
||
|
||
if __name__ == "__main__":
|
||
main()
|
||
```
|
||
|
||
现在我们逐条梳理一下程序的代码:
|
||
|
||
1. 每个应用都扩展自 `App` 类。其中最重要的有 `compose` 与 `mount` 等方法。但在当前应用中,我们只实现了 [compose][8]。
|
||
2. 在 `compose` 方法中,你会返回一系列 <ruby>[组件][9]<rt>Widget</rt></ruby>,并按顺序添加到主屏幕中。每一个组件都有定制自身外观的选项。
|
||
3. 你可以设定单字母的 <ruby>[绑定][10]<rt>binding</rt></ruby>,比如此处我们设定了 `q` 键来退出应用(参见 `action_quit_app` 函数和 `BINDINGS` 列表)。
|
||
4. 利用 `SelectionList` 组件,我们展示了待运行的命令列表。然后,你可以通过 `@on(SelectionList.SelectedChanged)` 注解以及 `on_selection` 方法告知应用获取所选的内容。
|
||
5. 对于无选定元素的应对很重要,我们会根据运行的命令数量来决定是否禁用 “exec” 按钮。
|
||
6. 我们使用类似的监听器( `@on(Button.Pressed)` )来执行命令。我们做的就是将我们的选择送到一个新的屏幕,该屏幕会负责执行命令并收集结果。
|
||
|
||
你注意到 `CSS_PATH = "os_app.tcss"` 这个变量了吗?Textual 允许你使用 CSS 来控制单个或多个组件的外观(色彩、位置、尺寸):
|
||
|
||
```
|
||
Screen {
|
||
layout: vertical;
|
||
}
|
||
|
||
Header {
|
||
dock: top;
|
||
}
|
||
|
||
Footer {
|
||
dock: bottom;
|
||
}
|
||
|
||
SelectionList {
|
||
padding: 1;
|
||
border: solid $accent;
|
||
width: 1fr;
|
||
height: 80%;
|
||
}
|
||
|
||
Button {
|
||
width: 1fr
|
||
}
|
||
```
|
||
|
||
引自 Textual 官方网站:
|
||
|
||
> Textual 中使用的 CSS 是互联网上常见 CSS 的简化版本,容易上手。
|
||
|
||
_这真是太棒了_,只需要用一哥独立的 [样式表][12],就可以轻松调整应用的样式。
|
||
|
||
好,我们现在来看看如何在新屏幕上展示结果。
|
||
|
||
#### 在新屏幕上展示结果
|
||
|
||
![][13]
|
||
|
||
以下是在新屏幕上处理输出的代码:
|
||
|
||
```
|
||
import asyncio
|
||
from typing import List
|
||
from textual import on, work
|
||
from textual.reactive import reactive
|
||
from textual.screen import ModalScreen
|
||
from textual.widgets import Button, Label, Log
|
||
from textual.worker import Worker
|
||
from textual.app import ComposeResult
|
||
|
||
class LogScreen(ModalScreen):
|
||
count = reactive(0)
|
||
MAX_LINES = 10_000
|
||
ENABLE_COMMAND_PALETTE = False
|
||
CSS_PATH = "log_screen.tcss"
|
||
|
||
def __init__(
|
||
self,
|
||
name: str | None = None,
|
||
ident: str | None = None,
|
||
classes: str | None = None,
|
||
selections: List = None
|
||
):
|
||
super().__init__(name, ident, classes)
|
||
self.selections = selections
|
||
|
||
def compose(self) -> ComposeResult:
|
||
yield Label(f"Running {len(self.selections)} commands")
|
||
event_log = Log(
|
||
id='event_log',
|
||
max_lines=LogScreen.MAX_LINES,
|
||
highlight=True
|
||
)
|
||
event_log.loading = True
|
||
yield event_log
|
||
button = Button("Close", id="close", variant="success")
|
||
button.disabled = True
|
||
yield button
|
||
|
||
async def on_mount(self) -> None:
|
||
event_log = self.query_one('#event_log', Log)
|
||
event_log.loading = False
|
||
event_log.clear()
|
||
lst = '\n'.join(self.selections)
|
||
event_log.write(f"Preparing:\n{lst}")
|
||
event_log.write("\n")
|
||
|
||
for command in self.selections:
|
||
self.count += 1
|
||
self.run_process(cmd=command)
|
||
|
||
def on_worker_state_changed(self, event: Worker.StateChanged) -> None:
|
||
if self.count == 0:
|
||
button = self.query_one('#close', Button)
|
||
button.disabled = False
|
||
self.log(event)
|
||
|
||
@work(exclusive=False)
|
||
async def run_process(self, cmd: str) -> None:
|
||
event_log = self.query_one('#event_log', Log)
|
||
event_log.write_line(f"Running: {cmd}")
|
||
# Combine STDOUT and STDERR output
|
||
proc = await asyncio.create_subprocess_shell(
|
||
cmd,
|
||
stdout=asyncio.subprocess.PIPE,
|
||
stderr=asyncio.subprocess.STDOUT
|
||
)
|
||
stdout, _ = await proc.communicate()
|
||
if proc.returncode != 0:
|
||
raise ValueError(f"'{cmd}' finished with errors ({proc.returncode})")
|
||
stdout = stdout.decode(encoding='utf-8', errors='replace')
|
||
if stdout:
|
||
event_log.write(f'\nOutput of "{cmd}":\n')
|
||
event_log.write(stdout)
|
||
self.count -= 1
|
||
|
||
@on(Button.Pressed, "#close")
|
||
def on_button_pressed(self, _) -> None:
|
||
self.app.pop_screen()
|
||
|
||
```
|
||
|
||
你会注意到:
|
||
|
||
1. `LogScreen` 类扩展自 `ModalScreen` 类, 该类负责处理模态模式的屏幕。
|
||
2. 这个屏幕同样有一个 `compose` 方法,我们在这里添加了组件以展示 Unix 命令的内容。
|
||
3. 我们创建了一个叫做 `mount` 的新方法。一旦你用 `compose` 编排好组件,你就可以运行代码来获取数据,并再进一步定制它们的外观。
|
||
4. 我们使用 [asyncio][14] 运行命令,这样我们就能让 TUI 主工作线程在每个命令的结果出来时就及时更新内容。
|
||
5. 对于“工作线程”,请注意 `run_process` 方法上的 `@work(exclusive=False)` 注解,该方法用于运行命令并捕获 STDOUT + STDERR 输出。使用 [工作线程][15] 来管理并发并不复杂,尽管它们在手册中确实有专门的章节。这主要是因为运行的外部命令可能会执行很长时间。
|
||
6. 在 `run_process` 中,我们通过调用 `write` 以命令的输出内容来更新 `event_log`。
|
||
7. 最后,`on_button_pressed` 把我们带回到前一屏幕(从堆栈中移除屏幕)。
|
||
|
||
这个小应用向你展示了如何一份不到 200 行的代码来编写一个简单的前端,用来运行非 Python 代码。
|
||
|
||
现在我们来看一个更复杂的例子,这个例子用到了我们还未探索过的 Textual 的新特性。
|
||
|
||
### 示例二:展示赛事成绩的表格
|
||
|
||
![通过 Textual 创建的表格应用][16]
|
||
|
||
本示例将展示如何使用 `DataTable` 组件在表格中展示赛事成绩。你能通过这个应用实现:
|
||
|
||
* 通过列来排序表格
|
||
* 选择表格中的行,完整窗口展示赛事细节,我们将使用我们在日志浏览器中看到的 “推送屏幕” 技巧。
|
||
* 能进行表格搜索,查看选手详情,或执行其他操作如退出应用。
|
||
|
||
下面,我们来看看应用代码:
|
||
|
||
```
|
||
|
||
#!/usr/bin/env python
|
||
"""
|
||
Author: Jose Vicente Nunez
|
||
"""
|
||
from typing import Any, List
|
||
|
||
from rich.style import Style
|
||
from textual import on
|
||
from textual.app import ComposeResult, App
|
||
from textual.command import Provider
|
||
from textual.screen import ModalScreen, Screen
|
||
from textual.widgets import DataTable, Footer, Header
|
||
|
||
MY_DATA = [
|
||
("level", "name", "gender", "country", "age"),
|
||
("Green", "Wai", "M", "MYS", 22),
|
||
("Red", "Ryoji", "M", "JPN", 30),
|
||
("Purple", "Fabio", "M", "ITA", 99),
|
||
("Blue", "Manuela", "F", "VEN", 25)
|
||
]
|
||
|
||
class DetailScreen(ModalScreen):
|
||
ENABLE_COMMAND_PALETTE = False
|
||
CSS_PATH = "details_screen.tcss"
|
||
|
||
def __init__(
|
||
self,
|
||
name: str | None = None,
|
||
ident: str | None = None,
|
||
classes: str | None = None,
|
||
row: List[Any] | None = None,
|
||
):
|
||
super().__init__(name, ident, classes)
|
||
# Rest of screen code will be show later
|
||
|
||
class CustomCommand(Provider):
|
||
|
||
def __init__(self, screen: Screen[Any], match_style: Style | None = None):
|
||
super().__init__(screen, match_style)
|
||
self.table = None
|
||
# Rest of provider code will be show later
|
||
|
||
class CompetitorsApp(App):
|
||
BINDINGS = [
|
||
("q", "quit_app", "Quit"),
|
||
]
|
||
CSS_PATH = "competitors_app.tcss"
|
||
# Enable the command palette, to add our custom filter commands
|
||
ENABLE_COMMAND_PALETTE = True
|
||
# Add the default commands and the TablePopulateProvider to get a row directly by name
|
||
COMMANDS = App.COMMANDS | {CustomCommand}
|
||
|
||
def action_quit_app(self):
|
||
self.exit(0)
|
||
|
||
def compose(self) -> ComposeResult:
|
||
yield Header(show_clock=True)
|
||
|
||
table = DataTable(id=f'competitors_table')
|
||
table.cursor_type = 'row'
|
||
table.zebra_stripes = True
|
||
table.loading = True
|
||
yield table
|
||
yield Footer()
|
||
|
||
def on_mount(self) -> None:
|
||
table = self.get_widget_by_id(f'competitors_table', expect_type=DataTable)
|
||
columns = [x.title() for x in MY_DATA[0]]
|
||
table.add_columns(*columns)
|
||
table.add_rows(MY_DATA[1:])
|
||
table.loading = False
|
||
table.tooltip = "Select a row to get more details"
|
||
|
||
@on(DataTable.HeaderSelected)
|
||
def on_header_clicked(self, event: DataTable.HeaderSelected):
|
||
table = event.data_table
|
||
table.sort(event.column_key)
|
||
|
||
@on(DataTable.RowSelected)
|
||
def on_row_clicked(self, event: DataTable.RowSelected) -> None:
|
||
table = event.data_table
|
||
row = table.get_row(event.row_key)
|
||
runner_detail = DetailScreen(row=row)
|
||
self.show_detail(runner_detail)
|
||
|
||
def show_detail(self, detailScreen: DetailScreen):
|
||
self.push_screen(detailScreen)
|
||
|
||
def main():
|
||
app = CompetitorsApp()
|
||
app.title = f"Summary".title()
|
||
app.sub_title = f"{len(MY_DATA)} users"
|
||
app.run()
|
||
|
||
if __name__ == "__main__":
|
||
main()
|
||
```
|
||
|
||
有哪些部分值得我们关注呢?
|
||
|
||
1. `compose` 方法中添加了 [表头][17],“命令面板” 就位于此处,我们的表格([DataTable][18])也在这里。表格数据在 `mount` 方法中填充。
|
||
2. 我们设定了预期的绑定(`BINDINGS`),并指定了外部的 CSS 文件来设置样式(`CSS_PATH`)。
|
||
3. 默认情况下,我们无需任何设置便能使用 [命令面板][19],但在此我们显式启用了它(`ENABLE_COMMAND_PALETTE = True`)。
|
||
4. 我们的应用有一个自定义表格搜索功能。当用户输入一名选手的名字后,应用会显示可能的匹配项,用户可以点击匹配项查看该选手的详细信息。这需要告诉应用我们有一个定制的命令提供者(`COMMANDS = App.COMMANDS | {CustomCo_ mmand}`),即类 `CustomCommand(Provider)`。
|
||
5. 如果用户点击了表头,表格内容会按照该列进行排序。这是通过 `on_header_clicked` 方法实现的,该方法上具有 `@on(DataTable.HeaderSelected)` 注解。
|
||
6. 类似地,当选中表格中的一行时, `on_row_clicked` 方法会被调用,这得益于它拥有 `@on(DataTable.RowSelected)` 注解。当方法接受选中的行后,它会推送一个新的屏幕,显示选中行的详细信息(`class DetailScreen(ModalScreen)`)。
|
||
|
||
现在,我们详细地探讨一下如何显示选手的详细信息。
|
||
|
||
#### 利用多屏展示复杂视图
|
||
|
||
![][20]
|
||
|
||
当用户选择表格中的一行,`on_row_clicked` 方法就会被调用。它收到的是一个 `DataTable.RowSelected` 类型的事件。从这里我们会用选中的行的内容构建一个 `DetailScreen(ModalScreen)` 类的实例:
|
||
|
||
```
|
||
from typing import Any, List
|
||
from textual import on
|
||
from textual.app import ComposeResult
|
||
from textual.screen import ModalScreen
|
||
from textual.widgets import Button, MarkdownViewer
|
||
|
||
MY_DATA = [
|
||
("level", "name", "gender", "country", "age"),
|
||
("Green", "Wai", "M", "MYS", 22),
|
||
("Red", "Ryoji", "M", "JPN", 30),
|
||
("Purple", "Fabio", "M", "ITA", 99),
|
||
("Blue", "Manuela", "F", "VEN", 25)
|
||
]
|
||
|
||
class DetailScreen(ModalScreen):
|
||
ENABLE_COMMAND_PALETTE = False
|
||
CSS_PATH = "details_screen.tcss"
|
||
|
||
def __init__(
|
||
self,
|
||
name: str | None = None,
|
||
ident: str | None = None,
|
||
classes: str | None = None,
|
||
row: List[Any] | None = None,
|
||
):
|
||
super().__init__(name, ident, classes)
|
||
self.row: List[Any] = row
|
||
|
||
def compose(self) -> ComposeResult:
|
||
self.log.info(f"Details: {self.row}")
|
||
columns = MY_DATA[0]
|
||
row_markdown = "\n"
|
||
for i in range(0, len(columns)):
|
||
row_markdown += f"* **{columns[i].title()}:** {self.row[i]}\n"
|
||
yield MarkdownViewer(f"""## User details:
|
||
{row_markdown}
|
||
""")
|
||
button = Button("Close", variant="primary", id="close")
|
||
button.tooltip = "Go back to main screen"
|
||
yield button
|
||
|
||
@on(Button.Pressed, "#close")
|
||
def on_button_pressed(self, _) -> None:
|
||
self.app.pop_screen()
|
||
```
|
||
|
||
这个类的职责很直接:
|
||
|
||
1. `compose` 方法取得此行数据,并利用一个 [支持 Markdown 渲染的组件][21] 来展示内容。它的便利之处在于,它会为我们自动生成一个内容目录。
|
||
2. 当用户点击 “close” 后,方法 `on_button_pressed` 会引导应用回到原始屏幕。注解 `@on(Button.Pressed, "#close")` 用来接收按键被点击的事件。
|
||
|
||
最后,我们来详细讲解一下那个多功能的搜索栏(也叫做命令面板)。
|
||
|
||
#### 命令面板的搜索功能
|
||
|
||
![][22]
|
||
|
||
任何使用了表头的 Textual 应用都默认开启了 [命令面板][19]。有意思的是,你可以在 `CompetitorsApp` 类中添加自定义的命令,这会增加到默认命令集之上:
|
||
|
||
```
|
||
COMMANDS = App.COMMANDS | {CustomCommand}
|
||
```
|
||
|
||
然后是执行大部分任务的 `CustomCommand(Provider)` 类:
|
||
|
||
```
|
||
from functools import partial
|
||
from typing import Any, List
|
||
from rich.style import Style
|
||
from textual.command import Provider, Hit
|
||
from textual.screen import ModalScreen, Screen
|
||
from textual.widgets import DataTable
|
||
from textual.app import App
|
||
|
||
class CustomCommand(Provider):
|
||
|
||
def __init__(self, screen: Screen[Any], match_style: Style | None = None):
|
||
super().__init__(screen, match_style)
|
||
self.table = None
|
||
|
||
async def startup(self) -> None:
|
||
my_app = self.app
|
||
my_app.log.info(f"Loaded provider: CustomCommand")
|
||
self.table = my_app.query(DataTable).first()
|
||
|
||
async def search(self, query: str) -> Hit:
|
||
matcher = self.matcher(query)
|
||
|
||
my_app = self.screen.app
|
||
assert isinstance(my_app, CompetitorsApp)
|
||
|
||
my_app.log.info(f"Got query: {query}")
|
||
for row_key in self.table.rows:
|
||
row = self.table.get_row(row_key)
|
||
my_app.log.info(f"Searching {row}")
|
||
searchable = row[1]
|
||
score = matcher.match(searchable)
|
||
if score > 0:
|
||
runner_detail = DetailScreen(row=row)
|
||
yield Hit(
|
||
score,
|
||
matcher.highlight(f"{searchable}"),
|
||
partial(my_app.show_detail, runner_detail),
|
||
help=f"Show details about {searchable}"
|
||
)
|
||
|
||
class DetailScreen(ModalScreen):
|
||
def __init__(
|
||
self,
|
||
name: str | None = None,
|
||
ident: str | None = None,
|
||
classes: str | None = None,
|
||
row: List[Any] | None = None,
|
||
):
|
||
super().__init__(name, ident, classes)
|
||
# Code of this class explained on the previous section
|
||
|
||
class CompetitorsApp(App):
|
||
# Add the default commands and the TablePopulateProvider to get a row directly by name
|
||
COMMANDS = App.COMMANDS | {CustomCommand}
|
||
# Most of the code shown before, only displaying relevant code
|
||
def show_detail(self, detailScreen: DetailScreen):
|
||
self.push_screen(detailScreen)
|
||
```
|
||
|
||
1. 所有继承自 `Provider` 的类需实现 `search` 方法。在我们的例子中,我们还覆盖了 `startup` 方法,为了获取到我们应用表格(和其内容)的引用,这里使用到了 `App.query(DataTable).first()`。在类的生命周期中, `startup` 方法只会被调用一次。
|
||
2. 在 `search` 方法内,我们使用 `Provider.matcher` 对每个表格行的第二列(即名字)进行模糊搜索,以与用户在 TUI 中输入的词条进行比较。`matcher.match(searchable)` 返回一个整型的评分,大于零说明匹配成功。
|
||
3. 在 `search` 方法中,如果评分大于零,则返回一个 `Hit` 对象,以告知命令面板搜索查询是否成功。
|
||
4. 每个 `Hit` 都有以下信息:评分(用于在命令面板中对匹配项排序)、高亮显示的搜索词、一个可调用对象的引用(在我们的案例中,它是一个可以将表格行推送到新屏幕的函数)。
|
||
5. `Provider` 类的所有方法都是异步的。这使你能释放主线程,只有当响应准备好后才返回结果,这个过程不会冻结用户界面。
|
||
|
||
理解了这些信息,我们就可以现在展示赛手的详细信息了。
|
||
|
||
尽管这个架构的追踪功能相对直观,但是组件间传递的消息复杂性不可忽视。幸运的是,Textual 提供了有效的调试工具帮助我们理解背后的工作原理。
|
||
|
||
### Textual 应用的问题排查
|
||
|
||
对于 Python 的 Textual 应用进行 [调试][23] 相较而言更具挑战性。这是因为其中有一些操作可能是异步的,而在解决组件问题时设置断点可能颇为复杂。
|
||
|
||
根据具体情况,你可以使用一些工具。但首先,确保你已经安装了 textual 的开发工具:
|
||
|
||
```
|
||
pip install textual-dev==1.3.0
|
||
```
|
||
|
||
#### 确保你能捕捉到正确的按键
|
||
|
||
不确定 Textual 应用是否能捕捉到你的按键操作?运行 keys 应用:
|
||
|
||
```
|
||
textual keys
|
||
```
|
||
|
||
这让你能够验证一下你的按键组合,并确认在 Textual 中产生了哪些事件。
|
||
|
||
#### 图片比千言万语更直观
|
||
|
||
如果说你在布局设计上遇到了问题,想向他人展示你当前的困境,Textual 为你的运行应用提供了截图功能:
|
||
|
||
```
|
||
textual run --screenshot 5 ./kodegeek_textualize/log_scroller.py
|
||
```
|
||
|
||
就像你所看到的,我是通过这种方式为这篇教程创建了插图。
|
||
|
||
#### 捕获事件并输出定制消息
|
||
|
||
在 Textual 中,每一个应用实例都有一个日志记录器,可以使用如下方式访问:
|
||
|
||
```
|
||
my_app = self.screen.app
|
||
my_app.log.info(f"Loaded provider: CustomCommand")
|
||
```
|
||
|
||
想要查看这些消息,首先需要开启一个控制台:
|
||
|
||
```
|
||
. ~/virtualenv/Textualize/bin/activate
|
||
textual console
|
||
```
|
||
|
||
然后在另一个终端运行你的应用程序:
|
||
|
||
```
|
||
. ~/virtualenv/Textualize/bin/activate
|
||
textual run --dev ./kodegeek_textualize/log_scroller.py
|
||
```
|
||
|
||
在运行控制台的终端中,你可以看到实时的事件和消息输出:
|
||
|
||
```
|
||
|
||
▌Textual Development Console v0.46.0
|
||
▌Run a Textual app with textual run --dev my_app.py to connect.
|
||
▌Press Ctrl+C to quit.
|
||
─────────────────────────────────────────────────────────────────────────────── Client '127.0.0.1' connected ────────────────────────────────────────────────────────────────────────────────
|
||
[20:29:43] SYSTEM app.py:2188
|
||
Connected to devtools ( ws://127.0.0.1:8081 )
|
||
[20:29:43] SYSTEM app.py:2192
|
||
---
|
||
[20:29:43] SYSTEM app.py:2194
|
||
driver=<class 'textual.drivers.linux_driver.LinuxDriver'>
|
||
[20:29:43] SYSTEM app.py:2195
|
||
loop=<_UnixSelectorEventLoop running=True closed=False debug=False>
|
||
[20:29:43] SYSTEM app.py:2196
|
||
features=frozenset({'debug', 'devtools'})
|
||
[20:29:43] SYSTEM app.py:2228
|
||
STARTED FileMonitor({PosixPath('/home/josevnz/TextualizeTutorial/docs/Textualize/kodegeek_textualize/os_app.tcss')})
|
||
[20:29:43] EVENT
|
||
```
|
||
|
||
此外,以开发者模式运行的另一大好处是,如果你更改了 CSS,应用会尝试重新渲染,而无需重启程序。
|
||
|
||
### 如何编写单元测试
|
||
|
||
为你全新开发的 Textual 应用编写 [单元测试][24],应该如何操作呢?
|
||
|
||
在 [官方文档][25] 展示了几种用于测试我们应用的方式。
|
||
|
||
我将采用 [unittest][24] 进行测试。为了处理异步例程,我们会需要特别的类 `unittest.IsolatedAsyncioTestCase` :
|
||
|
||
```
|
||
import unittest
|
||
from textual.widgets import Log, Button
|
||
from kodegeek_textualize.log_scroller import OsApp
|
||
|
||
class LogScrollerTestCase(unittest.IsolatedAsyncioTestCase):
|
||
async def test_log_scroller(self):
|
||
app = OsApp()
|
||
self.assertIsNotNone(app)
|
||
async with app.run_test() as pilot:
|
||
# Execute the default commands
|
||
await pilot.click(Button)
|
||
await pilot.pause()
|
||
event_log = app.screen.query(Log).first() # We pushed the screen, query nodes from there
|
||
self.assertTrue(event_log.lines)
|
||
await pilot.click("#close") # Close the new screen, pop the original one
|
||
await pilot.press("q") # Quit the app by pressing q
|
||
|
||
|
||
if __name__ == '__main__':
|
||
unittest.main()
|
||
```
|
||
|
||
现在让我们详细看看 `test_log_scroller` 方法中的操作步骤:
|
||
|
||
1. 通过 `app.run_test()` 获取一个 `Pilot` 实例。然后点击主按钮,运行包含默认指令的查询,随后等待所有事件的处理。
|
||
2. 从我们新推送出的屏幕中获取 `Log`,确保我们已获得几行返回的内容,即它并非空的。
|
||
3. 关闭新屏幕并重新呈现旧屏幕。
|
||
4. 最后,按下 `q`,退出应用。
|
||
|
||
可以测试表格吗?
|
||
|
||
```
|
||
import unittest
|
||
from textual.widgets import DataTable, MarkdownViewer
|
||
from kodegeek_textualize.table_with_detail_screen import CompetitorsApp
|
||
|
||
|
||
class TableWithDetailTestCase(unittest.IsolatedAsyncioTestCase):
|
||
async def test_app(self):
|
||
app = CompetitorsApp()
|
||
self.assertIsNotNone(app)
|
||
async with app.run_test() as pilot:
|
||
|
||
"""
|
||
Test the command palette
|
||
"""
|
||
await pilot.press("ctrl+\\")
|
||
for char in "manuela".split():
|
||
await pilot.press(char)
|
||
await pilot.press("enter")
|
||
markdown_viewer = app.screen.query(MarkdownViewer).first()
|
||
self.assertTrue(markdown_viewer.document)
|
||
await pilot.click("#close") # Close the new screen, pop the original one
|
||
|
||
"""
|
||
Test the table
|
||
"""
|
||
table = app.screen.query(DataTable).first()
|
||
coordinate = table.cursor_coordinate
|
||
self.assertTrue(table.is_valid_coordinate(coordinate))
|
||
await pilot.press("enter")
|
||
await pilot.pause()
|
||
markdown_viewer = app.screen.query(MarkdownViewer).first()
|
||
self.assertTrue(markdown_viewer)
|
||
# Quit the app by pressing q
|
||
await pilot.press("q")
|
||
|
||
|
||
if __name__ == '__main__':
|
||
unittest.main()
|
||
```
|
||
|
||
如果你运行所有的测试,你将看到如下类似的输出:
|
||
|
||
```
|
||
(Textualize) [josevnz@dmaf5 Textualize]$ python -m unittest tests/*.py
|
||
..
|
||
----------------------------------------------------------------------
|
||
Ran 2 tests in 2.065s
|
||
|
||
OK
|
||
```
|
||
|
||
这是测试 TUI 的一个不错的方式,对吧?
|
||
|
||
### 打包 Textual 应用
|
||
|
||
打包 Textual 应用与打包常规 Python 应用并没有太大区别。你需要记住,需要包含那些控制应用外观的 CSS 文件:
|
||
|
||
```
|
||
. ~/virtualenv/Textualize/bin/activate
|
||
python -m build
|
||
pip install dist/KodegeekTextualize-*-py3-none-any.whl
|
||
```
|
||
|
||
这个教程的 [pyproject.toml][26] 文件是一个打包应用的良好起点,告诉你需要做什么。
|
||
|
||
```
|
||
[build-system]
|
||
requires = [
|
||
"setuptools >= 67.8.0",
|
||
"wheel>=0.42.0",
|
||
"build>=1.0.3",
|
||
"twine>=4.0.2",
|
||
"textual-dev>=1.2.1"
|
||
]
|
||
build-backend = "setuptools.build_meta"
|
||
|
||
[project]
|
||
name = "KodegeekTextualize"
|
||
version = "0.0.3"
|
||
authors = [
|
||
{name = "Jose Vicente Nunez", email = "kodegeek.com@protonmail.com"},
|
||
]
|
||
description = "Collection of scripts that show how to use several features of textualize"
|
||
readme = "README.md"
|
||
requires-python = ">=3.9"
|
||
keywords = ["running", "race"]
|
||
classifiers = [
|
||
"Environment :: Console",
|
||
"Development Status :: 4 - Beta",
|
||
"Programming Language :: Python :: 3",
|
||
"Intended Audience :: End Users/Desktop",
|
||
"Topic :: Utilities"
|
||
]
|
||
dynamic = ["dependencies"]
|
||
|
||
[project.scripts]
|
||
log_scroller = "kodegeek_textualize.log_scroller:main"
|
||
table_detail = "kodegeek_textualize.table_with_detail_screen:main"
|
||
|
||
[tool.setuptools]
|
||
include-package-data = true
|
||
|
||
[tool.setuptools.packages.find]
|
||
where = ["."]
|
||
exclude = ["test*"]
|
||
|
||
[tool.setuptools.package-data]
|
||
kodegeek_textualize = ["*.txt", "*.tcss", "*.csv"]
|
||
img = ["*.svg"]
|
||
|
||
[tool.setuptools.dynamic]
|
||
dependencies = {file = ["requirements.txt"]}
|
||
```
|
||
|
||
### 未来计划
|
||
|
||
这个简短的教程只覆盖了 Textual 的部分方面。还有很多需要探索和学习的内容:
|
||
|
||
* 强烈建议你查看 [官方教程][27]。有大量的示例和指向参考 [API][28] 的链接。
|
||
* Textual 可以使用来自 [Rich][29] 项目的组件,这个项目是一切的起源。我认为其中一些甚至可能所有这些组件在某些时候都会合并到 Textual 中。Textual 框架对于使用高级 API 的复杂应用更能胜任,但 Rich 也有很多漂亮的功能。
|
||
* 创建你自己的组件!同样,在设计 TUI 时,[拿一张纸,画出你希望这些组件如何布局的][30],这会为你后期省去很多时间和麻烦。
|
||
* 调试 Python 应用可能会有点复杂。有时你可能需要 [混合使用不同的工具][31] 来找出应用的问题所在。
|
||
* 异步 IO 是一个复杂的话题,你应该 [阅读开发者文档][32] 来了解更多可能的选择。
|
||
* Textual 被其他项目所使用。其中一个非常易于使用的项目是 [Trogon][33]。[它会让你的 CLI 可以自我发现][34]。
|
||
* [Textual-web][35] 是个很有前景的项目,能让你在浏览器上运行 Textual 应用。尽管它不如 Textual 成熟,但它的进化速度非常快。
|
||
* 最后,[查看这些外部项目][36]。在项目组合中有许多有用的开源应用。
|
||
|
||
*(题图:DA/f11b0eb0-8e16-4cbe-986a-0fe978f6732a)*
|
||
|
||
--------------------------------------------------------------------------------
|
||
|
||
via: https://fedoramagazine.org/crash-course-on-using-textual/
|
||
|
||
作者:[Jose Nunez][a]
|
||
选题:[lujun9972][b]
|
||
译者:[ChatGPT](https://linux.cn/lctt/ChatGPT)
|
||
校对:[wxy](https://github.com/wxy)
|
||
|
||
本文由 [LCTT](https://github.com/LCTT/TranslateProject) 原创编译,[Linux中国](https://linux.cn/) 荣誉推出
|
||
|
||
[a]: https://fedoramagazine.org/author/josevnz/
|
||
[b]: https://github.com/lujun9972
|
||
[1]: https://fedoramagazine.org/wp-content/uploads/2024/01/textual_course-816x345.jpg
|
||
[2]: https://unsplash.com/@bernardhermant?utm_content=creditCopyText&utm_medium=referral&utm_source=unsplash
|
||
[3]: https://unsplash.com/photos/gray-and-clear-digital-wallpaper-VUBAE-Bmugk?utm_content=creditCopyText&utm_medium=referral&utm_source=unsplash
|
||
[4]: https://docs.python.org/3/library/tkinter.html
|
||
[5]: https://textual.textualize.io/
|
||
[6]: https://tutorials.kodegeek.com/Textualize/img/output_of_multiple_well_known_unix_commands_2023-12-28T19_13_32_605621.svg
|
||
[7]: https://manpages.org/bash
|
||
[8]: https://textual.textualize.io/tutorial/#composing-the-widgets
|
||
[9]: https://textual.textualize.io/widget_gallery/
|
||
[10]: https://textual.textualize.io/api/binding/
|
||
[11]: https://textual.textualize.io/guide/screens/
|
||
[12]: https://textual.textualize.io/guide/styles/
|
||
[13]: https://tutorials.kodegeek.com/Textualize/img/output_of_multiple_well_known_unix_commands_2023-12-28T19_13_40_503695.svg
|
||
[14]: https://docs.python.org/3/library/asyncio.html
|
||
[15]: https://textual.textualize.io/guide/workers/
|
||
[16]: https://tutorials.kodegeek.com/Textualize/img/summary_2023-12-28T19_05_20_213933.svg
|
||
[17]: https://textual.textualize.io/widgets/header/
|
||
[18]: https://textual.textualize.io/widgets/data_table/#guide
|
||
[19]: https://textual.textualize.io/guide/command_palette/
|
||
[20]: https://tutorials.kodegeek.com/Textualize/img/summary_2023-12-28T19_05_44_404837.svg
|
||
[21]: https://textual.textualize.io/widget_gallery/#markdownviewer
|
||
[22]: https://tutorials.kodegeek.com/Textualize/img/summary_2023-12-28T19_05_55_822030.svg
|
||
[23]: https://github.com/josevnz/tutorials/blob/main/docs/PythonDebugger/README.md
|
||
[24]: https://docs.python.org/3/library/unittest.html
|
||
[25]: https://textual.textualize.io/guide/testing/
|
||
[26]: https://tutorials.kodegeek.com/Textualize/pyproject.toml
|
||
[27]: https://textual.textualize.io/tutorial/
|
||
[28]: https://textual.textualize.io/api/
|
||
[29]: https://github.com/Textualize/rich
|
||
[30]: https://textual.textualize.io/how-to/design-a-layout/
|
||
[31]: https://github.com/josevnz/DebuggingApplications/blob/main/StracePythonWireshark/README.md
|
||
[32]: https://docs.python.org/3/library/asyncio-dev.html
|
||
[33]: https://github.com/Textualize/trogon
|
||
[34]: https://github.com/josevnz/CLIWithClickAndTrogon/blob/main/README.md
|
||
[35]: https://github.com/Textualize/textual-web
|
||
[36]: https://www.textualize.io/projects/
|
||
[37]: https://github.com/josevnz
|
||
[0]: https://img.linux.net.cn/data/attachment/album/202401/15/110359fptrxpqffztniz45.jpg |