Access-自定义控件TabControl
p{
font-size: 15px;
}
.alexrootdiv>div{
background: #eeeeee; border: 1px solid #aaa; width: 99%; padding: 5px; margin: 1em 0 1em 0;
}
.alextitlep{
font-size: 18px; font-weight: bold; color: red;
}
.alexrootdiv table{
margin:10px;border-collapse:collapse;border:1px solid #aaa;width:100%;
}
.alexrootdiv table th{
vertical-align:baseline;padding:5px 15px 5px 6px;background-color:#d5d5d5;border:1px solid #aaa;text-align:left;
}
.alexrootdiv table td{
vertical-align:text-top;padding:6px 15px 6px 6px;background-color:#efefef;border:1px solid #aaa;
}
.attentationp{
font-size: 15px; text-indent: 2em; background: #ee0; color: red; font-weight: bold;
}
.borderstylediv{
width:12vw;text-align:center;
}
img{box-shadow:0 0 14px;}
-->
后来还发了一篇该自定义控件的使用案例:AccTabControl自定义控件应用案例
说说帖子的由来:
在论坛也混了蛮长时间了,一直没有发表过什么专题性质的文章。主要是论坛上高手如云,很多学习过程中的问题在论坛上都能找到答案,特别是论坛的精华帖。通过不断学习,我也开始对一些问题形成了些自己的想法。比如最近一段时间碰到一个问题:关于Access中动态添加控件的问题,Access中要给Form动态添加控件之类的,必须切换到窗体的设计模式,即使通过VBA代码也必须这么做。以往碰到这个问题,一般的做法是在窗体中先添加固定数目的控件,然后窗体加载时将其隐藏,当需要动态添加时就将其显示出来,但是这个方法一旦超出当初添加控件的数目时,就没办法解决了,并且控件添加多了对窗体加载速度也有一定影响。另外的话也可以通过一些ActiveX控件来做到这些,不过要找到适合Access且适合自己需求的ActiveX控件并不是件容易的事情,鉴于此我就想怎么才能在窗体上动态添加控件。
其实这个问题纠结了差不多有1年了,当初也想到来自绘这个途径,但是有好几个问题都不懂,所以解决不了。这些问题包括:
1、自绘的话,用什么在窗体上自绘? 肯定不能用Access的控件,线条、框什么的都不能用,因为这些都不能动态添加到窗体上。只有选择通过API来绘图,可以使用的包括GDI、GDI+。但是我那时对GDI和GDI+是一点了解都没有,所以画了很长时间研究VBA中用API、GDI跟GDI+。
2、要用API绘图就要有窗体句柄、要获得设备环境(DC),Access里怎么获取这些了?
可能有些人马上会想到Access窗体有个hwnd属性啊,不就可以了吗?其实这里面还有些曲折,后面我会详细说。这里大家所要了解的是Access的窗体下面还包含了好几个,包括窗体页眉、主体跟窗体页脚,它们都有句柄,要进行绘图的话,你得获取对应的句柄,而不是直接使用Access窗体的hwnd属性。
3、以上2个问题解决了,还只是完成了在窗体上自绘,要怎样才能将这些自绘窗体像控件一样使用到其他窗体上了?
可能大家看完这个问题,对Access有些了解的朋友会马上想到子窗体。但是当时我是想了1个星期才想到用子窗体,因为当初对这个问题我的想法是怎么在Access中做自定义控件,而没有想到怎么将窗体放到窗体里面这个方法。使用子窗体作为类似“控件”容器的承载体,这就解决的自定义控件的“容器”问题。
好了,说完这几个问题,那么我再总结下要读懂本文内容所需要储备的知识,如果你还对以下内容完全不了解的话,我建议你首先百度下或者找找相关的书什么的了解下,当然你也可以继续读下去,因为我会尽力讲的通俗易懂。不过如果你感觉阅读的很吃力的话,那你最好还是补一补相关的内容再来。
1、VBA中如何使用API?
2、GDI是用来干什么的?如何使用GDI?GDI句柄跟设备环境的关系,如何用GDI绘图?
3、Access窗体的构成。
4、Access子窗体是什么?怎么使用子窗体?
5、VBA中的类模块是什么?怎么使用类模块?类模块属性、方法、事件怎么建立?
6、Access窗体与类模块的关系;
7、使用VBA代码怎么调用自定义“控件”?
8、集合在类模块中的使用;
另外我也想说明一下,由于本贴内容可能会比较长,我会分批将所写内容更新进来,由于平时工作比较忙,可能一次更新的内容也不会太多,所以希望大家也不要急躁,慢慢看慢慢消化。另外相应的代码部分也有很多在调试之中,但是大部分主体的代码已经完成,我暂时不把源代码随帖子一起发布,我会将其中的大部分代码写到本贴里面并讲解,希望有兴趣的将贴看下去。
下面我们就开始讲怎么在Access来做一个类似TabControl的“控件”。
首先,我们来看下最终的效果,示例中包含了2个窗体,frmTest是个测试窗体,TabControl就是我们所谓的当作控件来使用的子窗体。另外还有些模块跟类模块,有些模块是无用的,因为我在做这个的时候,借鉴了部分代码,只是没有删除,我在后面会说到有哪些模块跟代码会使用到的,所以这里就不再说明各个模块的作用了。
双击打开frmTest,默认会建立3个框,相当于3个Tab,点击添加按钮,会自动添加Tab,点删除按钮会从最后依次删除Tab,在某个Tab上点击,会弹出一个对话框显示当前Tab的序号。
第一部分、建立clsAccTabBar类模块
在动手编写代码前,首先我们得分析下TabControl控件的结构,搞清楚我们需要建立什么样的模块、类模块以及窗体模块。从上面我们已经看到了我们用了一个子窗体作为TabControl的容器,那么TabControl里面还包括了很多Tab,这些Tab会构成一个集合Tabs,所以这个控件的层级关系就是:
TabControl
+---Tabs
+----Tab
之所以要理清楚这个关系,是因为基于这个结构建立我们的“控件”,会大大方便对我们控件的访问。这里的TabControl对应我们的窗体,Tabs的话我们将在TabControl的窗体代码中建立一个私有集合变量mTabBars,而Tab这个东西就需要我们自己来写类模块了。我将这个类模块命名为clsAccTabBar,cls代码是类模块,Acc表示是Access中的,TabBar就是这个类模块的含义。
下面我们来分析下这个类模块的内容,这个类模块所代表的是TabControl中的一个TabBar:
1、与属性相关的:包括TabBar的位置信息(Top、Left、Right、Bottom)、鼠标是否在其上(IsMouseOn)、是否被单击(Selected)、显示文字内容(Text)、标识字符串(Key)。可能大家还会说有与颜色相关的属性,这些我都放在了TabControl里面了,因为这些颜色是所有Tab共用的,而不是某一个Tab专属的,即使是选中色、鼠标移动其上的颜色。
2、与方法相关:Tab重画,这个方法我将它写在了TabControl里面了,当然你如果有兴趣可以为Tab建立一个ReDraw的方法;
3、与事件相关:TabBar被单击事件,TabBar鼠标移动事件,这2个事件的实现有些特殊,按道理应该在Tab类模块里建立这2个事件,但是鼠标的移动跟单击触发都是在TabControl里面,所以这2个事件我都把实现做到了TabControl窗体的事件代码里面了,后面讲述TabControl的时候我会再讲;
从上面的描述来看,我基本上把这个clsAccTabBar类模块只让其用于保存各个Tab相关信息,下面是类模块里面的代码:
Option Compare Database Private mIndex As Integer
Private mKey As String
Private mText As String
Private mTargetFom As String
Private mSelected As Boolean
Private mIsMouseOn As Boolean Public Property Get Index() As Integer
Index = mIndex
End Property Public Property Let Index(Value As Integer)
mIndex = Value
End Property Public Property Get Key() As String
Key = mKey
End Property Public Property Get Text() As String
Text = mText
End Property Public Property Let Text(Value As String)
mText = Value
End Property Public Property Get TargetFom() As String
TargetForm = mtargetform
End Property Public Property Get Left() As Long
Left = mRect.Left
End Property Public Property Let Left(Value As Long)
mRect.Left = Value
End Property Public Property Get Right() As Long
Right = mRect.Right
End Property Public Property Let Right(Value As Long)
mRect.Right = Value
End Property Public Property Get Top() As Long
Top = mRect.Top
End Property Public Property Let Top(Value As Long)
mRect.Top = Value
End Property Public Property Get Bottom() As Long
Bottom = mRect.Bottom
End Property Public Property Let Bottom(Value As Long)
mRect.Bottom = Value
End Property Public Property Get Width() As Long
Width = Abs(mRect.Right - mRect.Left)
End Property Public Property Get Height() As Long
Height = Abs(mRect.Bottom - mRect.Top)
End Property Public Property Get IsMouseOn() As Boolean
IsMouseOn = mIsMouseOn
End Property Public Property Let IsMouseOn(Value As Boolean)
mIsMouseOn = Value
End Property Public Property Get Selected() As Boolean
Selected = mSelected
End Property Public Property Let Selected(Value As Boolean)
mSelected = Value
End Property
有些属性我在前面没有提到,而在代码里又有,比如Width、Height,这个是宽度、高度,这个都是根据其他属性值来计算得到的。当然这里我再给大家提一下类模块的属性建立问题。 前面有很多私有变量声明,我这里把它们叫做类模块的字段,它们都是以m开头的,之后我所有的代码都是以m开头来代表类模块中的字段,与这些字段对应的Get/Let属性方法表示对这些字段的读取/写入操作。类模块中建立字段、属性的标准范式就是如此,应该避免使用公用变量。如果你对类模块的属性建立不是很清楚,还请在论坛或百度查阅相关的内容。
第二部分 构建Tabs集合
前面第一部分大家已经看到了clsAccTabBar的代码,内容是不是比较简单?确实比较简单,因为很多东西我都把它放到了TabControl里面实现了。大家对于clsAccTabBar这个类模块牢记2点:其一:这个类模块与之前所分析的模型中Tab对应,它将是某个具体Tab对象的模板代码;
其二:这个类模块所实现的功能就是用于记录每一个Tab的信息,在运行时,这个类模块帮助我们把这些信息存储在内存中;当要进行重画时,我们又可以使用这个类模块读取数据,用GDI把所有Tab画出来,或者画其中某几个Tab;
下面我们就来看看TabControl跟Tabs的实现吧,关于绘图的内容我将会在后面再单独说,因为后面我们还会将绘图部分的功能单独写入一个类模块中。我们先从最简单的Tabs来分析吧,稍后再看TabControl。Tabs是一个Tab的集合,我们直接使用Collection对象,虽然可能使用这个集合对象对于集合项目数较多时,性能会下降,但是我想谁也不可能在一个程序界面里出来个成百上千的Tab标签页吧!对于一个集合,我们所需要的功能包括添加、删除以及查找,而Collection对象都有现成的,确实方便多了。
首先我们要在TabControl窗体代码里面声明一个mTabBars的Collection对象:
Private mTabBars As New Collection
这里我直接用New声明了,也就是说这个TabControl“控件”被初始化时,就会在内存里分配空间给mTabBars(当然大家也可以不这么做,而是在添加TabBar方法里面对mTabBars进行检测,如果是nothing,就使用Set mTabBars=New Collection)。然后在窗体的UnLoad事件里面将mTabBars置为Nothing。这里啰嗦一句,实际编程的时候,大家要养成习惯,对需要进行清理的对象变量或者API中的一些资源对象,当存在调用代码时,立即在相应处添加清理代码,这样可以减少很多莫名奇妙的错误,特别是在VBA中使用API进行GDI编程时,这个好习惯可以帮助你减少很多不必要的调试麻烦。例如下面的ReleaseDC,它是GDI操作中的一个API函数,用于清除设备环境(DC)引用,mFormMainHwnd是对应的窗口句柄,mMainDC就是这个设备环境,设备环境是Windows非常珍贵的系统资源,如果用了不记得及时“还回”给系统,会造成程序莫名其妙出错,而且没有任何错误提示,甚至造成系统崩溃!
Private Sub Form_Unload(Cancel As Integer)
ReleaseDC mFormMainHwnd, mMainDC
Set mTabBars = Nothing
End Sub
下面我们再来看看如何向这个集合对象添加TabBar进去:
Public Sub AddTabBar(ByVal Key As String, ByVal Text As String, ByVal TargetForm As String)
Dim mTabBar As New clsAccTabBar
Dim lngText As Long
Dim mTextSize As Size lngText = LenB(StrConv(Text, vbFromUnicode))
GetTextExtentPoint32 mMainDC, Text, lngText, mTextSize If TabCount = Then
mTabBar.Left =
mTabBar.Top =
mTabBar.Right = mTextSize.cx +
mTabBar.Bottom =
Else
mTabBar.Left = mTabBars(TabCount).Right + 0.6
mTabBar.Top =
mTabBar.Right = mTabBar.Left + mTextSize.cx +
mTabBar.Bottom =
End If
mTabBar.Text = Text
mTabBars.Add mTabBar
ReDrawTabBar mTabBars.count
End Sub
方法有3个参数,前2个通过英文名就知道是什么意思,里面的代码我还没有使用到Key,只使用了Text,最后一个参数是个预留参数,暂时也没有用到。下面讲下代码内容,声明了3个变量,第一个mTabBar用于保存需要添加的TabBar的相关数据,第二个lngText保存Text字符串的长度,这个参数传递给API函数GetTextExtentPoint32,用于获取字符串的实际显示像素宽度;第三个mTextSize用于保存GetTextExtentPoint32函数运算后,所获得的字符串实际显示像素宽高值,它是一个Size的数据结构,代码如下:
Private Declare Function GetTextExtentPoint32 Lib "gdi32" Alias "GetTextExtentPoint32A" ( _
ByVal hdc As Long, _
ByVal lpsz As String, _
ByVal cbString As Long, _
lpSize As Size) As Long Public Type Size
cx As Long
cy As Long
End Type
需要提醒的是,GetTextExtentPoint32的声明最好放在TabControl的代码窗口中,Size的声明最好放在单独的模块代码中。GetTextExtentPoint32函数所使用mMainDC参数指的是主体窗口的设备环境DC,大家只需要知道这个东西就可以了,因为只有得到这个才能调用GDI进行绘图,关于绘图我再专门讲述,所以这里大家不用纠结这个,记住它是个与主体相关的画图用的设备环境就行了。然后后面的代码意图是当mTabBars没有TabBar时,直接写入首个TabBar的数据,其中的Right值是字符宽度加上16(左右边距合计16个像素),当有TabBar时,根据前一个TabBar的数据设置当前添加TabBar的数据。随后将这个TabBar添加到集合中,并调用ReDrawTabBar方法把这个TabBar画出来。
下面我们再来说下TabBar的删除操作,删除TabBar不仅仅是将其从mTabBars集合中清除掉,还要将窗体上的图像进行重绘,用背景色填充掉原先TabBar所在的位置,给查看者的感觉就是被删除掉了。代码如下,其中有2行代码(首行与末行)被我注释掉了,因为关于GDI绘图的方法我暂时还是写在了TabControl的代码里面,还没有完成对clsAccGDI类模块的代码,后面在说到GDI绘图时我还是继续讲述TabControl中的代码,大家有兴趣可以自己写写clsAccGDI类模块。
Public Sub RemoveTabBar()
On Error GoTo Err_Handle
'Dim FormDrawer As New clsAccGDI
Dim mRect As Rect
Dim mLastIndex As Integer mLastIndex = mTabBars.count mRect.Left = mTabBars(mLastIndex).Left
mRect.Right = mTabBars(mLastIndex).Right
mRect.Bottom = mTabBars(mLastIndex).Bottom
mRect.Top = mTabBars(mLastIndex).Top
FillTargetRect RGB(, , ), mRect
mTabBars.Remove mLastIndex
GoTo Exit_Sub
Err_Handle:
MsgBox "出错!"
Exit_Sub:
'Set FormDrawer = Nothing
End Sub
接下来我们再来看看如何在mTabBars中找到制定的TabBar,由于我之前的代码没有使用到Key,所以这里也没有基于Key来定位TabBar,我也没有写一个专门用于定位TabBar的方法,只是使用了最通用的For循环来查找,如果大家觉得不好,可以自己写个定位TabBar的方法。我这里把主体的MouseMove事件代码列出来说明下我搜索的方法。
Private Sub 主体_MouseMove(Button As Integer, Shift As Integer, X As Single, Y As Single)
Dim intX As Integer
Dim pX As Long, pY As Long
Dim mCurrentOn As Integer pX = X / TwipsPerPixelX()
pY = Y / TwipsPerPixelY() For intX = To mTabBars.count
If pX >= mTabBars(intX).Left And pX <= mTabBars(intX).Right And _
pY >= mTabBars(intX).Top And pY <= mTabBars(intX).Bottom Then
mTabBars(intX).IsMouseOn = True
ReDrawTabBar intX
mCurrentOn = intX
Exit For
End If
'Debug.Print "(x,y):(" & pX & "," & py & ")" & vbTab & "(Rx):(" & mTabBars(intX).Left & ")"
Next
If mPreTabBarOn <> mCurrentOn Then
If mPreTabBarOn > Then
mTabBars(mPreTabBarOn).IsMouseOn = False
ReDrawTabBar mPreTabBarOn
End If
mPreTabBarOn = mCurrentOn
End If
'ReDraw
mMousePoint.X = pX
mMousePoint.Y = pY
End Sub
说明下以上代码的意思,intX是个循环变量,在For循环中作为Index来遍历mTabBars集合,px,py是鼠标的坐标位置(像素值),VBA中MouseMove事件中返回X,Y是以Twip为单位的值,所以需要使用TwipsPerPixelX、TwipsPerPixelY自定义函数将其转换为像素值。建立一个模块mdlSysInfo,然后复制一下代码到模块中。随后以上的代码通过鼠标坐标值来判断其所在TabBar,找到时,完成一系列的设置操作,包括设置TabBar的IsMouseOn属性,重画TabBar并保存当前所处TabBar在mTabBars中的序号。随后再对之前鼠标所在的TabBar重画,并修改其IsMouseOn属性、保存之前鼠标所在TabBar的序号。最后保存鼠标当前位置数据,这个数据会在Click事件中使用到。
Option Compare Database
Option Explicit Public Type Size
cx As Long
cy As Long
End Type Private Declare Function GetDC Lib "user32" (ByVal Hwnd As Long) As Long
Private Declare Function ReleaseDC Lib "user32" (ByVal Hwnd As Long, ByVal hdc As Long) As Long
Private Declare Function GetDeviceCaps Lib "gdi32" (ByVal hdc As Long, ByVal nIndex As Long) As Long Private Const HWND_DESKTOP As Long =
Private Const LOGPIXELSX As Long =
Private Const LOGPIXELSY As Long = 'Returns the width of a pixel, in twips.
Public Function TwipsPerPixelX() As Single
Dim lngDC As Long lngDC = GetDC(HWND_DESKTOP)
TwipsPerPixelX = & / GetDeviceCaps(lngDC, LOGPIXELSX)
ReleaseDC HWND_DESKTOP, lngDC
End Function 'Returns the height of a pixel, in twips.
Public Function TwipsPerPixelY() As Single
Dim lngDC As Long lngDC = GetDC(HWND_DESKTOP)
TwipsPerPixelY = & / GetDeviceCaps(lngDC, LOGPIXELSY)
ReleaseDC HWND_DESKTOP, lngDC
End Function
Access-自定义控件TabControl的更多相关文章
- C# WinForm 自定义控件,DataGridView背景透明,TabControl背景透明
注意: 以下代码,属性直接赋值的语法糖要vs2015以上才支持. using System.ComponentModel; using System.Drawing; using System. ...
- 【基于WinForm+Access局域网共享数据库的项目总结】之篇一:WinForm开发总体概述与技术实现
篇一:WinForm开发总体概述与技术实现 篇二:WinForm开发扇形图统计和Excel数据导出 篇三:Access远程连接数据库和窗体打包部署 [小记]:最近基于WinForm+Access数据库 ...
- WPF自定义控件与样式(15)-终结篇 & 系列文章索引 & 源码共享
系列文章目录 WPF自定义控件与样式(1)-矢量字体图标(iconfont) WPF自定义控件与样式(2)-自定义按钮FButton WPF自定义控件与样式(3)-TextBox & Ric ...
- WPF自定义控件与样式(15)-终结篇
原文:WPF自定义控件与样式(15)-终结篇 系列文章目录 WPF自定义控件与样式(1)-矢量字体图标(iconfont) WPF自定义控件与样式(2)-自定义按钮FButton WPF自定义控件与 ...
- WPF 自定义TabControl控件样式
一.前言 程序中经常会用到TabControl控件,默认的控件样式很普通.而且样式或功能不一定符合我们的要求.比如:我们需要TabControl的标题能够居中.或平均分布:或者我们希望TabContr ...
- WPF自定义控件(五)の用户控件(完结)
用户控件,WPF中是继承自UserControl的控件,我们可以在里面融合我们的业务逻辑. 示例:(一个厌恶选择的用户控件) 后端: using iMicClassBase; using iMicCl ...
- WPF自定义控件(一)の控件分类
一.什么是控件(Controls) 控件是指对数据和方法的封装.控件可以有自己的属性和方法,其中属性是控件数据的简单访问者,方法则是控件的一些简单而可见的功能.控件创建过程包括设计.开发.调试(就是所 ...
- TabControl TabPage添加关闭按钮
自定义控件代码如下: using System.Drawing; using System.Windows.Forms; namespace Demo.UC { public class KKTab ...
- Winform自定义控件实例
本文转自http://www.cnblogs.com/hahacjh/archive/2010/04/29/1724125.html 写在前面: .Net已经成为许多软件公司的选择,而.Net自定义W ...
随机推荐
- 打造“黑客“手机--Kali Nethunter
从三月份开始,继续更新技术文章.一个月没有更新技术文章了,这一个月有一部分时间是在休息,另一部分时间是在学习汇编和操作系统,沉淀底层和逆向方面的技术. 今年年初,为了玩一下 kali NetHunte ...
- 每天一个Linux命令(05)--rm命令
自从学会了用mkdir创建目录之后,整个系统里就只能看到一堆空目录了,囧~ 那么今天我们来学一下如何清理这些空目录吧--rm命令,该命令的功能为删除一个目录中的一个或多个文件或目录,它也可以将某个目录 ...
- js方法提纲
Math.random() 日期时间函数(需要用变量调用):var b = new Date(); //获取当前时间b.getTime() //获取时间戳b.getFullYear() //获取年份b ...
- 用Less定义常用的CSS3效果函数及常用颜色搭配(让CSS写起来更有趣)
定义圆角及调用 /* 定义圆角 @radius 圆角大小 */ .round(@radius:5px){ border-radius:@radius; -webkit-border-radius: @ ...
- 1. Two Sum★
题目内容:Given an array of integers, return indices of the two numbers such that they add up to a specif ...
- webpack 引用 jquery + bootstrap 报错解决
webpack 引用 jquery + bootstrap , error : jQuery is not defind 在webpack.dev.conf.js plugins[] 加入 new w ...
- grpc-gateway:grpc对外提供http服务的解决方案
我所在公司的项目是采用基于Restful的微服务架构,随着微服务之间的沟通越来越频繁,就希望可以做成用rpc来做内部的通讯,对外依然用Restful.于是就想到了google的grpc. 使用grpc ...
- Android布局管理详解(1)—— LinearLayout 线性布局
Android的布局方式共有6种,分别是LinearLayout(线性布局).TableLayout(表格布局).FrameLayout(帧布局).RelativeLayout(相对布局).GridL ...
- java笔记---equals和==的区别
摔在这里几次,还是记下来吧.原文:http://www.cnblogs.com/shenliang123/archive/2012/04/16/2452156.html --------------- ...
- 好多鱼 Java
牛客网的题目: 链接:https://www.nowcoder.com/questionTerminal/e3dd485dd23a42899228305658457927牛牛有一个鱼缸.鱼缸里面已经有 ...