我从来没有真正想过要当一名 C++ 程序员
因为我太懒了
不能那么辛苦地工作
但我必须承认
我过去常常嫉妒那些 C++ 程序员
嫉妒他们编写可视控件的能力
Visual Basic® 及其早期版本中的控件仅限于复合控件(由其他控件组成的控件)这种控件称为 UserControl在 Visual Basic 中编写能够在屏幕上呈现其特有可视外观的控件几乎是不可能的
现在好了可以使用功能强大的 Visual 编写各种类型的可视控件了!不仅可以编写复合的 UserControl还能继承现有的控件(如 TextBox)并扩展其新功能更重要的是还可以从头编写能够呈现其特有界面的可视控件
在本文中我将从头创建一个完整的可视控件以说明 Visual Basic NET 的后一种功能该控件是一个红绿灯 一个包含三个圆(分别代表红黄绿三个灯)的矩形图 显示各个灯亮时该控件的外观控件的背景颜色设置为系统颜色 ControlDark
图 带有三个 TrafficLight 控件的窗体每个控件亮不同的灯
我们称它为 TrafficLight 控件它可以通过代码或让用户单击灯来改变亮起的灯
因为 TrafficLight 是一个可视的 Windows 窗体控件它将继承 SystemWindowsForms 命名空间中的 Control 类这样它将具有很多预定义的属性方法和事件包括控制其外观的属性如 ForeColorBackColorSize 和 Location还包括事件如 MouseOver 和 Click您可以查看 NET 文档获得 Control 类成员的完整列表
红绿灯也需要具有特殊的属性和事件如下所示
Status 属性 | 确定亮起哪种颜色的灯
必须为以下三个枚举值之一
StatusRed红灯亮
StatusYellow黄灯亮
StatusGreen绿灯亮
BorderWidth 属性 | 红绿灯周围边框的宽度
StatusChanged 事件 | 当通过代码或由用户单击不同的灯改变 Status 属性的值时
触发该事件
由于这些成员不属于 Control 基类所以我们需要包括完整的代码以处理它们我们还需要绘制边框和三个相应颜色的灯的代码以便在屏幕上绘制红绿灯最后我们需要处理用户单击圆以更改亮起灯的操作并在更改亮起的灯时更改 Status 属性
为了使本示例尽可能接近实际应用环境我们还将包括能够确保在 Visual Studio® NET IDE 中更好地使用控件的代码我们为工具箱设置适当的图标并包括能够使属性更好地与各属性窗口集成的逻辑
现在让我们开始吧
第 步创建类型正确的项目
要创建一个保存 Windows 窗体控件的库需要在 Visual 中启动一个新项目选择 Windows Control Library(Windows 控件库)项目类型然后将项目命名为 MyControls
所创建的项目实际上可以保存多个 Windows 窗体控件每个控件都属于其各自的类但我们只需在其中创建一个控件
第 步更改基类
在控件库中创建的类自动命名为 UserControl默认情况下从 UserControl 类继承如果我们要创建复合控件那非常容易只需将其他控件从工具箱中拖到设计表面上即可
但是由于我们要从头创建自己的控件因此需要做一些更改将控件类的名称从 UserControl 更改为 TrafficLight然后将以下行
Inherits SystemWindowsFormsUserControl
更改为
Inherits SystemWindowsFormsControl
这样使最一般的 Control 类成为基类您会发现不再显示可视设计表面而是替换为组件设计表面
为保持代码的一致性也要将代码文件名从 UserControlVB 更改为 TrafficLightvb可以在 Solution Explorer(解决方案资源管理器)中进行更改右键单击代码文件的名称并选择 Rename(重命名)
还需要在类模块的顶部添加几行代码将 Option Strict 设置为 On并导入包含我们将来要用到的某些属性的命名空间下面是要放到代码最上面的两行 Option Strict On
Imports SystemComponentModel
第 步实现属性和事件
要实现 Status 属性首先要为可能的属性值创建枚举将以下几行插入以 Inherits 开始的行下面
Public Enum TrafficLightStatus
statusRed =
statusYellow =
statusGreen =
End Enum
此枚举是公开的也就是说使用该控件的窗体可以访问它
在这些行下面添加以下三行
Dim mStatus As TrafficLightStatus = TrafficLightStatusstatusGreen
Dim msngBorderWidth As Single = !
Public Event StatusChanged(ByVal NewStatus As TrafficLightStatus)
前两行中的两个变量可用于存储 Status 和 BorderWidth 属性的属性值还为这些属性设置了默认值保存 BorderWidth 的变量必须为 Single 类型因为它是绘制边框所用的图形语句需要的类型默认值中的惊歎号也表明它是 Single 类型此集合中的最后一行声明了 StatusChanged 事件
现在我们为 BorderWidth 属性编写代码在标记为 Windows Form Designer Generated Code(Windows 窗体设计器生成的代码)的代码区域下插入以下行
<DefaultValue(!) _
Description(红绿灯周围边框的宽度)> _
Public Property BorderWidth() As Single
Get
Return msngBorderWidth
End Get
Set(ByVal Value As Single)
If msngBorderWidth <> Value Then
msngBorderWidth = Value
MeInvalidate()
End If
End Set
End Property
前两行包括使该属性更好地使用 IDE 的属性DefaultValue 特性允许在 Properties(属性)窗口中将属性值重置为默认值(操作步骤稍后介绍)Description 特性提供选中该属性时在 Properties(属性)窗口底部显示的文本
DefaultValue 特性还有一个技巧如果将 TrafficLight 控件放到窗体上并保留 BorderWidth 属性的默认值那么窗体设计器将不生成设置属性值的代码行这使它与其他 Windows 窗体控件没有什么区别如果您查看典型控件(如 TextBox)的设计器生成的代码您会发现只包括设置为非默认值的属性的代码行我们赋予 TrafficLight 控件同样的能力
Property Get 简单明了Property Set 子句包括可视控件属性中常见的逻辑设置属性时重要的是在新属性值更改控件的外观时要能够重新绘制控件因此Set 子句负责确定传递的新值是否与属性中现有的值不相同如果相同则不执行操作如果不同则接受新值然后访问控件的 Invalidate 方法此方法表明控件的可视区域已过期控件需要重新绘制
Status 属性的处理有些不同因为它是枚举值DefaultValue 特性没有为枚举属性提供自动重置能力在这种情况下DefaultValue 也无法告诉设计器何时停止设置属性值的代码因此Status 属性的实现中不需要 DefaultValue 特性下面是 Status 属性的代码
<Description(红绿灯的状态(颜色))> _
Public Property Status() As TrafficLightStatus
Get
Status = mStatus
End Get
Set(ByVal Value As TrafficLightStatus)
If mStatus <> Value Then
mStatus = Value
RaiseEvent StatusChanged(mStatus)
MeInvalidate()
End If
End Set
End Property
看起来与 BorderWidth 属性的实现类似只有一点不同当 Status 属性发生改变时除了强制重新绘制控件外还会触发 StatusChanged 事件
要在 Properties(属性)窗口中处理属性的自动重置我们需要使用一种特殊的方法由于我们的属性命名为 Status因此必须将重置方法命名为 ResetStatus重置方法只是恢复属性的默认值以下是其代码
Public
Sub ResetStatus()
MeStatus = TrafficLightStatusstatusGreen
End Sub
为了提示设计器何时需要包括一行代码以便设置 Status 属性我们需要包括一个名为 ShouldSerializeStatus 的方法当属性需要一行代码时此方法返回布尔值 True否则则返回 False以下是其代码
Public Function ShouldSerializeStatus() As Boolean
If mStatus = TrafficLightStatusstatusGreen Then
Return False
Else
Return True
End If
End Function
第 步绘制控件的外观 要使控件具有一个可视的外观我们需要在 Paint 事件中放置逻辑然后每次控件需要刷新其可视外观时就会运行该逻辑
Windows 窗体中的 Paint 逻辑使用 中 GDI+ 部分中的类这些类基本上包括了 Windows API 图形功能由于适合 NET所以比 API 更易于使用但是有关它们的工作原理需要理解以下几点
在 Windows API 中图形操作需要一个窗口句柄有时称为 hWnd在 GDI+ 中它由 Graphics 对象取代该对象不仅代表了绘图区域还提供在该区域执行的操作(方法)
例如Graphics 对象具有以下方法可用来绘制各种屏幕元素
DrawCurve
DrawEllipse
DrawLine
DrawPolygon
DrawRectangle
DrawString
FillEllipse
FillPolygon
这些都是很容易理解的只是可用方法的示例一些更复杂的方法还允许旋转对象我们将使用 DrawRectangle 方法绘制边框使用 FillEllipse 方法绘制彩色的圆
大多数绘图方法都要求使用 Pen 或 Brush 对象Pen 对象用于绘制直线并确定直线的颜色和粗细Brush 对象用于填充区域确定填充区域所使用的颜色以及一些特殊效果(例如用位图填充区域)我们将使用特殊的 Brush 效果使当前没有亮起的灯的颜色变暗
下面是处理控件的 Paint 事件的代码
Protected Overrides Sub OnPaint(ByVal pe As _
SystemWindowsFormsPaintEventArgs)
MyBaseOnPaint(pe)
Dim grfGraphics As SystemDrawingGraphics
grfGraphics = peGraphics
首先绘制三个代表灯的圆
一个亮起其余两个熄灭
DrawLight(TrafficLightStatusstatusGreen grfGraphics)
DrawLight(TrafficLightStatusstatusYellow grfGraphics)
DrawLight(TrafficLightStatusstatusRed grfGraphics)
现在绘制红绿灯周围的轮廓
用画笔绘制轮廓将它涂成黑色
Dim penDrawingPen As New _
SystemDrawingPen(SystemDrawingColorBlack msngBorderWidth)
在控件上绘制红绿灯的轮廓
首先定义要绘制的矩形
Dim rectBorder As SystemDrawingRectangle
rectBorderX =
rectBorderY =
rectBorderHeight = MeHeight
rectBorderWidth = MeWidth
grfGraphicsDrawRectangle(penDrawingPen rectBorder)
释放图形对象
penDrawingPenDispose()
grfGraphicsDispose()
End Sub
首先使用基类绘制它通常使用控件的背景颜色绘制背景然后从事件参数中获取控件的 Graphics 对象
接下来用一个函数画出三个圆有关该函数的内容稍后介绍请注意我们必须向该函数传递一个 Graphics 对象的引用同时还要指示要画的圆(红黄绿)
然后是绘制轮廓的代码声明一个具有适当位置和大小的矩形然后传递给 Graphics 对象的 DrawRectangle 方法
最后图形对象激活其 Dispose 方法使用 GDI+ 时最好在完成图形对象后立即释放它们这有助于清除操作系统绘图时所用的资源如果要在 Windows® 或 Windows Me 中使用控件管理图形资源就更加重要因为这些操作系统处理这种资源的能力较差
下面是绘制圆的函数
Private Sub DrawLight(ByVal LightToDraw As TrafficLightStatus _
ByVal grfGraphics As Graphics)
Dim nCircleX As Integer
Dim nCircleY As Integer
Dim nCircleDiameter As Integer
Dim nCircleColor As Color
找到所有圆的 X 坐标和直径
nCircleX = CInt(MeSizeWidth * )
nCircleDiameter = CInt(MeSizeWidth * )
Select Case LightToDraw
Case TrafficLightStatusstatusRed
If LightToDraw = MeStatus Then
nCircleColor = ColorOrangeRed
Else
nCircleColor = ColorMaroon
End If
nCircleY = CInt(MeSizeHeight * )
Case TrafficLightStatusstatusYellow
If LightToDraw = MeStatus Then
nCircleColor = ColorYellow
Else
nCircleColor = ColorTan
End If
nCircleY = CInt(MeSizeHeight * )
Case TrafficLightStatusstatusGreen
If LightToDraw = MeStatus Then
nCircleColor = ColorLimeGreen
Else
nCircleColor = ColorForestGreen
End If
nCircleY = CInt(MeSizeHeight * )
End Select
Dim bshBrush As SystemDrawingBrush
If LightToDraw = MeStatus Then
bshBrush = New SolidBrush(nCircleColor)
Else
bshBrush = New SolidBrush(ColorFromArgb( nCircleColor))
End If
绘制代表红绿灯的圆
grfGraphicsFillEllipse(bshBrush nCircleX nCircleY nCircleDiameter nCircleDiameter)
释放笔刷
bshBrushDispose()
End Sub
这是整个控件中唯一的一个复杂图形在 GDI+ 中在要绘制椭圆的矩形中指定左上角的 X 坐标和 Y 坐标然后指定矩形的高度和宽度即可绘制一个椭圆我们分别将 X 坐标和 Y 坐标称为 nCircleX 和 nCircleY因为我们要绘制一个圆因此矩形的高度等于宽度用变量 nCircleDiameter 来控制该值
将 nCircleX 设置为刚好放到控件内(控件的宽度乘以 )nCircleY 取决于要绘制哪个灯可以设置成靠近控件的顶部(红灯)大约向下三分之一(黄灯)或大约向下三分之二(绿灯)直径 nCircleDiameter 设置为等于控件宽度的 %
要绘制实心椭圆还需完成一件事即确定要使用的颜色颜色取决于正在绘制哪个灯以及正在绘制的灯是否亮起亮起的灯的颜色要比熄灭的灯的颜色亮
创建绘图要使用的笔刷时需要使用这些颜色如果正在绘制的灯是亮起的即使用该颜色如果绘制的灯是熄灭的则要使用不同的方法实例化笔刷下面是熄灭的灯所使用笔刷的代码行
bshBrush = New SolidBrush(ColorFromArgb( nCircleColor))
这并不是 NET 中较好的方法名但 FromArgB 方法的作用是创建笔刷并通过将笔刷与背景颜色相结合来淡化颜色第一个参数使用的数字介于 至 之间数字越小背景颜色渗透越深我们使用的值为 它将大大降低处于熄灭状态的灯的颜色您可以尝试对该参数使用不同的值(或将它设置成可设置属性)以获得不同的效果
最后Graphics 对象的 DrawEllipse 方法绘制出该圆函数结束记住该函数需要调用三次以绘制三个不同的圆
第 步使控件响应用户
要允许用户更改灯的颜色必须检测到用户的鼠标单击操作有经验的 Visual Basic 开发人员都知道可以使用多种方法实现这一目的我们使用最简单的一种方法即检测 MouseUp 事件下面是检测用户单击并更改 Status 属性以与之匹配的代码
Private Sub TrafficLight_MouseUp(ByVal sender As Object _
ByVal e As SystemWindowsFormsMouseEventArgs) _
Handles MyBaseMouseUp
Dim nMidPointX As Integer = CInt(MeSizeWidth * )
Dim nCircleRadius As Integer = nMidPointX
If Distance(eX eY nMidPointX CInt(MeSizeHeight / )) _
< nCircleRadius Then
MeStatus = TrafficLightStatusstatusRed
Exit Sub
End If
If Distance(eX eY nMidPointX CInt(MeSizeHeight / )) _
< nCircleRadius Then
MeStatus = TrafficLightStatusstatusYellow
Exit Sub
End If
If Distance(eX eY nMidPointX CInt(( * MeSizeHeight) / )) _
< nCircleRadius Then
MeStatus = TrafficLightStatusstatusGreen
End If
End Sub
Private Function Distance(ByVal X As Integer _
ByVal Y As Integer _
ByVal X As Integer _
ByVal y As Integer) As Integer
Return CInt(SystemMathSqrt((X X) ^ + (Y y) ^ ))
End Function
事件处理非常简单检查鼠标单击的位置和每个圆心之间的距离(请注意圆心分别位于控件下方 // 和 / 的位置如果不太明白可以在纸上画出来看看)如果计算出的距离小于圆的半径则更改 Status 属性
距离由 Distance 函数使用您可能在代数课中学过的公式计算请注意平方根函数是从 SystemMath 命名空间中获得的数学函数通常都保存在该命名空间中
第 步清理
为了使控件顺利地运作我们还需要执行一些其他操作例如大小改变时需要重新绘制控件而且为了不改变控件的比例我们需要检测影响大小的属性发生更改的时间然后强制宽度等于高度的三分之一下面是完成这两项任务的事件处理程序
Private Sub TrafficLight_Resize(ByVal sender As Object _
ByVal e As SystemEventArgs) Handles MyBaseResize
MeInvalidate()
End Sub
Private Sub TrafficLight_Layout(ByVal sender As Object _
ByVal e As SystemWindowsFormsLayoutEventArgs) _
Handles MyBaseLayout
Select Case eAffectedProperty
Case Bounds
MeWidth = CInt(MeHeight * )
Case Else
不执行任何操作
End Select
End Sub
最后设置控件在工具箱中使用的图标控件已经有一个看似齿轮的默认图标但是我们要使用 Visual 附带的红绿灯图标
控件的工具箱图标是由名为 ToolboxBitmap 的类中的特性设置的在以 Public Class 开始的行上面插入以下行
<ToolboxBitmap(C:\Program Files\Microsoft Visual Studio
NET\Common\Graphics\icons\Traffic\TRFFCICO)> _
注意所有内容都应在一行中为了便于阅读我们在 Studio 后放置了一个回车粘贴该代码时要确保它们位于一行中Studio 和 NET 之间只需一个空格并删除回车如果您已经将 Visual Studio NET 安装到其默认位置那么上述代码将用 Visual Studio 目录中的图标设置该特性如果您没有将 Visual Studio NET 安装到其默认位置则需要相应地更改图标的路径名
第 步生成和测试控件
现在 TrafficLight 控件的设计就完成了选择 Build | Build MyControls(生成 | 生成 MyControls)以创建最终的控件库
要测试控件我们需要一个 Windows 窗体项目您可以在其他解决方案中执行此操作但在开发控件所用的解决方案中执行会更容易从菜单中选择 File | Add Project | New Project(文件 | 添加项目 | 新项目)选择 Windows Application(Windows 应用程序)项目类型将项目命名为 TestTrafficLight单击 OK(确定)启动测试所需的 Windows 应用程序
必须先将 TrafficLight 控件放到工具箱中才能将其拖放到测试应用程序的空白窗体 中右键单击工具箱中的 Windows 窗体选项卡然后选择 Customize Toolbox(自定义工具箱)选择 NET Framework Components(NET Framework 组件)选项卡然后单击 Browse(浏览)按钮浏览到您的 MyControls 项目所在的位置然后转到该项目的 /bin 目录选择 MyControlsdll 组件并单击 OK(确定)现在该对话框应如图 所示
图 在 Customize Toolbox(自定义工具箱)对话框中TrafficLight 控件被选中
您可以看到 TrafficLight 控件旁边有一个复选标记单击 OK(确定)按钮在工具箱的 Windows Forms(Windows 窗体)选项卡上TrafficLight 控件将出现在控件列表的底部图 显示了底部为 TrafficLight 控件的工具箱
图 工具箱底部的 TrafficLight 控件
现在您可以将 TrafficLight 控件拖放到 TestTrafficLight 的空白窗体 中默认情况下它被命名为 TrafficLight您可以调整控件的大小重新设置控件的属性包括 Status 属性该属性有一个下拉菜单菜单中包含该属性的三个可能的值请注意调整控件的大小或更改其属性时控件将在设计器中自动刷新
要恢复属性的默认值请将 Status 属性更改为 statusRed然后右键单击 Properties(属性)窗口中的 Status(状态)属性并选择 Reset(重置)如图 所示该属性将更改回 statusGreen如果将 BorderWidth 属性设置为 之外的其他值也可以使用同样的方法恢复其默认值
图 Properties(属性)窗口中 Status(状态)属性的 Reset(重置)选项请注意窗口底部有关 Status(状态)属性的说明
如果需要还可以为控件插入 StatusChanged 事件然后可以使用该事件中的以下代码行查看更改后的状态
MsgBox(新状态为 & NewStatusToString)
要在操作中测试该控件您需要启动 TestTrafficLight 项目此时它还不是该解决方案的启动项目因此您需要解决它在 Solution Explorer(解决方案资源管理器)中右键单击 Solution(解决方案)名称 Solution Explorer(解决方案资源管理器)中的第一行选择 Properties(属性)然后将 Single Startup Project(单启动项目)设置从 MyControls 更改为 TestTrafficLight然后单击 OK(确定)
按 F 键启动该项目将显示带有 TrafficLight 控件的窗体测试控件按下不同的灯查看它们是否亮起您还可以测试 BorderWidth 属性尝试在代码中设置灯的 Status 属性
小结
尽管 TrafficLight 是一个简单的控件(虽然曾有开发人员要把它用到真实的项目中)但它却显示了开发复杂控件所需要的所有原理包括
在控件中添加属性
使用默认值和说明使属性与 Visual Studio IDE 协调
在 Paint 事件中插入逻辑以绘制控件
在绘图逻辑中使用 GDI+
为控件设置位图以便在工具箱中显示
创建复杂控件的关键在于熟悉 GDI+ 的绘图能力如果理解了 TrafficLight 绘制边框和彩色圆的原理那么您就有了一个好的起点关键是有了 Visual Basic NET即使象我这么懒惰的程序员也能创建高级的 Windows 窗体