首页 > 编程语言 >使用wxpython开发跨平台桌面应用,对wxpython控件实现类似C#扩展函数处理的探究

使用wxpython开发跨平台桌面应用,对wxpython控件实现类似C#扩展函数处理的探究

时间:2024-11-01 11:23:38浏览次数:4  
标签:box 控件 items combo 扩展 跨平台 wxpython ComboBox wx

本人之前对C#开发非常喜欢,也从事开发C#开发桌面开发、Web后端、Vue前端应用开发多年,最近一直在研究使用Python,希望能够把C#的一些好的设计模式、开发便利经验引入到Python开发中,很多时候类似的开发方式,可以极大提高我们开发的效率,本篇随笔对wxpython控件实现类似C#扩展函数处理的探究总结。

1、C#扩展函数特点及便利性回顾

C# 的扩展方法具有以下几个特点和便利性:

  1. 语法简洁:扩展方法允许在不修改原始类型的情况下,向现有类型添加新功能。调用时看起来像是实例方法。

  2. 易于使用:可以在调用扩展方法时使用点语法,这使得代码更易读且更自然。例如,对于字符串类型,可以直接调用扩展方法而不是传递实例。

  3. 静态类和静态方法:扩展方法必须定义在静态类中,并且本身也是静态方法。第一个参数指定了要扩展的类型,并使用 this 关键字修饰。

  4. 提升代码组织性:将相关功能组织到扩展方法中,可以减少主类中的代码量,提高可维护性。

  5. 与 LINQ 的结合:扩展方法在 LINQ 中的应用非常广泛,使得集合操作更加直观和简洁。

  6. 支持多种数据类型:可以为基本类型、集合类型甚至自定义类型添加扩展方法,从而提供更广泛的功能。

总的来说,扩展方法在提高代码可读性和可维护性方面具有明显的优势,是C#语言设计中的一项重要特性。

我在开发C#Winform应用前端的时候,在自己的公用类库上实现了很多扩展方法,特别是对于一些控件,增加了很多如绑定数据列表、绑定字典大类名称后直接加载数据列表等,以及一些对数据类型的通用处理,如字符串的格式判断或者裁剪等等。

如在随笔《使用扩展函数方式,在Winform界面中快捷的绑定树形列表TreeList控件和TreeListLookUpEdit控件》中介绍过,对于一些常规控件的数据绑定处理。

对于常规的列表绑定,我们可以用简单的一个扩展函数实现,如下所示。

    //常规类别绑定
    this.txtProjectList4.BindDictItems(list, "Text", "Value", true, columns.ToArray());

定义了扩展方法,就很容易实现数据的绑定,减少涉及控件处理的细节。

那么对于Python如何使用类似的方式实现呢,我们需要对Python 的语言特性进行了解和实际测试下。

 

2、基于Python实现数据的快速绑定

我们让ChatGPT来进行解答,它给出的代码答案是如下所示。

1)属性绑定方式

import wx

# 定义扩展方法
def BindDictItems(self, items_dict):
    self.Clear()  # 清空现有项
    for key, value in items_dict.items():
        self.Append(value, key)  # 添加项,key 为用户数据,value 为显示内容

# 将扩展方法绑定到 wx.ComboBox 类
setattr(wx.ComboBox, "BindDictItems", BindDictItems)

# 测试应用
class MyApp(wx.App):
    def OnInit(self):
        frame = wx.Frame(None, title="ComboBox BindDictItems Example", size=(300, 200))
        panel = wx.Panel(frame)

        combo_box = wx.ComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10))
        
        # 使用扩展方法绑定字典项
        items = {
            "item1": "Item One",
            "item2": "Item Two",
            "item3": "Item Three"
        }
        combo_box.BindDictItems(items)

        frame.Show()
        return True

if __name__ == "__main__":
    app = MyApp()
    app.MainLoop()

