自己动手写客户端UI库——事件机制(设计思路大放送)

简介: 在上一篇文章中我们创建了一个Button控件,并把这个控件显示在界面上, 在这一篇文章中,我们将为这个控件增加一个事件和一个方法 一:怎么绑定事件的问题 在Winform中,我们对一个按钮绑定事件的方式如下(这是真正的事件) 然而,在WUI库中,为一个...
上一篇文章中我们创建了一个Button控件,并把这个控件显示在界面上,
在这一篇文章中,我们将为这个控件增加一个事件和一个方法

一:怎么绑定事件的问题

在Winform中,我们对一个按钮绑定事件的方式如下(这是真正的事件)
然而,在WUI库中,为一个按钮绑定事件是这样的,(这不是一个事件,这只是调用了一个方法,给这个方法传递了一个i额委托)
问题:
为什么会有这样的差异呢?实在是无奈之举(也希望园友多提意见)
回答:
我们在给一个WUI按钮绑定事件的时候,这个按钮有可能已经呈现在界面上了;也有可能还没有呈现在界面上;
如果还没有呈现在界面上,那也倒简单,我只要在呈现的时候(也就是把html代码append到浏览器之前),顺便用js给他绑定一个click事件就好了。
但如果他已经呈现在界面上了,该怎么办呢?我虽然也可以用JS绑定事件,但我却不知道该什么时候执行这段JS,这一段代码“btn2.Click += btn2_Click;”是我的用户写的,我不知道他们会什么时候用这一段代码。
所以,无奈之下,只能用这种方法“btn.BindClickEvent(OnClick);”来让用户绑定事件,这样我就可以在BindClickEvent方法内执行那一段JS代码了,毕竟BindClickEvent这个方法是我写的,我可以随意的控制他,让他做我想做的事情

二:Button的BindClickEvent方法

第一:
这个方法接收一个类型为Action<Button,EventArgs>类型的参数,Action其实就是一个委托,如果对这个东西不了解的朋友,可以看看我之前写的一篇文章《30分钟Linq教程》泛型委托那个小节
第二:
我们把这个参数存入了一个私有的List容器中,为什么这么做呢?一个按钮可以绑定多个Click事件,而且还要有先后顺序,所以按顺序存好,后面点击事件触发的时候,就可以直接遍历这个容器,按顺序执行这个容器中的委托就好了
第三:
Button实例IsRendered属性标致只着当前控件是否已经渲染在界面上了
第四:
我们在界面上添加一个Button,就把这个Button的实例存在这个字典中。为以后使用这个按钮(比如说触发他的事件)打下基础
第五:
我们判断是不是第一次对这个Button的实例做Click事件的绑定,如果是,那么就做下面的工作,如果不是,就不必做了;也就是说不管我给这个按钮绑定多少个Click事件,下面的工作也只做一次
第六:
我们让浏览器执行了一段JS脚本,这段Js脚本执行过之后,事件才算绑定成功;这段脚本给Button的Dom元素绑定了一个click事件,这个事件调用了C#中的ButtonClick方法,并给这个方法传递了一个参数,这个参数就是Button的ID

三:RenderContext的ButtonClick方法

第一:
在本系列的 第一篇文章中,我们介绍了C#是怎么和JS通讯的,这里就不多做介绍,只说2点:
1、JS要通过window.external调用C#里的方法
2、要把浏览器的ObjectForScripting设置给一个对象,这个对象必须是ComVisible的
第二:
所有的按钮,Dom元素的所有的click事件都会流入这个方法,这个方法是个路由器,把事件路由给用户的委托
第三:
我们根据按钮的ID,从字典(上一个小节有介绍)中拿出了按钮的实例,然后调用了实例方法Click,

四:Button类的Click方法

