Skip to content

nicegui的中文入门教程。This is a chinese guide of nicegui.

License

Notifications You must be signed in to change notification settings

python-and-novella/chinese_guide_of_nicegui_for_beginner

Repository files navigation

nicegui的中文入门教程

[TOC]

0 前言

对于python语言的使用者来说,nicegui是一款优秀的WebUI、GUI框架,只需学习一定量的前端知识,就能使用nicegui快速搭建出美观的UI界面。但是,由于官方作者不提供系统性的入门、中文教程,很多中文初学者望而却步。于是,本教程应运而生。

本教程旨在用中文提供官方文档没有的系统性入门教程,并将部分社区讨论问题汉化、简化,方便中文学习者尽快上手并解决常见难题。虽然教程的名字叫入门教程,但本教程并没有停留在翻译官网文档的阶段,能够解决常见问题的高阶技巧也有。对于官方提供的各个控件的详细API,本教程并不会照本宣科,而是在提供思路之后,由读者自行查阅。正所谓“授人以鱼不如授人以渔”,掌握方法比掌握结果更有效。

->教程全文详见仓库

1 环境准备

本章主要介绍运行和开发nicegui程序的环境准备,包括虚拟环境的建立、开发工具的选择、如何自托管文档。

1.1 运行环境

为了保证最佳开发体验,所有的环境准备优先使用Windows系统,使用Linux、Mac的话,请自己根据系统变通。

首先,需要准备python解释器和pip。其中,Python解释器是运行后续代码、工具的基础,只需到官网安装最新稳定版(当前为3.12.*)即可;pip是Python官方的包管理工具,安装解释器时务必勾选此选项,如果没有勾选或者想要后续单独安装pip,可以到pypi官方查看安装方法,这里不在赘述。

环境管理工具有pdm和poetry,使用以下命令全部安装:

pip install pdm poetry

环境管理工具是快捷管理python运行环境的工具,可以创建出独立的python运行环境,各个运行环境内安装的软件包不会干扰其他环境,也不会影响到默认的python环境。

pdm是一款国人创建的环境管理工具,语法简单,操作方便,因此教程采用此工具。

poetry也是一款环境管理工具,是nicegui官方仓库采用的,可以基于官方源代码自己编译安装包,也用于后续自托管文档,因此需要安装。

但是,poetry的检查依赖速度比pdm慢太多,故这里采用pdm作为学习工具,如果读者有能力,可以只使用poetry。

1.1.1 基础环境初始化

先在纯英文、无空格、无特殊字符的路径下创建纯英文、无空格、无特殊字符的空白文件夹,进入该文件夹后,右键,在此处打开终端或者命令行,运行以下命令:

pdm init

会看到以下输出:

PS E:\PSF\git\chinese_guide_of_nicegui_for_beginner> pdm init
Creating a pyproject.toml for PDM...
Please enter the Python interpreter to use
 0. [email protected] (D:\Programs\Python\Python312\python.EXE)
 1. [email protected] (D:\mingw64\bin\python3.11.exe)
 2. [email protected] (D:\mingw64\bin\python3.exe)
 3. [email protected] (D:\mingw64\bin\python.exe)
 4. [email protected] (C:\Users\peter\AppData\Local\pdm\pdm\python\c[email protected]\python.exe)
Please select (0): 0
Virtualenv is created successfully at E:\PSF\git\chinese_guide_of_nicegui_for_beginner\.venv
Project name (chinese_guide_of_nicegui_for_beginner):
Project version (0.1.0):
Do you want to build this project for distribution(such as wheel)?
If yes, it will be installed by default when running `pdm install`. [y/n] (n):
License(SPDX name) (MIT):
Author name (): Peter
Author email (): peter@linux
Python requires('*' to allow any) (==3.12.*):
Project is initialized successfully
PS E:\PSF\git\chinese_guide_of_nicegui_for_beginner>

过程为交互式,需要自己输入之后回车才能继续,不输入直接回车则采用默认。

Please select (0):为选择python版本,pdm会自动识别当前电脑安装的所有python解释器,部分工具(如gcc)也会自带python解释器,需要正确选择自己安装的、可以直接运行pip命令的python解释器,这一步根据实际情况选择,一般默认第一个,可以直接回车。

Project name (chinese_guide_of_nicegui_for_beginner):为设置项目名称,通常为当前文件夹名字,可以自己输入来修改。这个不会修改当前文件夹名字,只会影响项目描述文件中的项目名称和后续创建的源代码文件夹名称。这里可以直接回车。

Project version (0.1.0):为设置当前项目的版本号,该版本号符合语义化规则,不懂或者不想了解的可以直接回车。

If yes, it will be installed by default when running pdm install. [y/n] (n):这一步是问你要不要将项目构建成分发包(可以用pip安装的),如果选择y,使用pdm install就会默认安装项目。这里选择不创建分发包,所以直接回车。

License(SPDX name) (MIT):Author name (): Author email ():分别是许可协议、作者名字、作者邮箱地址,了解、知道的可以修改,不清楚或者不想写的可以直接回车。

Python requires('*' to allow any) (==3.12.*):为python的版本要求,如果后续要用低版本或者高版本python运行,这里需要修改,否则此项目会限制python的版本。这个后续可以自己了解,这里直接回车即可。

