VB SendMessage API 简单用法

1、

在Windows编程中,向文本框控件、列表控件、按钮控件等是我们最常接触的控件了。但是在VB中这些控件有时无法实现我们的需要。在这时,我们只要简单的利用Windows API函数就可以扩充这些控件的功能了。

顾名思义,SendMessage函数就是向窗口(这里的窗口指的是向按钮、列表框、编辑框等具有hWnd属性的控件)发送消息的函数,该函数的定义如下:

Declare Function SendMessage Lib "user32" Alias "SendMessageA" _

(ByVal hwnd As Long, _

ByVal wMsg As Long, _

ByVal wParam As Long, _

lParam As Any) As Long

其中hwnd指定接受消息的窗口,参数wMsg指定消息值,参数wParam lParam分别定义传递到窗口的附加参数。而在Windows系统的很多消息中,有一些不仅仅是提供一个窗口消息那么简单。它们可以控制窗口的动作和属性。下面我将分次向向大家介绍SendMessage函数在扩充基本控件功能方面的应用。

一、列表(ListBox)控件

在Windows中,有一系列的以LB_开头的列表消息,这里介绍的就是利用LB消息控制的ListBox的应用

1、使列表中光标移动到不同的列表项上有不同的提示(ToolTip)

在列表框控件中有一个ToolTipText属性,该属性决定了当光标在列表框上移动时出现的提示文字。但是如何使得当光标在不同的列表项上移动时的提示文字也不同呢?问题的关键是要知道在光标移动时光标所在的列表项的索引,使用SendMessage函数发送LB_ITEMFROMPOINT消息就可以获得。下面是程序范例:

Option Explicit

Const LB_ITEMFROMPOINT = &H1A9

Private Declare Function SendMessage Lib "user32" Alias "SendMessageA" _

(ByVal hwnd As Long, _

ByVal wMsg As Long, _

ByVal wParam As Long, _

lParam As Any) As Long

Private Sub Form_Load()

Dim i

For i = 1 To 200

List1.AddItem Str(i) + " Samples in this list is " + Str(i)

Next i

End Sub

Private Sub List1_MouseMove(Button As Integer, Shift As Integer, X As Single, Y As Single)

Dim lXPoint As Long

Dim lYPoint As Long

Dim lIndex As Long

If Button = 0 Then '确定在移动鼠标的同时没有按下功能键或者鼠标键

'获得光标的位置,以像素为单位

lXPoint = CLng(X / Screen.TwipsPerPixelX)

lYPoint = CLng(Y / Screen.TwipsPerPixelY)

'

With List1

'获得 光标所在的标题行的索引

lIndex = SendMessage(.hwnd, LB_ITEMFROMPOINT, 0, _

ByVal ((lYPoint * 65536) + lXPoint))

'将ListBox的Tooltip设置为该标题行的文本

If (lIndex >= 0) And (lIndex <= .ListCount) Then

.ToolTipText = .List(lIndex) 'Return the text = .list(lIndex)

Else

.ToolTipText = ""

End If

End With

End If

End Sub

首先在Form1中加入一个ListBox控件,然后再将上面的代码加入到Form1的代码窗口中。运行程序,当光标在列表中移动时,可以看到根据光标所在的不同的列表项,提示文字也不相同。

2、向列表中加入横向滚动条使得可以浏览长列表项

当向列表中加入的列表项超出了列表的显示范围后,列表并不会出现横向滚动条让你可以通过滚动来浏览项目的全部内容。利用LB_SETHORIZONTALEXTENT消息可以设置列表的横向滚动条以及滚动长度。下面是范例程序:

Option Explicit

Private Type RECT

Left As Long

Top As Long

Right As Long

Bottom As Long

End Type

Private Declare Function DrawText Lib "user32" Alias "DrawTextA" _

(ByVal hdc As Long, _

ByVal lpStr As String, _

ByVal nCount As Long, _

lpRect As RECT, _

ByVal wFormat As Long) As Long

Private Declare Function SendMessage Lib "user32" Alias "SendMessageA" _

(ByVal hwnd As Long, _

ByVal wMsg As Long, _

ByVal wParam As Long, _

lParam As Any) As Long

Const LB_SETHORIZONTALEXTENT = &H194

Const DT_CALCRECT = &H400

Public Function ListTextWidth(ByRef lstThis As ListBox) As Long