解释

  1. 定义 BindDictItems 方法:这个方法接受一个字典,将字典中的项绑定到 ComboBox 中。
    • 使用 self.Clear() 清空现有项。
    • 遍历字典,并使用 Append 方法将项添加到 ComboBox
  2. 使用 setattr 绑定方法:通过 setattr 将定义的方法添加到 wx.ComboBox 类中。
  3. 测试应用:在 MyApp 中创建一个 wx.ComboBox 并调用 BindDictItems 方法。

这样,你就可以为 wx.ComboBox 提供一个 BindDictItems 方法。

测试发现,对于功能实现没有问题,但是对于开发的时候,combo_box 没有函数提示,也就是没有强类型的函数提示,这样只是进行了动态绑定,无法在开发的时候,获得IDE的智能提示,很容易写错。

另外也可以使用下面代码实现类似setattr的处理,相当于直接附加方法,同样在使用的时候也是没有函数的智能提示,但是能够正常工作,俗称 猴子补丁

# 使用猴子补丁将方法添加到 wx.ComboBox
wx.ComboBox.BindDictItems = bind_dict_items

 

2)自定义子类方式

对该结果不满意,继续追问实现该功能的其他途径,给出的另一个解决方案是继承父类,在子类实现更多的自定义函数。

为了实现更好的智能提示和类型检查,可以通过使用类型注解和创建一个自定义子类来扩展 wx.ComboBox。这样可以确保 IDE 提供正确的代码补全和提示。

给出的代码如下所示。

import wx
from typing import Dict

class MyComboBox(wx.ComboBox):
    def BindDictItems(self, items_dict: Dict[str, str]) -> None:
        """Bind a dictionary of items to the ComboBox."""
        self.Clear()  # 清空现有项
        for key, value in items_dict.items():
            self.Append(value, key)  # 添加项,key 为用户数据,value 为显示内容

class MyApp(wx.App):
    def OnInit(self):
        frame = wx.Frame(None, title="ComboBox BindDictItems Example", size=(300, 200))
        panel = wx.Panel(frame)

        combo_box = MyComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10))
        
        # 使用扩展方法绑定字典项
        items = {
            "item1": "Item One",
            "item2": "Item Two",
            "item3": "Item Three"
        }
        combo_box.BindDictItems(items)

        frame.Show()
        return True

if __name__ == "__main__":
    app = MyApp()
    app.MainLoop()

通过这种方式,可以确保在使用 BindDictItems 时能够获得智能提示,从而减少错误。

这样的方式,确实有强类型的函数提示了。虽然可以获得智能提示,但是也就是破坏了界面类的代码,也就是需要修改相关的使用代码,而非C#扩展方法那样,隐式的扩展了。

3)通过中介辅助类的方式实现

有些情况下,不适合继承关系,不可能为每个类都提供一个子类来封装,有时候提供一些辅助类可能更具有性价比。

要在不继承父类的情况下实现类似 C# 的扩展方法,并确保获得智能提示,推荐使用类型注解和一个中介类来包装扩展方法。通过这种方式,IDE 可以识别这些扩展方法并提供智能提示。

创建一个名为 ComboBoxExtensions.py 的文件,其中定义扩展方法。

# combo_box_extensions.py
from typing import Dict
import wx

class ComboBoxExtensions:
    @staticmethod
    def bind_dict_items(combo_box: wx.ComboBox, items_dict: Dict[str, str]) -> None:
        """Bind a dictionary of items to the ComboBox."""
        combo_box.Clear()  # 清空现有项
        for key, value in items_dict.items():
            combo_box.Append(value, key)  # 添加项,key 为用户数据,value 为显示内容

在主应用程序中,导入扩展类并使用其方法。

import wx
from combo_box_extensions import ComboBoxExtensions  # 导入扩展类

