Eel:利用 HTML 技术构建 Python GUI

官方 Github: ChrisKnott/Eel

Eel 是一个轻量的 Python 库,用于制作简单的类似于 Electron(但是比它更轻量) 的离线 HTML/JS GUI 应用程序,并具有对 Python 功能(capabilities)和库的完全访问权限。

Eel 托管一个本地 Web 服务器,然后允许您使用 Python 注释函数(annotate functions),以便可以从 JavaScript 调用它们,反之亦然。

Eel 旨在消除编写简短的 GUI 应用程序的麻烦。如果您熟悉 Python 和 Web 开发,则可以跳到 示例,该示例从给定文件夹中选择随机文件名(这在浏览器中是不可能的)。

1 使用方法

一个 Eel 应用程序将分为由各种Web技术文件(.html, .js, .css)组成的前端和由各种 Python 脚本组成的后端。

所有前端文件都应放在一个目录中(如果需要,可以将它们进一步分成多个文件夹)。

my_python_script.py     <-- Python scripts
other_python_module.py
static_web_folder/      <-- Web folder
  main_page.html
  css/
    style.css
  img/
    logo.png

假设您将所有前端文件都放在一个名为 web 的目录中,包括起始页 index.html,则该应用程序将按以下方式启动:

import eel
eel.init('web')
eel.start('index.html')

这将在默认设置(http://localhost:8000)的地址上启动网络服务器,并打开浏览器到 http://localhost:8000/index.html

如果安装了 Chrome 或 Chromium,则默认情况下,无论操作系统的默认浏览器设置为什么,它都将在“应用程序模式”(App Mode)下打开(带有 --app 的命令行标志)(可以覆写此行为)。

2 App 选项

可以将其他选项作为关键字参数传递给 eel.start()。一些选项包括应用程序所处的模式(例如'chrome'),应用程序运行的端口,应用程序的主机名以及添加其他命令行标志。

从 Eel v0.12.0开始,以下选项可用于 start()

  • mode, a string specifying what browser to use (e.g. 'chrome', 'electron', 'edge', 'custom'). Can also be None or False to not open a window. Default: 'chrome'
  • host, a string specifying what hostname to use for the Bottle server. Default: 'localhost')
  • port, an int specifying what port to use for the Bottle server. Use 0 for port to be picked automatically. Default: 8000.
  • block, a bool saying whether or not the call to start() should block the calling thread. Default: True
  • jinja_templates, a string specifying a folder to use for Jinja2 templates, e.g. my_templates. Default: None
  • cmdline_args, a list of strings to pass to the command to start the browser. For example, we might add extra flags for Chrome; eel.start('main.html', mode='chrome-app', port=8080, cmdline_args=['--start-fullscreen', '--browser-startup-dialog']). Default: []
  • size, a tuple of ints specifying the (width, height) of the main window in pixels Default: None
  • position, a tuple of ints specifying the (left, top) of the main window in pixels Default: None
  • geometry, a dictionary specifying the size and position for all windows. The keys should be the relative path of the page, and the values should be a dictionary of the form {'size': (200, 100), 'position': (300, 50)}. Default: {}
  • close_callback, a lambda or function that is called when a websocket to a window closes (i.e. when the user closes the window). It should take two arguments; a string which is the relative path of the page that just closed, and a list of other websockets that are still open. Default: None
  • app, an instance of Bottle which will be used rather than creating a fresh one. This can be used to install middleware on the
    instance before starting eel, e.g. for session management, authentication, etc.

3 暴露函数

除了前端文件夹中的文件之外,还将在 /eel.js 中提供 JavaScript 库。您应该在任何页面中包括以下内容:

<script type="text/javascript" src="/eel.js"></script>

Python 代码中的任何函数可以用 @eel.expose 装饰:

@eel.expose
def my_python_function(a, b):
    print(a, b, a + b)

而 Python 中的代码可以直接在 JavaScript 中像下面的方式调用:

console.log("Calling Python...");
eel.my_python_function(1, 2); // This calls the Python function that was decorated

类似的任何 JavaScript 函数也可以这样暴露:

eel.expose(my_javascript_function);
function my_javascript_function(a, b, c, d) {
  if (a < b) {
    console.log(c * d);
  }
}

然后在 Python 中调用:

print('Calling Javascript...')
eel.my_javascript_function(1, 2, 3, 4)  # This calls the Javascript function

exposed 的名称也可以通过传入第二个参数来覆盖。如果您的应用在构建过程中缩小了 JavaScript,则可能有必要确保在 Python 端可以解析函数:

eel.expose(someFunction, "my_javascript_function");

当将复杂对象作为参数传递时,请记住,在内部将它们转换为 JSON 并通过websocket 发送(该过程可能会丢失信息)。

下面看一个实例:

4 Hello World!

创建一个 HTML 文件 web/hello.html

<!DOCTYPE html>
<html>
  <head>
    <title>Hello, World!</title>

    <!-- Include eel.js - note this file doesn't exist in the 'web' directory -->
    <script type="text/javascript" src="/eel.js"></script>
    <script type="text/javascript">
      eel.expose(say_hello_js); // Expose this function to Python
      function say_hello_js(x) {
        console.log("Hello from " + x);
      }

      say_hello_js("Javascript World!");
      eel.say_hello_py("Javascript World!"); // Call a Python function
    </script>
  </head>

  <body>
    Hello, World!
  </body>