Dim i As Long

Dim tR As RECT

Dim lW As Long

Dim lWidth As Long

Dim lHDC As Long

With lstThis.Parent.Font

.Name = lstThis.Font.Name

.Size = lstThis.Font.Size

.Bold = lstThis.Font.Bold

.Italic = lstThis.Font.Italic

End With

lHDC = lstThis.Parent.hdc

'便历所有的列表项以找到最长的项

For i = 0 To lstThis.ListCount - 1

DrawText lHDC, lstThis.List(i), -1, tR, DT_CALCRECT

lW = tR.Right - tR.Left + 8

If (lW > lWidth) Then

lWidth = lW

End If

Next i

'返回最长列表项的长度(像素)

ListTextWidth = lWidth

End Function

Private Sub Form_Load()

Dim astr As String

Dim i

Dim l As Long

l = List1.FontSize * 20 / Screen.TwipsPerPixelX

For i = 1 To 10

astr = astr + "我们This is a very long item " + Str(i)

Next i

List1.AddItem astr + "aaa"

'加入一个很长的列表项

l = ListTextWidth(List1)

SendMessage List1.hwnd, LB_SETHORIZONTALEXTENT, l, 0

End Sub

首先在Form1中加入一个ListBox控件,然后再将上面的代码加入到Form1的代码窗口中。运行程序,可以看到列表中出现了横向滚动条,而且滚动范围正好是列表项的长度。

3、使列表可以响应用户击键

有时我们需要列表根据用户的敲入字符串自动调整列表的ListIndex到最接近的列表项,就象VB中动态感应用户输入控件属性的编辑器一样。问题的关键是如何在列表中查找含有指定字符串的列表项,使用LB_FINDSTRING消息可以在列表中查找指定字符串。下面是范例:

Private Declare Function SendMessageStr Lib "user32" Alias "SendMessageA" _

(ByVal hwnd As Long, _

ByVal wMsg As Long, _

ByVal wParam As Long, _

ByVal lParam As String) As Long

Const LB_FINDSTRING = &H18F

Dim astr As String

Private Sub Form_KeyPress(KeyAscii As Integer)

Dim l As Long

astr = astr + Chr(KeyAscii)

l = SendMessageStr(List1.hwnd, LB_FINDSTRING, -1, astr)

If l Then

List1.ListIndex = l

End If

End Sub

Private Sub Form_Load()

'向List中加入列表项

For i = 65 To 85

For j = 65 To 85

List1.AddItem Chr(i) + Chr(j)

Next j

Next i

End Sub

Private Sub List1_DblClick()

'清除原来的查找字符串

astr = ""

End Sub

Private Sub List1_KeyPress(KeyAscii As Integer)

'如果按下的是字母键就将击键消息传递到Form1

If ((KeyAscii >= 65 And KeyAscii <= 90) Or (KeyAscii >= 97 _

Or KeyAscii <= 122)) Then

KeyAscii = 0

End If

End Sub

首先在Form1中加入一个ListBox控件,然后再将上面的代码加入到Form1的代码窗口中。并将List1的Sorted属性设置为True。运行程序,在列表中敲入字符,例如"av" "gm",列表就会高亮显示相近的列表项,双击列表就可以清除原来的输入。

在上一篇文章中我向大家介绍了关于ListBox类控件消息的应用,在这一章我将向大家介绍如何利用消息操控TextBox类控件。

1、获得光标所在的行和列

一般的比较完善的文本编辑器一般都有在状态栏中显示当前光标所在行和列的功能。利用SendMessage向TextBox控件发送编辑控件类型消息。也可以实现这样的功能。下面首先来看程序,然后再分析。

首先在VB中建立一个新工程,并在Form1中加入一个TextBox控件和两个Label控件。将TextBox控件的MultiLine属性设置为True。然后在Form1的代码窗口中加入如下代码:

Option Explicit

Private Declare Function SendMessage Lib "user32" Alias "SendMessageW" _

(ByVal hwnd As Long, ByVal wMsg As Long, ByVal wParam As Long, _

lParam As Any) As Long

Private Declare Function SendMessageByRef Lib "user32" Alias "SendMessageA" _

(ByVal hwnd As Long, ByVal wMsg As Long, wParam As Long, _

lParam As Long) As Long

