Dash应用浏览器端回调常用方法总结
本文示例代码已上传至我的
Github
仓库https://github.com/CNFeffery/dash-master
大家好我是费老师,回调函数是我们在Dash
应用中实现各种交互功能的核心,在绝大多数情况下,我们只需要以纯Python
的方式编写常规服务端回调函数即可,这也贯彻了Dash
无需编写javascript
即可构建web应用的理念。
但这并不代表在Dash
应用中我们只能使用Python
,更自由地,Dash
针对回调函数编写还提供了client side callback(我们通常称作浏览器端回调)相关功能,使得我们可以在仍然使用Python
编排回调函数角色的基础上,嵌入自定义的javascript
代码片段来执行相应的回调输入输出逻辑,从而解决一些特殊的需求。今天的文章中,我就将带大家一起学习Dash
浏览器端回调常用的方法和技巧。
阅读本文大约需要15分钟
浏览器端回调,顾名思义,其对应的函数体计算过程是在每个用户的本地浏览器中执行的,这在一些特殊的场景下,可以帮助我们节省服务器算力、网络传输带宽等消耗,还可以在用户网络状况很差时,提升一些用户交互功能的流畅度,亦或是可以让我们在Dash
应用中额外引入javascript
生态的功能(譬如在Dash
应用中高效渲染原生echarts
图表)。
而在Dash
中,我们主要有两种定义浏览器端回调的方式:
1 基于app.clientside_callback
编写简单浏览器端逻辑
此种浏览器端回调定义方式适用于执行非常简单的javascript
代码片段,只需要为app.clientside_callback()
的第一个参数传入字符串形式的javascript
函数体即可(推荐使用箭头函数),其中函数体内部参数的输入,以及结果的输出,原则类似常规的回调函数。
举个例子,我们来实现一段非常简单的逻辑,通过按钮的点击,来触发对应模态框的打开:
对应app.clientside_callback
的完整应用代码如下:
app1.py
import dash
from dash import html
import feffery_antd_components as fac
from dash.dependencies import Input, Output
app = dash.Dash(__name__)
app.layout = html.Div(
[
fac.AntdButton(
'打开模态框',
id='open-modal',
type='primary'
),
fac.AntdModal(
fac.AntdParagraph('测试内容'*100),
id='modal',
title='模态框示例'
)
],
style={
'padding': '50px 100px'
}
)
app.clientside_callback(
'(nClicks) => true',
Output('modal', 'visible'),
Input('open-modal', 'nClicks'),
prevent_initial_call=True
)
if __name__ == '__main__':
app.run(debug=True)
可以看到,写法非常简单,对于编写此类简单浏览器端回调的需求,我们只需要用到javascript
最基础的语法,非常的方便,再来个稍微复杂一点的例子,我们基于轮询组件,实现当前系统时间的实时更新:
app2.py
import dash
from dash import html, dcc
import feffery_antd_components as fac
from dash.dependencies import Input, Output
app = dash.Dash(__name__)
app.layout = html.Div(
[
dcc.Interval(
id='interval',
interval=1000 # 每秒触发一次
),
fac.AntdStatistic(
id='current-datetime',
title='当前时间'
)
],
style={
'padding': '50px 100px'
}
)
app.clientside_callback(
'''(n_intervals) => {
return `${new Date().toLocaleDateString().replaceAll("/", "-")} ${new Date().toLocaleTimeString()}`
}''',
Output('current-datetime', 'value'),
Input('interval', 'n_intervals')
)
if __name__ == '__main__':
app.run(debug=True)
2 基于ClientsideFunction
编写复杂浏览器端回调
如果我们想要执行的浏览器端回调逻辑比较复杂和冗长,那么在app.clientside_callback
里用字符串的方式写大段的javascript
代码就不太高效了♂️,相应的我们可以改为使用ClientsideFunction
来定义。
使用ClientsideFunction
来定义浏览器端回调,我们首先需要在我们的Dash
应用静态资源目录下(默认为assets
)建立相应的js
文件(名称随意,Dash
应用会自动加载静态资源目录下的js
文件到用户浏览器中),并在该js
文件中按照下列格式定义若干javascript
回调函数:
window.dash_clientside = Object.assign({}, window.dash_clientside, {
clientside: {
func1: () => {
// write your code logic
}
}
});
接着在相应的Python
程序中配合ClientsideFunction
按照下列格式关联编排回调函数即可:
app.clientside_callback(
ClientsideFunction(
namespace='clientside',
function_name='函数名称'
),
# 照常编排回调角色
)
废话不多说,我们直接将上文中实时刷新系统时间的示例改造成ClientsideFunction
形式以便理解:
assets/clientside_callbacks.js
window.dash_clientside = Object.assign({}, window.dash_clientside, {
clientside: {
update_datetime: (n_intervals) => {
return `${new Date().toLocaleDateString().replaceAll("/", "-")} ${new Date().toLocaleTimeString()}`
}
}
});
app3.py
import dash
from dash import html, dcc
import feffery_antd_components as fac
from dash.dependencies import Input, Output, ClientsideFunction
app = dash.Dash(__name__)
app.layout = html.Div(
[
dcc.Interval(
id='interval',
interval=1000 # 每秒触发一次
),
fac.AntdStatistic(
id='current-datetime',
title='当前时间'
)
],
style={
'padding': '50px 100px'
}
)
app.clientside_callback(
ClientsideFunction(
namespace='clientside',
function_name='update_datetime'
),
Output('current-datetime', 'value'),
Input('interval', 'n_intervals')
)
if __name__ == '__main__':
app.run(debug=True)
这样做的好处在于,我们可以把相对复杂的javascript
逻辑在原生的js
程序里编写,从而配合现代化ide获得更高效的编程体验,并且利用ClientsideFunction
形式,可以很方便地实现外部js
框架的引入使用,譬如引入使用原生echarts
,篇幅有限,今天先按下不表,之后另外发文举例介绍。
3 编写浏览器端回调的常用技巧
通过上文,我们知晓了Dash
中构建浏览器端回调的基本形式,下面我们补充一些有关浏览器端回调的实用技巧:
3.1 配合插件快捷生成模板代码
编写浏览器端回调,尤其是配合ClientsideFunction
时,其代码格式还是有些特殊的,不过别担心,如果你恰好在使用vscode
编写Dash
应用,可以在拓展里安装由我开发维护的插件feffery-dash-snippets
,安装完成后,可以通过输入一些快捷短语,进行相关代码模板的生成。
目前针对浏览器端回调+ClientsideFunction
,在py
文件中可用的快捷短语有:
callback-cs:oi
:快速初始化具有Input
和Output
角色的浏览器端回调函数
callback-cs:ois
:快速初始化具有Input
、Output
及State
角色的浏览器端回调函数
在js
文件中可用的快捷短语有:
callback:init
:快捷生成浏览器端回调函数定义模板
3.2 常用对象在浏览器端回调中的写法
在常规的服务端回调函数中我们经常会使用到dash.no_update
、PreventUpdate
、dash.callback_context
等对象来辅助回调函数功能逻辑的完成,而在浏览器端回调中,这些对象的写法要做一定变化:
dash.no_update
dash.no_update
在浏览器端回调中写作window.dash_clientside.no_update
,你也可以用feffery-dash-snippets
插件中的dash.no_update
快捷短语生成:
PreventUpdate
PreventUpdate
在浏览器端回调中写作PreventUpdate
,你也可以用feffery-dash-snippets
插件中的PreventUpdate
快捷短语生成(注意,在浏览器端回调中throw window.dash_clientside.PreventUpdate
等价于常规回调中的raise PreventUpdate
):
dash.callback_context
dash.callback_context
在浏览器端回调中写作window.dash_clientside.callback_context
,你也可以用feffery-dash-snippets
插件中的dash.callback_context
快捷短语生成:
3.3 在浏览器端回调中返回组件元素
我们在常规回调函数中,经常会以一些组件的children
或其他组件型参数为Output
目标,直接返回组件元素,在Python
中这样做很稀疏平常,但是在浏览器端回调中,我们如果有此类需求,则需要返回规定的JSON
数据格式,来表示一个组件元素:
{
props: {
// 定义当前组件的各属性,如
id: '组件id'
},
type: '组件完整名称,如AntdButton',
namespace: '组件所属组件库完整名称,如feffery_antd_components'
}
我们还是结合实际案例来做演示,这里我们的演示功能实现了通过按钮点击触发新的消息提示弹出:
具体代码如下,可以看到只要我们按照格式返回相应的组件JSON
数据,Dash
就会在浏览器中自动进行转换及渲染:
app4.py
import dash
from dash import html
import feffery_antd_components as fac
from dash.dependencies import Input, Output
app = dash.Dash(__name__)
app.layout = html.Div(
[
fac.AntdButton(
'新的消息',
id='new-message',
type='primary'
),
html.Div(id='new-message-container')
],
style={
'padding': '50px 100px'
}
)
app.clientside_callback(
'''(nClicks) => ({
props: {
content: "新的消息,nClicks:" + nClicks,
type: "info"
},
type: "AntdMessage",
namespace: "feffery_antd_components"
})''',
Output('new-message-container', 'children'),
Input('new-message', 'nClicks'),
prevent_initial_call=True
)
if __name__ == '__main__':
app.run(debug=True)
以上就是本文的全部内容,更多有关dash
应用开发的前沿知识和技巧欢迎持续关注玩转dash公众号。
Dash应用浏览器端回调常用方法总结的更多相关文章
- [SignalR]SignalR与WCF双工模式结合实现服务端数据直推浏览器端
原文:[SignalR]SignalR与WCF双工模式结合实现服务端数据直推浏览器端 之前开发基于WinForm监控的软件,服务端基于Wcf实现,里面涉及双工模式,在客户端里面,采用心跳包机制保持与服 ...
- html5 浏览器端数据库
为什么使用浏览器端数据库:随着浏览器的处理能力不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少用户等待从服务器获取数据的时间. 一.localStorage — 本地存储 可 ...
- HTML5浏览器端图片预览&生成Base64
本文主要介绍如何通过拖拽方式在浏览器端实现图片预览,并生成图片的Base64编码. 工具链接:图片转Base64. 首先介绍一下FileReader, FileReader对象允许浏览器使用File或 ...
- Web Storage:浏览器端数据储存机制
目录 概述 操作方法 存入/读取数据 清除数据 遍历操作 storage事件 参考链接 概述 这个API的作用是,使得网页可以在浏览器端储存数据.它分成两类:sessionStorage和localS ...
- SeaJS:一个适用于 Web 浏览器端的模块加载器
什么是SeaJS?SeaJS是一款适用于Web浏览器端的模块加载器,它同时又与Node兼容.在SeaJS的世界里,一个文件就是一个模块,所有模块都遵循CMD(Common Module Definit ...
- 在浏览器端用H5实现图片压缩上传
一.需求的场景: 在我们的需求中需要有一个在手机浏览器端,用户实现上传证件照片的功能,我们第一版上了一个最简版,直接让用户在本地选择图片,然后上传到公司公共的服务器上. 功能实现后我们发现一个问题,公 ...
- 不用Blazor WebAssembly,开发在浏览器端编译和运行C#代码的网站
本文中,我将会为大家分享一个如何用.NET技术开发"在浏览器端编译和运行C#代码的工具",核心的技术就是用C#编写不依赖于Blazor框架的WebAssembly以及Roslyn技 ...
- js接收对象类型数组的服务端、浏览器端实现
1.服务端 JSONArray jsonArr = JSONUtil.generateObjList(objList); public static generateObjList(List<O ...
- 为什么commonjs不适合于浏览器端
有了服务器端模块以后,很自然地,大家就想要客户端模块.而且最好两者能够兼容,一个模块不用修改,在服务器和浏览器都可以运行. 但是,由于一个重大的局限,使得CommonJS规范不适用于浏览器环境.还是上 ...
- js IndexedDB:浏览器端数据库的demo实例
IndexedDB具有以下特点. (1)键值对储存. IndexedDB内部采用对象仓库(object store)存放数据.所有类型的数据都可以直接存入,包括JavaScript对象.在对象仓库中, ...
随机推荐
- 2021-7-6 VUE笔记
v-cloak:使用的display:none: 直到编译完成后开始显示: v-text和插值表达式,非必要响应式用v-text会比较好,使用插值表达式要加上v-cloak; v-html:不推荐使用 ...
- jsp中的一些问题
jsp里的${pageContext.request.contextPath} 在JSP中,${pageContext.request.contextPath}是一个EL表达式(Expression ...
- 记一次线上问题 → Deadlock 的分析与优化
开心一刻 今天女朋友很生气 女朋友:我发现你们男的,都挺单纯的 我:这话怎么说 女朋友:脑袋里就只想三件事,搞钱,跟谁喝点,还有这娘们真好看 我:你错了,其实我们男人吧,每天只合计一件事 女朋友:啥事 ...
- 如何配置Linux的互信
如何配置Linux的互信? 这里针对的是root用户的,普通用户家目录/home/test/.ssh. 1.在客户端生成公钥私钥对 [root@auto1 ~]# ssh-keygen -t rsa ...
- html标签tr td是什么意思
<table>代表表格</table><tr>代表表格中的一行</tr><td>代表表格中的一列</td>'tr'与'td'交成 ...
- [glibc2.23源码]阅读源码&调试,找出free_hook-0x13分配失败的原因
0x00 写在前面 发freebuf了:https://www.freebuf.com/articles/endpoint/373258.html 本次阅读源码是本人第一次,算是一个全新的开始.本次看 ...
- 《SQL与数据库基础》21. 分库分表(一)
目录 分库分表(一) 拆分策略 垂直拆分 垂直分库 垂直分表 水平拆分 水平分库 水平分表 技术实现 MyCat概述 概念介绍 环境准备 目录介绍 MyCat入门 配置 分片配置(schema.xml ...
- ShardingSphere实战
前言 本文主要从sharding最新版本5.1.2版本入手搭建,按主键ID和时间进行分表. 本文主要介绍搭建过程,有兴趣了解shardingsphere的同学可以先自行查阅相关资料. shardsph ...
- 《Hadoop大数据技术开发实战》新书上线
当今互联网已进入大数据时代,大数据技术已广泛应用于金融.医疗.教育.电信.政府等领域.各行各业每天都在产生大量的数据,数据计量单位已从B.KB.MB.GB.TB发展到PB.EB.ZB.YB甚至BB.N ...
- Go学习笔记3
九.错误处理 1.defer+recover机制处理异常错误 展示错误: 发现:程序中出现错误/恐慌以后,程序被中断,无法继续执行. 错误处理/捕获机制: 内置函数recover: 2.自定义错误 需 ...