至此,基础环境已经准备完毕,可以得到以下目录结构:

chinese_guide_of_nicegui_for_beginner
├─.venv
│  ├─Lib
│  │  └─site-packages
│  └─Scripts
├─src
│  └─chinese_guide_of_nicegui_for_beginner
└─tests

./src/chinese_guide_of_nicegui_for_beginner/下存放项目的源代码,后续的代码操作(创建修改)均在此目录。此外,项目根目录下有pyproject.toml(项目描述文件)和README.md(自述文件),本教程不涉及手动修改。

1.1.2 nicegui运行环境

基础环境初始化完毕之后,项目还不能直接运行基于nicegui框架的代码,因为基础环境还没有安装nicegui。因此,需要在项目根目录下使用以下命令安装:

pdm add nicegui

安装过程取决于网速,耐心等待。

如果后续项目中需要使用其他库,可以使用pdm add 库对应的pip安装命令中的名字来添加到项目环境中。

对于调试使用nicegui的程序,通常在native mode下比较方便,因此,建议安装pywebview来增加native mode的支持,命令是:

pdm add pywebview

对于没有安装过Microsoft Edge WebView2或者版本较旧的Windows系统,建议访问 Microsoft Edge WebView2下载 安装最新版本。

1.2 开发工具

VSCode或者PyCharm,其中VSCode比PyCharm轻量,但需要手动安装python插件,而PyCharm自带插件,操作简单。这里推荐使用VSCode,比较流畅,如果是使用PyCharm,后续操作根据VSCode对应即可。

对于VSCode,建议安装以下插件:

ms-python.python
ms-python.autopep8
ms-python.black-formatter

1.3 自托管文档【可选】

因为官网文档是可交互的,只有连接到官网才能操作。但是,部分地区的访问官网存在网络不佳的情况,如果需要自托管官网文档,可以遵循以下步骤。

首先要安装git,具体可以看Git - 安装 Git

安装完成后,额外找一个空白文件夹,打开终端,执行以下命令:

git clone https://github.com/zauberzeug/nicegui.git

没有git,可以打开以下链接:

https://github.com/zauberzeug/nicegui/archive/refs/heads/main.zip

下载压缩包之后解压,结果一样。对于GItHub难以访问的问题,可以自行搜索GitHub加速的方法。

使用以下命令进入nicegui的源代码文件夹,并执行安装操作:

cd nicegui
poetry install

等安装完成之后(poetry的install时间会比较长),执行以下命令,会开启一个服务器托管官网文档,并自动调用浏览器打开自托管的官网:

poetry run python .\main.py

关掉终端就可以关闭服务器,下次运行这条命令就可以直接开启自托管的官网文档,无需再次安装。

如果官方有源代码更新,后续将源代码解压覆盖之后,执行一次安装操作即可更新。

2 入门基础

本章主要介绍nicegui的基础知识,系统性了解nicegui的基本结构,在自学nicegui、查阅官网文档时有方向。

2.1 认识nicegui

nicegui 是一个开源的python库,可以搭建运行在浏览器的图形界面,也就是WebUI,也可以理解为和网站一样。nicegui的学习过程并不难,但不意味着你可以零基础入门。对于开发nicegui的开发者而言,python的基础是必须的;如果掌握python不常用语法和用法更好,后续在使用nicegui的过程中,可以很方便理解一些为了达成效果而使用的骚操作。尽管大部分python的学习者是零基础入门,除了转全栈、爬虫的开发者,后续学习过程中很少接触前端(HTML、CSS、JavaScript),但还是在使用nicegui之前,要有学习前端的心理准备。nicegui为了方便python工程师快速搭建图形界面,专注于python代码本身,做了不少前端细节的隐藏。不过,默认的样式不一定符合预期效果,为了达成效果,还是需要了解一些前端知识,才能让界面完全符合心意。

nicegui的底层使用了fastapi作为运行服务器,Quasar作为前端框架,支持tailwindcss的CSS语法,因为Quasar的内部使用VUE搭建,nicegui也不可避免地用到了VUE的语法。因此,对于想要用好nicegui的开发者,在后续使用nicegui的过程中,需要对上述提到的项目有一定了解。想要对后端部分定制、修改的,需要了解fastapi以及其基于的其他组件;想要让界面美化、随心的,需要掌握Quasar、tailwindcss基础;对于有能力和需求完全定制界面的,则需要掌握VUE基础。

相关链接:

nicegui官方文档:https://nicegui.io/documentation

fastapi官方文档:https://fastapi.tiangolo.com/zh/

Quasar官方文档:https://quasar.dev/docs

tailwindcss官方文档:https://tailwindcss.com/

VUE官方文档:https://cn.vuejs.org/guide/introduction

2.1.1 nicegui的Hello World!

如果你能看到这里,证明你有学好nicegui的基础和能力,并不畏惧上一节提到的那么多基础知识。那么,就用一段简单的Hello World!代码开启nicegui的入门之旅。

使用VSCode在项目文件夹下的src\chinese_guide_of_nicegui_for_beginner内,新建以py为后缀的python代码文件,复制以下代码到代码文件中,保存。

from nicegui import ui

ui.button('Say Hi',on_click=lambda :ui.notify('Hello World!'))

ui.run(native=True)