Const EM_LINEFROMCHAR = &HC9

Const EM_LINEINDEX = &HBB

Const EM_GETLINE = &HC4

Const EM_GETSEL = &HB0

Dim iLineX, iLineY As Long

Sub GetCurPos(txtA As TextBox)

Dim l, l1, l2 As Long

Dim astr As String * 256

l = SendMessage(txtA.hwnd, EM_LINEINDEX, -1, 0)

iLineY = SendMessage(txtA.hwnd, EM_LINEFROMCHAR, l, 0)

SendMessageByRef txtA.hwnd, EM_GETSEL, l1, l2

iLineX = l1 - l

Label1.Caption = "列:" + Str(iLineX)

Label2.Caption = "行:" + Str(iLineY)

End Sub

Private Sub Form_Load()

Dim iFile

Dim astr As String

Label1.Height = 300: Label2.Height = 300

Text1.Left = 0: Text1.Top = 0

Text1.Text = ""

Label1.Caption = ""

Label2.Caption = ""

iFile = FreeFile

Open "C:\windows\readme.txt" For Input As #iFile

Do

Line Input #iFile, astr

Text1.Text = Text1.Text + astr + vbCrLf

Loop Until EOF(iFile)

Close iFile

End Sub

Private Sub Form_Resize()

Label1.Top = Me.ScaleHeight - 300

Label2.Top = Me.ScaleHeight - 300

Label1.Left = 0: Label2.Left = 1200

Label1.Width = 1200

Label2.Width = 1200

Text1.Width = Me.ScaleWidth

Text1.Height = Me.ScaleHeight - Label1.Height

End Sub

Private Sub Text1_Click()

GetCurPos Text1

End Sub

Private Sub Text1_KeyUp(KeyCode As Integer, Shift As Integer)

GetCurPos Text1

End Sub

在运行程序前,确保在你的硬盘上有 c:\windows\readme.txt 这个文件。否则程序会出错。然后运行程序。当在编辑文本时,可以看到在窗口底部可以显示当前光标所在的行、列值。在上面的程序中。我们首先发送EM_LINEINDEX消息,发送该消息可以返回某一行的第一个字符在整个文本控件中的位置,如果wParam参数设置为-1,则返回当前行的字符位置。然后发送EM_LINEFROMCHAR,发送该消息可以根据参数wParam指定的字符位置返回该字符所在的行号,文本第一行的位置为0。这样使用这两个消息就获得当前光标所在的行号。要取得列号,首先发送EM_GETSEL消息,发送该消息返回当前被选中文本的起始位置,如果没有文本被选中,则返回当前光标所在字符在文本中的位置。由于上面的EM_LINEINDEX消息返回的是当前行的第一个字符在文本中的位置。所以将两值相减,就是光标所在字符的列位置。在上面的程序中,如果你的文本中有中文字符的话,当你的光标在中文字符中移动一个位置,你会看到标签中的列位置增加了2,这是由于SendMessage发送的消息所得到的结果是不支持中文的,它将一个中文字算做两个字符。这也算是程序中的一个Bug吧(这也就是为什么我要使用EM_GETSEL消息而不直接使用TextBox控件的SelStart属性来获取光标所在字符位置了,因为如果使用SelStart返回的值将一个中文算一个字符,同EM_LINEINDEX返回值相减有可能得到负值).

2、声明API

Public Declare Function SendMessage Lib "user32" Alias "SendMessageA" (ByVal hwnd As Long, ByVal wMsg As Long, ByVal wParam As Long, lParam As Any) As Long '声明的 API

'API常量申明

Public Const WM_SETHOTKEY = &H32

Public Const HOTKEYF_SHIFT = &H1

Public Const HOTKEYF_CONTROL = &H2

Public Const HOTKEYF_ALT = &H4

例子: 用sendmessage 给窗体发送个消息! 让他显示出来

Private Sub Form_Load()

Dim l As Long

Dim wHotkey As Long

wHotkey = (HOTKEYF_ALT Or HOTKEYF_CONTROL) * (2 ^ 8) + 66 '设置热键

l = SendMessage(Me.hwnd, WM_SETHOTKEY, wHotkey, 0) 'WM_SETHOTKEY设置热键消息

MsgBox "只需按Ctl+Alt+B 即可调出窗体"

Me.WindowState = 1 '窗体最小化

End Sub