class MyApp(wx.App):
    def OnInit(self):
        frame = wx.Frame(None, title="ComboBox Extensions Example", size=(300, 200))
        panel = wx.Panel(frame)

        combo_box = wx.ComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10))

        # 使用扩展方法绑定字典项
        items = {
            "item1": "Item One",
            "item2": "Item Two",
            "item3": "Item Three"
        }
        ComboBoxExtensions.bind_dict_items(combo_box, items)  # 这里应该有智能提示

        frame.Show()
        return True

if __name__ == "__main__":
    app = MyApp()
    app.MainLoop()

ComboBoxExtensions 类包含一个静态方法 bind_dict_items,该方法接受 wx.ComboBox 实例和字典作为参数。

在主应用程序中,调用 ComboBoxExtensions.bind_dict_items(combo_box, items),这将获得智能提示。

4)使用协议类型的方式处理,并在使用的时候转换为协议类

为了确保在不继承的情况下实现扩展方法并获得智能提示,最佳方案是结合类型注解和一个特定的函数注册过程。以下是一个经过验证的方式,确保能够在实例上调用扩展方法,同时获得 IDE 的智能提示。

combo_box_extensions.py 中定义扩展函数,并使用 cast 来确保类型正确。

# combo_box_extensions.py
from typing import Dict, Protocol
import wx
from typing import cast

class ComboBoxWithBindDictItems(Protocol):
    def BindDictItems(self, items_dict: Dict[str, str]) -> None:
        ...

def bind_dict_items(self: wx.ComboBox, items_dict: Dict[str, str]) -> None:
    """Bind a dictionary of items to the ComboBox."""
    self.Clear()  # 清空现有项
    for key, value in items_dict.items():
        self.Append(value, key)  # 添加项,key 为用户数据,value 为显示内容

# 将扩展方法绑定到 wx.ComboBox
wx.ComboBox.BindDictItems = bind_dict_items

在主应用程序中调用扩展方法,并确保正确使用类型注解。

import wx
from combo_box_extensions import ComboBoxWithBindDictItems  # 导入协议类型

class MyApp(wx.App):
    def OnInit(self):
        frame = wx.Frame(None, title="ComboBox Extensions Example", size=(300, 200))
        panel = wx.Panel(frame)

        combo_box = wx.ComboBox(panel, style=wx.CB_DROPDOWN, pos=(10, 10))

        # 确保类型为 ComboBoxWithBindDictItems,以获得智能提示
        cast(ComboBoxWithBindDictItems, combo_box).BindDictItems({
            "item1": "Item One",
            "item2": "Item Two",
            "item3": "Item Three"
        })  # 这里应该有智能提示

        frame.Show()
        return True

if __name__ == "__main__":
    app = MyApp()
    app.MainLoop()

可以看到,通过cast的方式转换后,具有函数代码的智能提示了。

协议类型:定义 ComboBoxWithBindDictItems,它确保 BindDictItems 方法存在。

使用 cast:在调用 BindDictItems 方法时,使用 cast 来明确指定 combo_box 的类型为 ComboBoxWithBindDictItems,这样 IDE 能够识别并提供智能提示。

智能提示:通过类型注解和 cast,IDE 能够识别扩展方法并提供智能提示。

 无继承:避免了复杂的继承结构,同时实现了功能扩展。

 

以上几种在Python开发中,对于实现C#扩展函数方法的实现,不过总体来说,虽然能够实现类似的方式,却没有C#那种简洁明了,不知道以后Python发展后解决或者是我没有研究透彻的原因,很多时候如果要实现自定义函数的处理方式,估计我只能结合子类继承和辅助类的方式一起解决这个问题。

 

标签:box,控件,items,combo,扩展,跨平台,wxpython,ComboBox,wx
From: https://www.cnblogs.com/wuhuacong/p/18519670