我们在这个方法中,遍历了所有绑定到Button实例上的“事件”,并且执行了这些事件。
遗留问题:这里没有太关注事件的执行顺序,以后会改进
五:PanelMain的AddChild方法
第一:
假设一个控件还没有渲染到界面上,那么是否允许开发人员对他绑定事件呢?当然是允许的!那么对于这一类使用方式,是在什么时候绑定事件的呢?就是在渲染的时候绑定的!
我们把控件添加到页面之后,马上就执行了这项工作,Button的ToJs方法就是在做这个工作,稍后介绍这个方法
第二:
只有当一个控件渲染到界面上之后,我们才会把它存入静态字典中,就是这行代码:RenderContext.ControlDic.Add(ctl.Id, ctl);
第三:
对于一个容器控件来说,他有一个Children集合,用来存储他自身的子控件,就是这行代码:this.Children.Add(ctl);
六:Button的ToJs方法
在这个方法中,我们把绑定事件的JS脚本得到,并反馈给调用者,
以后可能还会有其他的脚本,所以智力使用了StringBuilder
七:移除一个事件绑定
第一:
事件列表中应该存在待移除的事件
第二:
事件列表中就剩这么一个待移除的事件,并且,这个按钮已经渲染在界面上了;就执行js的解绑脚本
第三:
在事件列表中移除这个事件
八:移除所有事件绑定
第一:
当事件列表中存在事件记录
第二:
这个按钮已经被渲染在页面上,那么就执行JS解绑脚本
第三:
清空事件记录容器
修改记录
2015-1-22:
完成了文章的部分内容,修改了昨天写的代码
 
目录
相关文章
|
2月前
|
计算机视觉 Python
基于Dlib的人脸识别客户端(UI界面)
基于Dlib的人脸识别客户端(UI界面)
63 2
|
26天前
|
人工智能 API Apache
推荐3款开源、美观且免费的WinForm UI控件库
推荐3款开源、美观且免费的WinForm UI控件库
|
26天前
|
API C# 开发者
基于Material Design风格开源、免费的WinForms UI控件库
基于Material Design风格开源、免费的WinForms UI控件库!
|
2月前
|
Linux C# Android开发
分享3款开源、免费的Avalonia UI控件库
分享3款开源、免费的Avalonia UI控件库
194 0
|
4月前
|
编解码 前端开发 vr&ar
从零开始的PICO教程(4)--- UI界面绘制与响应事件
这篇文章是PICO开发系列教程的第四部分,主要介绍了如何在PICO 4 VR环境中创建UI界面,包括Canvas和Panel的配置、UI元素的绘制、以及Button和Slider的事件响应绑定,并通过示例展示了数字增减和滑块功能的具体实现。
从零开始的PICO教程(4)--- UI界面绘制与响应事件
|
3月前
|
搜索推荐 前端开发 C#
推荐7款美观且功能强大的WPF UI库
推荐7款美观且功能强大的WPF UI库
133 2
|
4月前
|
API Android开发
Android项目架构设计问题之选择和使用合适的UI库如何解决
Android项目架构设计问题之选择和使用合适的UI库如何解决
53 0
|
4月前
|
C# 开发者 Windows
一款基于Fluent设计风格、现代化的WPF UI控件库
一款基于Fluent设计风格、现代化的WPF UI控件库
105 1
|
4月前
|
数据可视化 数据挖掘 持续交付
Axure Web端元件库:从Quick UI到500+组件的飞跃
在快速变化的数字世界中,产品设计不仅仅是功能的堆砌,更是用户体验的精心雕琢。原型设计作为产品开发过程中的关键环节,其重要性不言而喻。Axure,作为业界领先的原型设计工具,凭借其强大的交互设计和丰富的功能,赢得了全球设计师和开发者的信赖。而Axure Web端元件库,则是这一平台上的一颗璀璨明珠,它以超过500个精心设计的组件为基础,为设计师们打开了一扇通往高效、高质量原型设计的大门。
180 0
|
5月前
|
前端开发 API
前端框架与库 - Material-UI组件库
【7月更文挑战第20天】Material-UI 是一个基于 React 的 UI 组件库,它遵循 Google 的 Material Design 设计规范,提供了丰富的预构建组件,极大地简化了前端开发过程。本文将深入浅出地介绍 Material-UI 的常见问题、易错点及如何避免这些问题,并附带代码示例,帮助你更好地掌握 Material-UI 的使用技巧。
168 0