categories | tags | ||||
---|---|---|---|---|---|
|
|
虽然VBA是Excel的官配二次开发语言,但鉴于python的简洁易用高可维护性,相信不少人更倾向于使用python来处理Excel相关的二次开发工作。目前已有一些优秀的python第三方库,例如xlrd
,xlwt
和xlutils
,可以分别实现Excel文件的读、写和读写转换工作;尤其推荐xlwings
,可以如同VBA一样操作Excel工作簿、工作表、单元格区域等。这些库的一个共同特征是完全不依赖于Excel本身,仅仅将其作为一个对象进行处理。但有时候,我们还是限定在Excel环境下,只是希望借助python而不是VBA来处理数据。
本文介绍一个作者从实际需求中总结设计的Excel插件模板设计工具pyAddin
,实现VBA控制显示、python驱动计算的效果。
对于开发者来说,利用开篇提及的python第三方库足以将任何繁杂的Excel数据处理工作转化为python编程来解决。但对于某些用户,他们已经习惯于在Excel环境下开展数据处理工作,并且系统通常没有配置而他们往往也不情愿安装python开发环境。此时如何使双方都达到舒适的状态——用户愉快地在Excel中使用开发者愉快地用python实现的功能?
-
辅助开发者设计Excel插件。
根据开发者提供的插件Ribbon区域界面信息,例如分组、按钮名称、回调函数名称等,
pyAddin
自动生成一个插件框架,包含了预定义的菜单按钮及回调函数。开发者只需在相应的回调函数中填写具体业务代码即可。 -
连接VBA与Python。
pyAddin
预定义了VBA与Python交互的接口函数,VBA调用python脚本执行主要任务,然后获取其结果显示回Excel。开发者可以设定Python解释器路径,因此可以内置便携式Python与插件一起发布给用户,这样便不依赖于用户系统配置Python开发环境与否。
从pyAddin
克隆或者下载仓库,在仓库根目录下执行安装命令:
python setup.py install
或者以开发者方式安装,便于基于当前版本进行自定义的开发和修改:
python setup.py develop
安装完成后可以在命令行执行pyAddin
命令查看基本使用方法:
usage: pyaddin [-h] [-n NAME] [-v] {init,create,update}
positional arguments:
{init,create,update} init, create, update
optional arguments:
-h, --help show this help message and exit
-n NAME, --name NAME addin file name to be created/updated: [name].xlam
-v, --vba create VBA addin only, otherwise VBA-Python addin by default
最后采用pip
命令卸载即可:
pip uninstall pyaddin
开始使用之前,需要在配置文件main.cfg中指定Python解释器位置。
# common line starts with #
# set path for python interpreter
# relative path is allowable
[python]
\python\python.exe
# default folder name for outputs
[output]
outputs
# standard output/error files name under [output]
[stdout]
output.log
[stderr]
errors.log
实际发布时推荐使用便携式Python,可以大大减少不必要的文件体积。
!!! warning "注意"
默认情况下,便携式Python并未预置`pip`。如果需要安装第三方库,可以先安装`pip`,然后`python -m pip install xxx`安装需要的库。最后发布给用户使用时,删除`pip`以减小体积,且不影响已经安装的库。
> [嵌入式Python : 如何在U盘安装绿色版 Python](https://zhuanlan.zhihu.com/p/33900815)
D:\GitHub\PyAddin>mkdir examples
D:\GitHub\PyAddin>cd examples
D:\GitHub\PyAddin\examples>pyaddin init
当前examples
目录下将产生一个Ribbon区域界面设计文件CustomUI.xml
。
Ribbon区域由CustomUI.xml
定义,本文示例如下:
<!--
Add custom UI definition between <tabs> and </tabs>, e.g.
<tab id="userRibbon" label="PyAddin">
<group id="group_about" label="About">
<button id="about" imageMso="About" size="large" label="About" onAction="callback_about"/>
</group>
</tab>
Please refer to the link below for detail:
https://docs.microsoft.com/en-us/previous-versions/office/developer/office-2007/aa338202(v%3doffice.12)
-->
<customUI xmlns="http://schemas.microsoft.com/office/2006/01/customui">
<ribbon startFromScratch="false">
<tabs>
<tab id="userRibbon" label="PyAddin Test">
<group id="g1" label="GROUP_1">
<button id="g1_b1" imageMso="Calculator" size="large" label="Cal_division" onAction="callback_cal"/>
<button id="g1_b2" size="large" label="Cal_multiply" onAction="callback_mtp"/>
</group>
<group id="g2" label="HELP">
<button id="g2_b1" imageMso="About" size="large" label="About" onAction="callback_about"/>
</group>
</tab>
</tabs>
</ribbon>
</customUI>
表示创建名称为PyAddin Test
的Ribbon工具卡,其中包含两个分组GROUP_1
和HELP
。第一个分组定义了两个按钮,并分别设定相应函数为callback_cal()
和callback_mtp()
。
D:\GitHub\PyAddin\examples>pyaddin create --name my_first_addin
当前目录下将创建Excel插件my_first_addin.xlam
及相应的辅助文件:
main.py
是VBA与Python交互的主函数,main.py
将VBA请求发送到具体的Python脚本main.cfg
为插件模板的配置文件,主要包括自定义Python解释器的路径scripts
文件夹作为默认的Python脚本的保存位置
接下来转入下一步的具体业务逻辑的开发。在此过程中,如果需要增删新的功能菜单,可以回到第二步更新CustomUI.xml
,然后执行update
命令:
D:\GitHub\PyAddin\examples>pyaddin update --name my_first_addin
这样可以在不影响原有代码的基础上,引入新添加的菜单回调函数模板。
以进行除法计算为例:在当前工作表A1
、A2
单元格输入内容,测试A1/A2
的结果
-
在VBA模块中定义回调函数
在菜单回调函数中使用VBA处理数据输入和输出,调用Python脚本完成具体的业务。
Sub callback_cal(control As IRibbonControl) ''' ' get value in cell A1, A2, then calculate A1/A2 in python ' ''' Dim a1$, a2$, args, res$ a1 = ActiveSheet.Range("A1").Value a2 = ActiveSheet.Range("A2").Value args = Array(a1, a2) If RunPython("scripts.test.division", args, res) Then ActiveSheet.Range("A3").Value = res Else MsgBox res End If End Sub
其中
RunPython()
函数调用Python脚本处理具体事务,函数说明参考插件的general
模块:Function RunPython(method_name As String, args, ByRef res As String) As Boolean ''' ' :param method_name: a string refer to the called python method -> package.module.method ' :param args: array for python arguments ' :param res: python return string ' :returns: True if everything is OK else False ' ''' pass End Function
-
在
scripts
文件夹下创建python脚本从上一步可以看出
RunPython()
函数的第一个参数"scripts.test.division"
指明了被调用的Python方法——scripts
包下的test.py
模块的dicision()
方法,因此定义该方法如下:# scripts/test.py def division(a, b): assert a!='', 'cell A1 is empty' assert b!='', 'cell A2 is empty' return float(a)/float(b)