相关文章

  • 界面控件DevExpress JS & ASP.NET Core v24.1亮点 - 支持Angular 18
    DevExtreme拥有高性能的HTML5/JavaScript小部件集合,使您可以利用现代Web开发堆栈(包括React,Angular,ASP.NETCore,jQuery,Knockout等)构建交互式的Web应用程序。从Angular和Reac,到ASP.NETCore或Vue,DevExtreme包含全面的高性能和响应式UI小部件集合,可在传统Web和下一代移动应用程序中......
  • 界面控件DevExpress WPF中文教程:Data Grid——卡片视图概述
    DevExpressWPF拥有120+个控件和库,将帮助您交付满足甚至超出企业需求的高性能业务应用程序。通过DevExpressWPF能创建有着强大互动功能的XAML基础应用程序,这些应用程序专注于当代客户的需求和构建未来新一代支持触摸的解决方案。无论是Office办公软件的衍伸产品,还是以数据为中心......
  • 界面控件Kendo UI for Angular 2024 Q3亮点 - 全新的页面模板
    随着最新的2024Q3版本,Progress使用户能够使用现成的页面模板和构建块更快地构建令人惊叹的应用程序,使您的Telerik和KendoUI开发体验更好。Telerik和KendoUI 2024Q3版本将焦点放在新推出的页面模板和构建块上,每个页面模板和构建块都预先配置了TelerikUIforBlazor、KendoU......
  • 第三方控件学习汇总
    语法高亮-SynEdit在Delphi10.3上的安装和使用运行DOSCommand命令,然后获得返回文本EhLib11DBGridEH学习汇总TMSFlexCelVCL&FMXv7.8学习汇总DevExpress21控件学习汇总mORMot框架学习汇总FastReport6.8.11在Delphi10.3汇总uniGUI学习汇总Delphi10.3下SimpleGraphv2.9......
  • 基于Material Design风格开源、免费的WinForms UI控件库
    前言今天大姚给大家分享一个基于Google的MaterialDesign风格开源、免费的.NETWinFormsUI控件库:MaterialSkin。WinForms介绍WinForms是一个传统的桌面应用程序框架,它基于Windows操作系统的原生控件和窗体。通过简单易用的API,开发者可以快速构建基于窗体的应用程序,并且......
  • 【Python入门】7天速成Python桌面应用开发高手,WxPython vs PyQt:谁更胜一筹?
    ......
  • 基于wxpython的跨平台桌面应用系统开发
    我曾在随笔《基于Python后端构建多种不同的系统终端界面研究》介绍了多种系统终端界面开发的处理,其中涉及到的wxpython,是一个非常不错的原生界面效果组件,我们可以通过利用其各种界面控件,结合Python跨平台运行的特性,为Windows、MacOS、Ubuntu等Linux系统,开发一套界面效果一致的应用......
  • 1 Java 跨平台原理
    JVMJava的特性就是程序员一次编写,到处运行,意思是我们只需要写一份代码,就可以在不同的操作系统(windows、Linux、MacOS等)中运行。但是不同的操作系统能看懂的指令是不同的,所以实现方式就是给每个操作系统开发一个Java虚拟机(JVM)。JVM运行的过程是把我们代码编译后的字节码翻译成系......
  • 界面控件DevExpress WPF v24.1新版亮点:属性网格、轻量级主题升级
    DevExpressWPF拥有120+个控件和库,将帮助您交付满足甚至超出企业需求的高性能业务应用程序。通过DevExpressWPF能创建有着强大互动功能的XAML基础应用程序,这些应用程序专注于当代客户的需求和构建未来新一代支持触摸的解决方案。DevExpressWPF控件今年一个重大版本——v24.1全......
  • Rust 跨平台应用开发第一章:快速上手 Rust——实用示例
    1.3实用示例在这一节中,我们将通过一系列实用的示例来帮助您更好地理解Rust的特性,并展示如何在实际项目中使用这些特性。示例将涵盖文件操作、网络请求、并发编程、命令行工具以及使用Cargo管理依赖等多个方面。1.3.1文件操作示例Rust提供了强大的标准库来进行文件操......