注意vscode的右下角,务必确保使用的是venv下的python解释器,如果不是,点击右下角3.12.4{'.venv'}对应位置,选择输入解释器路径-查找,选择.venv\Scripts\下的python.exe

hello_world_vscode

点击VSCode右上角的运行按钮(三角形),就可以看到一个窗口弹出,点击SAY HI按钮,就能看到窗口底部弹出的Hello World!

hello_world

Hello World!示例可以看到,基于nicegui的python程序寥寥三行代码,除去导入语句和必不可少的ui.run,主体只有一行代码,就能实现一个完整的交互过程,足见nicegui的强大、简洁。后续教程中很多例子都可以做到几行代码实现不错的效果,这也是nicegui大受欢迎的原因。

2.2 nicegui的基本结构

2.2.1 图形界面的基础概念

在正式学习nicegui之前,需要先对图形界面有个基础的理解。

一般来说,搭建图形界面理解三个概念:控件、布局、交互。

2.2.1.1 控件

控件是搭建图形界面的基本元素,就像是盖房子用的砖、门、窗等最小搭建单位。控件通常是图形界面框架提供、直接可用的。如果使用过程中发现基本元素不够,可以结合布局功能,用基本元素组合出新的控件。

2.2.1.2 布局

布局是排布控件的方式,就像是房屋的基本框架。用砖可以铺地,也可以垒墙,对于砖而言,墙或是地,就是布局。控件是横向排列还是竖向排列,是像网格一样一一对应,还是大控件套着小控件,都是由布局控制。大部分图形程序框架提供的布局类似,除了基本的几种布局之外,部分图形程序框架还提供额外的组合布局。

2.2.1.3 交互

交互是图形界面的重中之重,也是一个程序最难的部分。论难度的话,前面的控件和布局的学习只是对照文档,按图索骥,交互则需要身经百战,不断积累经验。

事件机制是目前大部分图形界面采用的交互反馈机制,也就是基于特定的事件触发,执行对应的函数。微软的winform中采用的消息机制,Qt的信号与槽,现代网页开发中的event事件监听,都可以理解为事件机制,只是对于winform和Qt而言,他们框架内的事件分别叫做消息和信号而已。

除了事件机制,美化也是交互的一部分。大部分现代图形界面框架。如Qt、WPF以及一系列基于网页开发的图形界面框架,支持CSS或者类似语法的美化功能,让图形界面变得更加美观,也让控件的动画效果更加丰富,这个极大提升了用户的使用体验。

此外,基于图形界面框架的特性,后端的处理逻辑以及数据的传递也是交互的一部分。在函数内,对于控件的控制,如何做到符合要求,毕竟有的框架、编程语言不支持没有定义或者声明函数就调用,而有的语言不支持声明函数。如果需要让控件显示的文本与另一个控件的文本一致,如何处理数据同步过程也需要技巧。

2.2.2 nicegui与基础概念的对应

对图形界面有基础的理解之后,下面就可以根据nicegui与基础概念的对应,进一步理解nicegui的设计理念。