</html>

和一个简单的 Python 脚本 hello.py:

import eel

# Set web files folder and optionally specify which file types to check for eel.expose()
#   *Default allowed_extensions are: ['.js', '.html', '.txt', '.htm', '.xhtml']
eel.init('web', allowed_extensions=['.js', '.html'])

@eel.expose                         # Expose this function to Javascript
def say_hello_py(x):
    print('Hello from %s' % x)

say_hello_py('Python World!')
eel.say_hello_js('Python World!')   # Call a Javascript function

eel.start('hello.html')             # Start (this blocks and enters loop)

如果我们运行 Python 脚本(python hello.py),则会打开一个浏览器窗口,显示hello.html

并且在终端将看到:

Hello from Python World!
Hello from Javascript World!

5 返回值

尽管我们想将代码视为由单个应用程序组成,但 Python 解释器和浏览器窗口在单独的进程中运行。这可能会使它们之间来回通信变得一团糟,尤其是当我们总是必须显式地将值从一侧发送到另一侧时。

Eel 支持从应用程序另一端检索返回值的两种方法,这有助于使代码简洁。

为了防止在 Python 端永久挂起,已设定尝试从 JavaScript 端检索值的超时时间,默认为 10000 毫秒(10秒)。可以通过 _js_result_timeout 参数更改为 eel.init
JavaScript 端没有相应的超时。

5.1 回调

调用 exposed 函数时,可以立即传递回调函数。当函数在另一侧执行完毕时,将自动使用返回值自动调用此回调。

例如,如果我们在 JavaScript 中定义并暴露了以下函数:

eel.expose(js_random);
function js_random() {
  return Math.random();
}

然后在 Python 中,我们可以像这样从 JavaScript 端检索随机值:

def print_num(n):
    print('Got this from Javascript:', n)

# Call Javascript function, and pass explicit callback function
eel.js_random()(print_num)

# Do the same with an inline lambda as callback
eel.js_random()(lambda n: print('Got this from Javascript:', n))

反过来也一样。

5.2 同步返回

在大多数情况下,对另一端的调用是为了快速检索某些数据,例如小部件的状态或输入字段的内容。在这些情况下,同步几毫秒然后继续执行代码比将整个过程分解成回调更方便。

要同步获取返回值,只需将任何内容都传递给第二组括号即可。因此,在 Python 中,我们将编写:

n = eel.js_random()()  # This immediately returns the value
print('Got this from Javascript:', n)

您只能在浏览器窗口启动后(调用 eel.start() 之后)执行同步返回,否则显然会挂起调用。

在 JavaScript 中,该语言不允许我们在等待回调时进行阻塞,除非通过从 async 函数内部使用 await。 因此,JavaScript 方面的等效代码为:

async function run() {
  // Inside a function marked 'async' we can use the 'await' keyword.
  // Must prefix call with 'await', otherwise it's the same syntax
  let n = await eel.py_random()(); 
  console.log("Got this from Python: " + n);
}

run();

6 异步 Python

Eel基于 Bottle 和 Gevent 构建,它们提供了类似于 JavaScript 的异步事件循环。许多Python的标准库都隐式地假设只有一个执行线程-为了处理这个问题,Gevent 使用“monkey patch”的许多标准模块,例如 time。如果您需要猴子补丁,则应该 import gevent.monkey 并调用gevent.monkey.patch_all() 之后,再 import eel。猴子修补会干扰调试器之类的东西,因此除非有必要,否则应避免。

在大多数情况下,应该避免使用 time.sleep() 而使用 gevent 提供的版本。为了方便起见,直接从 Eel 提供了两个最常用的 gevent 方法sleep()spawn()(同样也节省了导入 time 和/或 gevent 的时间)。

例如:

import eel
eel.init('web')

def my_other_thread():
    while True:
        print("I'm a thread")
        eel.sleep(1.0)                  # Use eel.sleep(), not time.sleep()

eel.spawn(my_other_thread)

eel.start('main.html', block=False)     # Don't block on this call

while True:
    print("I'm a main loop")
    eel.sleep(1.0)                      # Use eel.sleep(), not time.sleep()

然后,我们将运行三个“线程”(greenlets):

  1. Eel 的内部线程用于服务Web文件夹
  2. my_other_thread 方法,重复打印 "I'm a thread"
  3. 主 Python 线程(将停留在最终的 while 循环中)重复打印 "I'm a main loop"

7 打包二进制文件

如果你想让用户下载你的软件使用, 而用户没有安装python, 你最好将你的程序打包成二进制可执行文件, 那么最好使用 PyInstaller

在你的 app 根目录下执行下面的命令:

python -m eel [your_main_script] [your_web_folder]

这将创建文件夹 dist/,如果你想要创建单文件程序,你需要使用 --onefile 参数, 如果不想程序运行的时候有一个黑色命令窗口, 你可以使用 --noconsole 参数。也可以运行像 python -m eel file_access.py web --exclude win32com --exclude numpy --exclude cryptography,排除一些包。更多信息见 PyInstaller 文档

8 Microsoft Edge

对于 Windows 10 用户,默认情况下会安装 Microsoft Edge(eel.start(.., mode='edge')),如果未安装首选浏览器,则会进行有用的回退。请参阅示例:

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容