Hello World!示例中,使用了导入语句from nicegui import ui导入了ui,顾名思义,ui就是用户界面,这也是nicegui调用控件的模块,也可以调用布局。具体的控件和布局用法可以参考下一节[nicegui中不得不学的功能](#2.3 nicegui中不得不学的功能),下一节将选取一些nicegui中常用、不好理解的功能重点讲解。

因为nicegui是基于Quasar这个Web框架做的python调用绑定,因此,nicegui的交互部分,很大程度与Web结合。在Web设计中,基于CSS实现的美化效果,基于JavaScript的event做的事件响应,都能在nicegui中看到。所以,如果用好nicegui,对Web三件套HTML、CSS、JavaScript的学习不可避免。此外,因为nicegui与Quasar这个Web框架的深度集成的关系,Quasar中的属性、slot、事件也需要掌握,才能让交互设计更加得心应手。

关于美化,下一节中的[外观美化](#2.3.7 外观美化)将会详细介绍,也可以查阅对应的官方文档。事件的学习,可以参阅下一节的[事件和执行](#2.3.8 事件和执行),也可以查阅对应的官方文档。

2.3 nicegui中不得不学的功能

以下是官网文档对于nicegui提供的功能做了大致的划分,本教程将会对每个部分中不好掌握、需要重点学习的控件、功能进行剖析:

  1. 文本控件:https://nicegui.io/documentation/section_text_elements

  2. 常用控件:https://nicegui.io/documentation/section_controls

  3. 多媒体控件:https://nicegui.io/documentation/section_audiovisual_elements

  4. 数据控件:https://nicegui.io/documentation/section_data_elements

  5. 属性绑定:https://nicegui.io/documentation/section_binding_properties

  6. 图形布局:https://nicegui.io/documentation/section_page_layout

  7. 外观美化:https://nicegui.io/documentation/section_styling_appearance

  8. 事件和执行:https://nicegui.io/documentation/section_action_events

  9. 网站页面:https://nicegui.io/documentation/section_pages_routing

  10. 部署与配置:https://nicegui.io/documentation/section_configuration_deployment

这部分对于官方内容的解析并不会完全覆盖,主要讲经常用到的参数、属性、方法,对于某些隐藏参数和不常用的属性方法,会在后面用到的时候详细介绍,这里最多提一嘴。

2.3.1 文本控件

文本类控件主要是一些静态展示可复制文本的控件,是构成网页显示效果的主要控件。

2.3.1.1 ui.label

文本标签,用法很简单,通过传入一个字符串类型的参数text,让网页显示字符串内的文字。注意,虽然参数支持多行文字,但是输出只能一行,需要原样输出多行文字的话,可以使用下面介绍的ui.html,将tag设置为pre

from nicegui import ui

ui.label(text='some label')

ui.run(native=True)
2.3.1.2 ui.link

超链接,一种点击之后跳转到指定地址的文本元素。可以传入texttargetnew_tab三个参数。代码如下:

from nicegui import ui

ui.link(text='NiceGUI on GitHub', target= 'https://github.com/zauberzeug/nicegui', new_tab=False)

ui.run(native=True)

text参数,字符串类型,表示超链接显示什么文字。

target参数,字符串类型、page functionui.element类型,表示超链接跳转到什么位置,这里只介绍字符串类型用法,其他类型可以自行探索或者后续需要用到的时候补充。字符串类型参数表示超链接跳转的url地址,可以使用协议开头的完整地址,也可以使用省略主机的绝对路径、相对路径。

new_tab参数,布尔类型,默认为False,表示要不要在新建标签页中打开超链接。

2.3.1.3 ui.element

通用元素,也是nicegui大部分界面控件的基类。很多控件都是通过继承这个类来调用自定义标签、JavaScript代码实现。通过继承实现自定义控件、修改默认风格属于高级用法,这里只说基本用法。

tag参数,字符串类型,默认为div,表示生成的元素用什么标签,实际使用时可以根据需要修改为其他HTML标签或者Quasar标签。代码如下:

from nicegui import ui

with ui.element('div').classes('p-2 bg-blue-100'):
    ui.label('inside a colored div')
    
ui.run(native=True)

move方法,将控件移动到指定控件之内,默认为defaultslot,也可以传递target_slot参数,指定slot。代码如下:

from nicegui import ui

with ui.card() as card:
    name = ui.input('Name', value='Paul')
    name.add_slot('append')
    icon = ui.icon('face')

ui.button('Move into input default slot', on_click=lambda: icon.move(name))
ui.button('Move into input append slot', on_click=lambda: icon.move(name, target_slot='append'))
ui.button('Move out of input', on_click=lambda: icon.move(card))

ui.run(native=True)

ui_element_move

2.3.1.4 ui.markdown和ui.html

ui.label类似,ui.markdownui.html,都可以用来展示文本,只是后两者支持markdown语法和HTML语法,因为markdown语法支持一部分HTML的标签,可以看到放在ui.markdown里的HTML标签也能被解析。以下是三种控件解析同一内容的代码:

from nicegui import ui

content = '''
This is **Markdown**.
This is <u>emphasized</u>
'''
ui.label(content)
ui.markdown(content)
ui.html(content)

ui.run(native=True)

此外,ui.html还支持传入字符串类型参数tag给基类ui.element,用于修改生成ui.html用的标签,比如:

from nicegui import ui

ui.html('This is <u>emphasized</u>.', tag='em')

ui.run(native=True)

2.3.2 常用控件

常用控件主要是一些支持点击、输入、拖动等交互功能的控件。

2.3.2.1 ui.button

按钮作是网页交互设计中最常见的基本元素,在移动互联网没有普及之前,使用鼠标点击为主要交互方式的时代,除了用于跳转网页的超链接,按钮就是网页中用的最多的可交互元素。在nicegui中,按钮控件可以传入位置参数text,关键字参数on_clickcoloricon

以下代码就是一个定义了基本交互的按钮,点击会弹出一个通知提示:

from nicegui import ui

ui.button('Click me!', color='green', icon='thumb_up', on_click=lambda: ui.notify('You clicked me!'))

ui.run(native=True)

text参数,字符串类型,表示显示在按钮上的文字,如果是英文的话,默认全部大写。该参数默认只支持字符串类型,但是整数和小数可以直接使用,其他类型需要先转换为字符串类型才能传入。

color参数,字符串类型或者None,表示按钮的颜色,支持传入字符串类型的颜色类(Quasar、 Tailwind、CSS的颜色名)或者None(即让按钮变成默认颜色)。

icon参数,字符串类型,表示按钮额外显示的图标,支持传入字符串类型的图标名,具体名字会在ui.icon中介绍,这里不做详细介绍。

on_click参数,可调用类型,表示点击按钮调用的函数,可以使用lambda表达式,也可以使用函数名。

如果觉得对按钮传入参数来自定义按钮内容的方法太死板,也可以使用以下语法,使用with来进入按钮的default slot,随意组合按钮内的内容:

from nicegui import ui

with ui.button(color='green', on_click=lambda: ui.notify('You clicked me!')):
    ui.icon('thumb_up')
    ui.label('Click me!')

ui.run(native=True)

button

对于on_click参数、后续会涉及到的on开头的on_*参数、on开头的on_*方法和on方法里的callback或者handler参数,均为可调用类型参数,既可以在创建控件时定义lambda表达式,也可以提前定义。对于复杂一点逻辑操作,应该定义函数而不是lambda表达式,比如:

from nicegui import ui

test = lambda :...
#如果要执行的操作比较多、复杂,应该定义函数
def test():
    pass
#在控件中传参可以直接使用test
ui.button('Click me!',on_click=test)

ui.run(native=True)

对于button等控件而言,除了支持通过传参创建响应动作,还支持调用对应的on_*方法(比如on_click)创建,这个方法极大提高了响应动作的灵活性,上面的例子就可以借助这个方法调整函数定义与按钮创建的先后顺序,代码如下:

from nicegui import ui

button = ui.button('Click me!')
def test():
    pass
button.on_click(test)

ui.run(native=True)
2.3.2.2 ui.input

输入框,大概是网页中仅次于按钮和超链接,用得最多的控件。HTML中输入框的变体很多,在nicegui中,输入框的参数也很多,基于输入框扩展的控件也多,使用输入框的逻辑设计、注意事项一样很多。不过,在这一节,针对输入框的学习并不会那么深入,只要能掌握常用的参数和基础的方法,那些疑难点会放到进阶和需要的时候细讲,以免一时不好理解而记混。

以下代码定义了一个名字输入框和密码输入框,并在输入名字的时候,自动弹出通知显示名字和密码:

from nicegui import ui

name = ui.input(label='Name:', value='Peter', on_change=lambda :ui.notify(f'{name.value=},{password.value=}'))
password = ui.input(label='Password:', value='123456', password=True, password_toggle_button=True)

ui.run(native=True)

input

label参数,字符串类型,直译的话是标签,表示显示在输入框上方的文本,但不是输入的文本,如果输入的内容是空的,点击输入的之前会显示在输入框内,点击之后会移动到输入框上方。

value参数,字符串类型,表示输入框内的内容,也就是输入框的值。对于后续介绍的以及其他支持交互输入的控件,都会有这么一个参数、属性、相关方法,来设置控件的值,这一点上,nicegui的设计倒是很统一。

on_change参数,可调用类型,表示输入框的值变化时执行的函数。

password参数,布尔类型,表示输入框是否设置为密码输入框,如果设置为True,输入的内容将不显示明文,转而显示统一的密码符号。

password_toggle_button参数,布尔类型,表示输入框内是否显示密码按钮,密码按钮可以切换输入框内的密码、明文状态。

输入框控件还有很多参数和方法,就留给读者自己探索了,后续如有相关案例,到时候再细讲。

2.3.2.3 ui.slider和ui.knob

ui.slider是滑动条,这个比较简单,参数也不多,按理说不用专门讲一下,自学即可。至于ui.knob——旋钮,这个的样式参数比较多,也有很大的自定义自由度,值得一讲。不过,这些都不是这里要专门放在一起介绍的原因,具体原因是什么,这里先卖个关子,后面再做解释。

先看一段代码:

from nicegui import ui

slider = ui.slider(min=0, max=100, step=1, value=50)
ui.knob(min=0, max=100, step=1, value=50,
        color='orange', center_color='blue', track_color='black',
        size='3.5em',
        show_value=True
        ).bind_value(slider, 'value')

ui.run(native=True)

ui_knob

可以看到,ui.knob的前四个参数和ui.slider的一样,都是浮点类型,分别代表最小值、最大值、每次调整的最小步长、当前值。

在ui.knob中,有colorcenter_colortrack_color三种可以传参修改的颜色,分别代表旋钮边缘的颜色、旋钮中间的颜色、旋钮边缘没有覆盖之前的颜色,支持传入字符串类型的颜色类(Quasar、 Tailwind、CSS的颜色名)、None(即让按钮变成默认颜色)或者"primary"(跟随主题颜色)。

size是旋钮的整体大小,字符串类型,采用CSS语法的大小表示方式。

show_value参数,布尔类型,是否在旋钮中间显示当前值。

2.3.3 多媒体控件

2.3.3.1 ui.image和ui.interactive_image

点开一个网页,最抓人眼球的是什么内容?没错,是图片。既然用nicegui设计网页,没有图片元素怎么行?在nicegui中,有两种显示图片的控件:ui.image和ui.interactive_image。前者可以简单理解为显示图片的简单标签,后者是基于前者扩展了很多交互功能的plus版本。两者的第一个参数都是source,支持字符串类型的本地图像路径、网络图像路径,或者base64编码的图像本身,这个没什么难点,这里不做细讲,接下来要重点讲的是ui.interactive_image的其他参数的用法,因为这个控件有时候比看似简单的ui.image更加好用趁手。

先看一段代码:

from nicegui import ui

src = 'https://picsum.photos/id/377/640/360'
img = ui.image(src)
img2 = ui.interactive_image(src)

ui.run(native=True)

ui_img

可以看到,同样的图片地址,都是不传入其他参数的情况下,即使可用空间大于图片大小,ui.interactive_image也不会随着页面大小而缩放图片,始终保持图片的原始大小,这个有别于ui.image的特性,可以在日后想要保持图片真实大小时使用。

除了source参数外,ui.interactive_image还有以下参数:

content参数,字符串类型,表示覆盖在图片之上的SVG内容,SVG的画布大小就是图片的大小。当然,不太理解SVG的话也没关系,后面用到会详细介绍,也可以专门找一下资料。这里可以简单理解SVG为一种用定义描述的几何图形,这种图形不会因为缩放变成马赛克,因为它是基于定义绘制的。

size参数,元组类型(宽度,高度),表示如果source没有设置的话,这就是默认图形的尺寸。这个对于绘制上面的SVG内容来说比较重要,因为这个尺寸就是画布的大小。对于想要交互创建SVG内容的操作,指定画布大小很重要。

on_mouse参数,可调用类型,表示触发鼠标事件之后要执行的操作,默认包含一个鼠标事件的参数,参数字典内的image_x和image_y的值是以像素为单位表示的鼠标交互位置。

events参数,字符串列表,表示JavaScript订阅的事件,默认订阅点击事件,即['click'],也可以自己指定要订阅的事件内容。

cross参数,字符串类型或者布尔类型,表示要不要显示十字线来指示鼠标位置,默认为False。如果为True或者表示颜色的字符串,就会显示指定颜色(即字符串表示的颜色)的十字线。

force_reload方法,强制重新载入图片,这个方法两者图片控件都有,对于某些时候网络不好造成图片加载失败、支持随机刷新图片的接口,这个方法还是很实用的。

对于图片控件,一样可以使用with嵌入其他内容。比如,下面的代码就嵌入了一个按钮,实现了点击图片和按钮有不同的通知内容:

from nicegui import ui

src = 'https://picsum.photos/id/377/640/360'
with ui.interactive_image(src,on_mouse=lambda :ui.notify('You clicked interactive_image.')):
    ui.button(on_click=lambda: ui.notify('You clicked button.'), icon='thumb_up')\
        .props('flat color=white').classes('absolute bottom-0 left-0 m-2')

ui.run(native=True)

ui_interactive_image

2.3.3.2 ui.icon和ui.avatar

在[ui.button](#2.3.2.1 ui.button)中挖了一个有关ui.icon的坑,现在,终于到了填坑的时候。先看一个简单的示例,了解一下ui.icon控件的用法:

from nicegui import ui

ui.icon(name='home', color='blue', size='xl')
ui.icon(name='o_home', color='blue', size='xl')
ui.icon(name='r_home', color='blue', size='xl')
ui.icon(name='sym_o_home', color='blue', size='xl')
ui.icon(name='sym_r_home', color='blue', size='xl')

ui.run(native=True)

ui_icon

ui.icon的参数不多,就三个:

name参数,字符串类型,表示图标字体中的图标名,ui.icon通过给定的图标名,从字体中加载图标,默认支持Material Icons图标字体。也可以自己添加其他图标字体,并结合对应图标字体的用法加载。这一部分需要根据具体情况灵活变通,这里不做详解。

color参数,字符串类型,表示图标的颜色。

size参数,字符串类型,表示图标的大小。

理解参数不难,但是,给出的示例代码并没有看上去那么简单。这个时候,聪明的读者已经发现了猫腻:几个图标看上去很像,但不完全一样。再仔细看的话就会发现,它们的后缀都是一样的,只有前缀不同。没错,这组图标本质上都是一个,只是名字的前缀有特定的风格含义,需要特别注意一下。

无前缀代表实心填充风格,"o_" 前缀表示轮廓线风格,"r_" 前缀表示圆角风格,"s_"前缀表示锐化风格,"sym_o_"前缀表示轮廓线符号,"sym_r_"前缀表示圆角符号,"sym_s_" 前缀表示锐化符号。

这个时候,有的读者就要头疼了,前缀就这么几个,还好记,可图标那么多,每个图标名字怎么记得住,有没有方便查询的网站?有,不过国内没法直接访问,那就是官网:

Material Icons查询网站(非中国大陆地区):https://fonts.google.com/icons?icon.set=Material+Icons

当然,国内也有很多渠道查询,为了避免广告嫌疑,这里就不具体指明了。

不过,只是提供一个404网站,教程就失去了意义,这里还有一个备用方法,那就是小工具:

Material Icons查询器:src\chinese_guide_of_nicegui_for_beginner\2_3_3_2_ui_icon.py

ui_icon_query

在输入框输入关键字,选定需要的名字,然后在下面查看各个风格的图标预览,最后确定要使用的图标名称。要注意的是,图标名是遍历字体库源代码下的文件夹名字,有的图标不支持全部风格,如果发现下面的预览存在图标异常或者各个风格没有区分,证明这个图标不支持全部风格,读者需酌情调整。

这一节到这里还没结束,还有一个控件要一并讲解。之所以放在一起讲,是因为这个控件主要参数和ui.icon一样,实际应用也不少,那就是头像控件——ui.avatar。

按照惯例先看代码:

from nicegui import ui

ui.avatar(icon='favorite_border', color='green-5', text_color='blue-10',
          size='60px', font_size='50px', square=False, rounded=True)
ui.avatar(icon='favorite_border', color='green-5', text_color='blue-10',
          size='60px', font_size='50px', square=True, rounded=False)
# 使用图片文件当做头像
url = 'https://nicegui.io/logo_square.png'
ui.avatar(f'img:{url}')
with ui.avatar():
    ui.image(url)

ui.run(native=True)

ui_avatar

从参数上看,icon就是ui.icon的name;颜色这里不太一样,colortext_color分别代表头像的背景颜色、图标颜色;size是头像大小,font_size是图标大小。此外,还有两个布尔类型的参数决定头像框的形状:square是方形, rounded是圆角方形,如果两个都是False,头像框就是圆形。

除了使用图标字体外,头像当然可以使用图片。一种是用以img:开头的图片地址,另一种是在defalut slot中加入图片控件。要注意的是,两种方法呈现的图片不一样,后者使用的是ui.image的样式,实际图片显示会有差别。

2.3.3.3 ui.audio和ui.video

ui.audio和ui.video从本质上讲是一样的,参数、方法基本一致,所以,只需学会其中一个的用法,另一个就是改个名字的事儿。

顾名思义,ui.audio和ui.video,分别代表着音频、视频控件,对于需要再网页播放音频、视频的需求,这两个控件再合适不过。

先看示例代码:

from nicegui import ui

v = ui.video(src='https://test-videos.co.uk/vids/bigbuckbunny/mp4/h264/360/Big_Buck_Bunny_360_10s_1MB.mp4',
             controls=True, autoplay=False, muted=False, loop=False)
ui.button('Play', on_click=v.play)
ui.button('Pause', on_click=v.pause)
ui.button('Jump to 0:05', on_click=lambda: v.seek(5))

ui.run(native=True)

ui_video

控件通过字符串类型参数src传入视频地址,四个布尔类型参数controlsautoplaymutedloop分别代表着是否显示内置工具栏、是否自动开始播放、是否静音、是否循环播放。

当然,除了上面代码里用按钮在外面控制播放暂停,还可以监听播放、暂停、播放结束的事件,执行特定的操作。

from nicegui import ui

v = ui.video(src='https://test-videos.co.uk/vids/bigbuckbunny/mp4/h264/360/Big_Buck_Bunny_360_10s_1MB.mp4',
             controls=True, autoplay=False, muted=False, loop=False)
v.on('play', lambda _: ui.notify('Started'))
v.on('pause', lambda _: ui.notify('Paused'))
v.on('ended', lambda _: ui.notify('Completed'))

ui.run(native=True)

2.3.4 图形布局

盖房子不是把砖简单地垒砌,做图形界面一样不是简单地添加控件,美观、简洁的布局除了让图形界面整体风格更加整齐,也让用户使用时更加方便、快捷,这才是图形界面的意义。在掌握了一定数量的常用控件之后,在继续深入学习其他控件之前,是时候了解一下NiceGUI支持的布局控件了。

在NiceGUI中,有一个概念非常重要,那就是上下文。前面代码中使用的with就是Python的上下文管理器关键字,而在NiceGUI中,进入某个控件的上下文,就意味着后续添加控件的操作都是在这个控件的上下文内,直观感受的话,就是后续添加的控件都在这个控件内部。

以代码为例:

from nicegui import ui

with ui.card():
    ui.label('Card content')
    ui.button('Add label1', on_click=lambda: ui.label('Click!'))
ui.button('Add label2', on_click=lambda: ui.label('Click!'))

ui.run(native=True)

Add label1按钮在ui.card内部,它添加的ui.label就在ui.card内。Add label2按钮在ui.card外,它添加的ui.label就在ui.card外。所以点击两个按钮的效果是这样的:

auto_context

2.3.4.1 ui.row、ui.column和ui.grid

在图形界面的布局设计中,最常用也是最基本的布局,就是行布局、列布局和网格布局,几乎所有的现代GUI框架都提供了这三种布局。说来神奇,哪怕不用框架提供的其他布局,只是这三种布局相互组合,就能构成了用户在程序里看到的各种控件排布,真可谓“三生万物”。

行(ui.row)布局,即所有控件排成一行。

列(ui.column)布局,即所有控件排成一列。

网格(ui.grid)布局,即所有控件按照给定的网格排布(几行几列),依次占据每个单元格,就和表格一样。

layout_sketch

在NiceGUI中,想要让指定的控件按照给定的布局排布,就要在布局的上下文内创建控件,因此,这三个常用布局的代码就要这样写:

from nicegui import ui

with ui.column().classes('border-4 border-black p-2'):
    ui.label('label 1')
    ui.label('label 2')
    ui.label('label 3')

with ui.row().classes('border-4 border-black p-2'):
    ui.label('label 1')
    ui.label('label 2')
    ui.label('label 3')

with ui.grid(columns=2,rows=2).classes('border-4 border-black p-2'):
    ui.label('label 1')
    ui.label('label 2')
    ui.label('label 3')

ui.run(native=True)

ui_layout

2.3.4.2 ui.space

在布局控件的时候,如果想要一些控件放在末尾,可按照布局控件的规则,内部的控件总是像排队一样一个挨着一个,似乎需要调整CSS才行。其实,还有一种不用CSS的方法,那就是使用ui.space。

ui.space可以默认填充浮动布局的可用空间,让布局不会因为默认的排列规则而显得末尾空空如也。

from nicegui import ui

with ui.row().classes('w-96 border-4 border-black p-2'):
    ui.label('label 1')
    ui.label('label 2')
    ui.space()
    ui.label('label 3')

with ui.column().classes('h-96 border-4 border-black p-2'):
    ui.label('label 1')
    ui.label('label 2')
    ui.space()
    ui.label('label 3')

ui.run(native=True)

ui_space

2.3.4.3 ui.separator

用ui.space可以填充空白,但是有读者担心,目的是达到了,能不能画条线在末尾的控件前,用来区分一下这个位置的控件与其他位置控件的区别?没问题,ui.separator就是用来干这个的,不管是水平还是垂直,都可以。代码如下:

from nicegui import ui

with ui.column().classes('border-4 border-black p-2'):
    ui.label('text above')
    ui.separator()
    ui.label('text below')

with ui.row().classes('border-4 border-black p-2'):
    ui.label('text above')
    ui.separator().props('vertical')
    ui.label('text below')

ui.run(native=True)

ui_separator

2.3.4.4 ui.menu和ui.context_menu

有用过图形程序的读者看到前面的内容,肯定联想到,ui.separator好像菜单里面用来分割退出功能与其他功能的分割符,NiceGUI有没有菜单?当然也有,普通的点击菜单ui.menu和右键菜单ui.context_menu。两个菜单的语法一致,只是触发方式不同。菜单的内部一般使用ui.menu_item当菜单项,一般可以当做按钮处理,因为它们的触发操作都是传给点击事件。

下面的代码给menu按钮创建点击菜单和右键菜单,可以分别用鼠标左击和右击查看弹出的菜单的异同,除了菜单内文字为了作出区分专门修改了之外,菜单的布局和样式一模一样,因此两种菜单可以很方便复用相同的内容:

from nicegui import ui

with ui.row().classes('w-full items-center'):
    result = ui.label().classes('mr-auto')
    with ui.button(text='menu', icon='menu'):
        with ui.context_menu() as menu1:
            ui.menu_item('Context Menu item 1', lambda: result.set_text('Selected item 1'))
            ui.menu_item('Context Menu item 2', lambda: result.set_text('Selected item 2'))
            ui.menu_item('Context Menu item 3 (keep open)',
                         lambda: result.set_text('Selected item 3'), auto_close=False)
            ui.separator()
            ui.menu_item('Close', menu1.close)
        with ui.menu() as menu2:
            ui.menu_item('Menu item 1', lambda: result.set_text('Selected item 1'))
            ui.menu_item('Menu item 2', lambda: result.set_text('Selected item 2'))
            ui.menu_item('Menu item 3 (keep open)',
                         lambda: result.set_text('Selected item 3'), auto_close=False)
            ui.separator()
            ui.menu_item('Close', menu2.close)

ui.run(native=True)

ui_menu

2.3.4.5 ui.tooltip

ui.tooltip可以给控件添加一种光标悬停之后弹出的提示。一般是在给定控件的上下文添加,也可以用给定控件的tooltip方法来添加,代码如下:

from nicegui import ui

with ui.button(icon='thumb_up'):
    ui.tooltip('I like this')

button = ui.button(icon='thumb_up')
button.tooltip('I like this')

ui.run(native=True)

但是要注意,如果想要美化ui.tooltip,只能给上下文添加的ui.tooltip美化,如果是用tooltip方法添加的,则不能美化。因为tooltip方法返回的是控件本身,而不是tooltip。可以看到,下面代码的tooltip方法后添加的美化,在按钮本身生效了,实际上ui_tooltip并没有生效。

from nicegui import ui

with ui.button(icon='thumb_up'):
    ui.tooltip('I like this').classes('bg-green')

button = ui.button(icon='thumb_up')
button.tooltip('I like this').classes('bg-green')

ui.run(native=True)

ui_tooltip

2.3.4.6 ui.notify

通知提示控件,在前面[ui.button](#2.3.2.1 ui.button)中已经用过,虽然简单传递一个字符串参数的用法很好用,但是到这一节结束都不说一下的话,实在是对不起读者。毕竟,这个控件的参数可不止看上去那么少,隐藏在其中的参数也很有用。

代码如下:

from nicegui import ui

ui.button('Notify',
          on_click=lambda: ui.notify(
              message='Hi!',
              position='bottom',
              close_button='OK',
              type=None,
              color='black-5',
              multi_line=False,
              progress=True,
              caption='5 minutes ago',
              timeout=5*1000,
              spinner= True
          ))

ui.run(native=True)

ui_notify

参数有点多,但不必害怕,习惯用法只需记住第一个参数,用起来也简单,其他参数只做了解。

message参数,字符串类型,信息文本,显示在通知中的主要内容。

position参数,字符串类型,通知出现的位置,有"top-left"、 "top-right"、"bottom-left"、 "bottom-right"、 "top"、 "bottom"、 "left"、 "right"、 "center"可选,默认为 "bottom"。

close_button参数,字符串类型或者布尔型,是否显示关闭按钮,如果是字符串类型,关闭按钮的文字就是给定的文字。

type参数,字符串类型,通知的类型,有"positive"、 "negative"、 "warning"、 "info"、 "ongoing",默认为None,不是其中的任何一种。

color参数,字符串类型,通知的背景颜色。

multi_line参数,布尔类型,是否让通知内容以多行格式显示。

progress参数,布尔类型,是否显示通知消失的进度条。

caption参数,字符串类型,显示在信息文本下的说明文字。

timeout参数,整数型,通知自动消失的时间,单位毫秒,为0就是不消失,但是要确保close_button不是False,否则通知没法正常消除,影响用户体验。

spinner参数,布尔类型,是否显示转盘动画。

(未完待续)

About

nicegui的中文入门教程。This is a chinese guide of nicegui.

Resources

License

Stars

Watchers

Forks

Sponsor this project

Packages

No packages published