📄 lion-tut-c34.htm
字号:
<HTML>
<head>
<link rel="stylesheet" href="../../asm.css">
<meta http-equiv="Content-Type" content="text/html; charset=gb2312">
<title>Iczelion's win32 asm tutorial</title>
</head>
<body bgcolor="#FFFFFF" background="../../images/back01.jpg">
<H1 align=center><FONT color=#0000ff face=Tahoma>教程 34: RichEdit 控件:更多的正文操作</FONT></H1>
<P align=left><FONT face=Tahoma size=-1>你将会了解到关于的RichEdit更多的正文操作。特别是你将会学习到如何搜索/替换正文,定位到某一指定的行号。</FONT></P>
<P align=left><FONT face=Tahoma size=-1>下载 <A
href="files/tut34.zip">例子程序</A>.</FONT></P>
<H3 align=left><FONT color=#0000ff
face="Times New Roman, Times, serif">Theory</FONT></H3>
<H3 align=left><FONT color=#990099 face="Arial, Helvetica, sans-serif">Searching
for Text</FONT></H3>
<P align=left><FONT face=Tahoma size=-1>RichEdit 控件具有几种正文操作,搜索指定正文就是其中的一种。搜索正文是通过发送 <FONT color=#006666><B>EM_FINDTEXT</B></FONT> 或者 <FONT
color=#006666><B>EM_FINDTEXTEX</B></FONT> 消息来完成的。这两个消息有一点很小的不同点。</FONT></P>
<BLOCKQUOTE><PRE align="left"><FONT face=Tahoma><B><FONT color=#006666>EM_FINDTEXT</FONT></B>
<FONT color=#000099><B>wParam</B></FONT> == 搜索选项。 可以是下表中的任意组合值。这些选项对 <FONT color=#006666><B>EM_FINDTEXT</B></FONT> 和 <FONT color=#006666><B>EM_FINDTEXTEX</B></FONT> 都是一样的。</FONT></PRE>
<TABLE align=center border=1 cellPadding=3>
<TBODY>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>FR_DOWN</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>如果指定了这个标志值,搜索操作从当前选定的 <FONT color=#0000cc><B>end</B></FONT> 位置开始,直到控件中正文的 <FONT color=#0000cc><B>end</B></FONT> 位置结束(<FONT color=#0000ff><B>向下搜索</B></FONT>)。这个标志仅影响 RichEdit 2.0 和以后版本: <FONT
color=#cc00cc><B>这个是 RichEdit 1.0 的缺省行为</B></FONT>。RichEdit 2.0 或以后版本的缺省行为是在当前选定正文内的从结尾搜索到开始位置(<FONT color=#0000ff><B>向前搜索</B></FONT>)。<BR>概括来说就是,如果你使用 RichEdit 1.0, 无论你做什么都没法影响搜索的方向:它总使用向后搜索。但是如果你使用 RichEdit 2.0 而且你想使用向后搜索的话,你<FONT color=#cc00cc><B>必须</B></FONT>指定这个标志值,否则使用的是向前搜索了。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>FR_MATCHCASE</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>如果指定了这个标志值,搜索操作是大小写敏感的,即区分大小写。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>FR_WHOLEWORD</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>如果设置了这个标志值,搜索操作就搜寻匹配指定搜索串的整个词。</FONT></TD></TR></TBODY></TABLE><PRE><FONT face=Tahoma>实际上,还有更多的标志值,但是它们都是跟非英语系正文操作相关的。
lParam == <FONT color=#006666><B>FINDTEXT</B></FONT> 结构的指针。
<FONT color=#990099><B> FINDTEXT <FONT color=#0000cc>STRUCT</FONT>
chrg <FONT color=#0000cc>CHARRANGE</FONT> <>
lpstrText <FONT color=#0000cc>DWORD</FONT> ?
FINDTEXT <FONT color=#0000cc>ENDS</FONT></B></FONT>
<FONT color=#990099><B>chrg</B></FONT> 是一个 <FONT color=#006666><B>CHARRANGE</B></FONT> 结构,其定义如下:
<FONT color=#cc00cc><B> CHARRANGE <FONT color=#0000cc>STRUCT</FONT>
cpMin <FONT color=#0000cc>DWORD</FONT> ?
cpMax <FONT color=#0000cc>DWORD</FONT> ?
CHARRANGE <FONT color=#0000cc>ENDS</FONT></B></FONT>
<FONT color=#0000cc><B>cpMin</B></FONT> 包含字符数组中第一个字符的字符索引。
<FONT color=#0000cc><B>cpMax</B></FONT> 包含紧跟在字符数组中最后一个字符的字符的字符索引。
基本上,要搜索一个正文串,你必须指定要搜索的字符范围。
<FONT color=#cc0033><B><FONT color=#0000cc>cpMin</FONT> 和 <FONT color=#0000cc>cpMax</FONT> 的具体意义根据搜索是向后还是向前是不同的</B></FONT>。
如果是向后搜索,<FONT color=#0033cc><B>cpMin</B></FONT> 指定搜索的开始字符索引,而 <FONT color=#0000cc><B>cpMax</B></FONT> 则是结束字符索引。
如果是向前搜索,则反过来才对,也就是说 <FONT color=#0000cc><B>cpMin</B></FONT> 包含结束字符索引而<FONT color=#0000cc><B>cpMax</B></FONT> 包含开始字符索引。
<FONT color=#cc00cc><B>lpstrText</B></FONT> 是要搜索的正文串的指针。
<FONT color=#0000cc><B>EM_FINDTEXT</B></FONT> 返回控件中跟搜索串匹配的的正文串的一个字符的索引。如果没找到匹配的则返回 -1。
<FONT color=#006666><B>EM_FINDTEXTEX</B></FONT>
<FONT color=#0000cc><B>wParam</B></FONT> == 搜索选项,跟 <FONT color=#006666><B>EM_FINDTEXT</B></FONT> 的一样。
<FONT color=#000099><B>lParam</B></FONT> == <FONT color=#006666><B>FINDTEXTEX</B></FONT> 结构的指针。
<FONT color=#990099><B> FINDTEXTEX <FONT color=#0000cc>STRUCT</FONT>
chrg <FONT color=#0000cc>CHARRANGE</FONT> <>
lpstrText <FONT color=#0000cc>DWORD</FONT> ?
chrgText <FONT color=#0000cc>CHARRANGE</FONT> <>
FINDTEXTEX <FONT color=#0000cc>ENDS
</FONT></B></FONT><FONT color=#006666><B>FINDTEXTEX</B></FONT> 中开始的两个成员是跟 <FONT color=#006666><B>FINDTEXT</B></FONT> 结构中的一样的。
chrgText 是一个 <FONT color=#006666><B>CHARRANGE</B></FONT> 结构,如果搜索到匹配串的话,其开始/结束字符索引会被填入这个结构中。
<FONT color=#006666><B>EM_FINDTEXTEX</B></FONT> 的返回值跟 <FONT color=#006666><B>EM_FINDTEXT</B></FONT>的是一样的。
<FONT color=#006666><B>EM_FINDTEXT </B></FONT> 跟 <FONT color=#006666><B> EM_FINDTEXTEX</B></FONT> 的不同处是 <FONT color=#006666><B>FINDTEXTEX</B></FONT> 结构有一个另外的<FONT color=#006666><B>chrgText</B></FONT>成员, 如果搜索到匹配串的话,
其开始/结束字符索引会被填入这个成员中。如果我们想对这个正文串进行更多的正文操作的话,有这个就方便多了。</FONT></PRE></BLOCKQUOTE>
<H3><FONT color=#990099 face="Arial, Helvetica, sans-serif">替换/插入正文</FONT><FONT face=Tahoma> </FONT></H3>
<P><FONT face=Tahoma size=-1>RichEdit 控件提供了 <FONT
color=#006666><B>EM_SETTEXTEX</B></FONT> 来进行正文替换/插入操作。这个消息混合了 <FONT
color=#006666><B>WM_SETTEXT</B></FONT> 和 <FONT
color=#006666><B>EM_REPLACESEL</B></FONT> 的功能. 它具有以下语法:</FONT></P><PRE><FONT face=Tahoma> <FONT color=#006666><B>EM_SETTEXTEX</B></FONT>
<FONT color=#0000cc><B>wParam</B></FONT> == <FONT color=#006666><B>SETTEXTEX</B></FONT> 结构的指针。
<FONT color=#990099><B> SETTEXTEX <FONT color=#0000cc>STRUCT</FONT>
flags <FONT color=#0000cc>DWORD</FONT> ?
codepage <FONT color=#0000cc>DWORD</FONT> ?
SETTEXTEX <FONT color=#0000cc>ENDS</FONT></B></FONT>
<FONT color=#990099><B>flags</B></FONT> 可以是以下值的组合:</FONT></PRE>
<TABLE align=center border=1 cellPadding=3>
<TBODY>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ST_DEFAULT</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>删除Undo堆栈,丢弃RTF格式,替换所有的正文。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ST_KEEPUNDO</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>保留Undo堆栈。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ST_SELECTION</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>替换选定正文并且保留RTF格式</FONT></TD></TR></TBODY></TABLE><PRE><FONT face=Tahoma> <FONT color=#990099><B>codepage</B></FONT> 是一个常量,指定你的正文想要的代码页。我们通常简单的使用 <FONT color=#006666><B>CP_ACP</B></FONT>。</FONT></PRE>
<H3><FONT color=#990099 face=Tahoma>正文选择</FONT></H3>
<P><FONT face=Tahoma size=-1>我们可以使用消息 <FONT
color=#000099><B><FONT color=#006666>EM_SETSEL</FONT></B></FONT> 或者 <FONT
color=#006666><B>EM_EXSETSEL</B></FONT> 来编程选择正文.其中任意的一个都可以工作的很好。要使用哪一个消息要根据可用的字符索引格式来选择。如果它们保存在一个 <FONT color=#006666><B>CHARRANGE</B></FONT> 结构中,则使用 <FONT color=#006666><B>EM_EXSETSEL</B></FONT>更容易实现。</FONT><FONT
face=Tahoma><BR></FONT></P><PRE><FONT face=Tahoma> <FONT color=#006666><B>EM_EXSETSEL</B></FONT>
<FONT color=#0000cc><B>wParam</B></FONT> == 没有使用,必须为 0 。
<FONT color=#0000cc><B>lParam</B></FONT> == <FONT color=#006666><B>CHARRANGE</B></FONT> 结构的指针,包含想要选定的正文字符范围。</FONT></PRE>
<H3><FONT color=#990099 face=Tahoma>事件通知</FONT></H3>
<P><FONT face=Tahoma size=-1>在使用多行Edit控件时,你必须子类化它以便得到输入信息象鼠标/键盘事件等。RichEdit 控件提供了一个更好的方案,它可以把这些消息通知父窗口。为了注册得到通知消息,父窗口发送 <FONT color=#006666><B>EM_SETEVENTMASK</B></FONT> 消息给 RichEdit 控件,指定它对哪些消息感兴趣。 <FONT
color=#006666><B>EM_SETEVENTMASK </B></FONT>具有以下的语法:</FONT></P><PRE><FONT face=Tahoma> <B><FONT color=#006666>EM_SETEVENTMASK</FONT></B>
<FONT color=#000099><B>wParam</B></FONT> == 没有使用,必须为 0 。
<FONT color=#0000cc><B>lParam</B></FONT> == 事件掩码值。他可以是以下表格里标志值的任意组合。</FONT></PRE>
<TABLE align=center border=1 cellPadding=3>
<TBODY>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_CHANGE</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#000099><B>EN_CHANGE</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_CORRECTTEXT</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#000099><B>EN_CORRECTTEXT</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_DRAGDROPDONE</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#0000cc><B>EN_DRAGDROPDONE</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_DROPFILES</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#0000cc><B>EN_DROPFILES</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_KEYEVENTS</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>为键盘消息发送 <FONT
color=#0000cc><B>EN_MSGFILTER </B></FONT>通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_LINK</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1><B>Rich Edit 2.0 或以后版本:</B>
当鼠标在具有 <FONT color=#990099><B>CFE_LINK</B></FONT> 风格的正文上面移过,而且执行了一个或几个鼠标动作时,就发送 <FONT color=#0000cc><B>EN_LINK</B></FONT> 通知。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_MOUSEEVENTS</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>为鼠标消息发送 <FONT
color=#0000cc><B>EN_MSGFILTER</B></FONT> 通知。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif"
size=-1>ENM_OBJECTPOSITIONS</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#0000cc><B>EN_OBJECTPOSITIONS</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_PROTECTED</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT color=#0000cc><B>
EN_PROTECTED</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_REQUESTRESIZE</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT color=#0000cc><B>
EN_REQUESTRESIZE</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_SCROLL</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#0000cc><B>EN_HSCROLL</B></FONT> 和 <FONT
color=#000099><B>EN_VSCROLL</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_SCROLLEVENTS</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>为鼠标滑轮发送 <FONT
color=#0000cc><B>EN_MSGFILTER</B></FONT> 通知。</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><B><FONT face="MS Sans Serif" size=-1>ENM_SELCHANGE</FONT></B></TD>
<TD><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#000099><B>EN_SELCHANGE</B></FONT> 通知</FONT></TD></TR>
<TR bgColor=#ffffcc>
<TD><FONT face="MS Sans Serif" size=-1><B>ENM_UPDATE</B></FONT></TD>
<TD>
<P><FONT face="MS Sans Serif" size=-1>发送 <FONT
color=#0000cc><B>EN_UPDATE </B></FONT>通知<BR><B>Rich Edit 2.0
和以后版本: </B>这个标志值会被忽略,而经常发送 <FONT
color=#0000cc><B>EN_UPDATE</B></FONT> 通知。然而如果 RichEdit 3.0 模拟 RichEdit 1.0的话,你必须使用这个标志值来发送 <FONT color=#000099><B>EN_UPDATE</B></FONT> 通知
</FONT></P></TD></TR></TBODY></TABLE>
<P><FONT face=Tahoma size=-1>上面的所有通知都被做为 <FONT
color=#006666><B>WM_NOTIFY </B></FONT> 消息来发送:你必须检查 <FONT color=#006666><B>NMHDR</B></FONT> 结构的 code 成员来得到通知消息。譬如,如果你想注册得到鼠标消息(也就是说,你想提供一给上下文相关的弹出菜单), 你需要象下面这样做:</FONT></P><PRE><FONT face=Tahoma><B> invoke SendMessage,hwndRichEdit,<FONT color=#006666>EM_SETEVENTMASK</FONT>,0,<FONT color=#0000ff>ENM_MOUSEEVENTS</FONT>
.....
.....
WndProc proc hWnd:DWORD, uMsg:DWORD, wParam:DWORD, lParam:DWORD
.....
....
.elseif uMsg==WM_NOTIFY
push esi
mov esi,lParam
assume esi:ptr NMHDR
<FONT color=#ff0033>.if [esi].code==EN_MSGFILTER</FONT>
....
[ do something here]
....
.endif
pop esi</B></FONT></PRE>
<H3><FONT face="Times New Roman, Times, serif">例子:</FONT></H3>
<P><FONT face=Tahoma size=-1>下面的例子是第33篇指南里的 IczEdit 的改进版。它为程序增加了搜索/替换功能和加速键。同时它处理鼠标消息,点右键时会出现一个弹出菜单。</FONT></P><PRE><FONT face=Tahoma><B>.386
.model flat,stdcall
option casemap:none
include \masm32\include\windows.inc
include \masm32\include\user32.inc
include \masm32\include\comdlg32.inc
include \masm32\include\gdi32.inc
include \masm32\include\kernel32.inc
includelib \masm32\lib\gdi32.lib
includelib \masm32\lib\comdlg32.lib
includelib \masm32\lib\user32.lib
includelib \masm32\lib\kernel32.lib
WinMain proto :DWORD,:DWORD,:DWORD,:DWORD
.const
IDR_MAINMENU equ 101
IDM_OPEN equ 40001
IDM_SAVE equ 40002
IDM_CLOSE equ 40003
IDM_SAVEAS equ 40004
IDM_EXIT equ 40005
IDM_COPY equ 40006
IDM_CUT equ 40007
IDM_PASTE equ 40008
IDM_DELETE equ 40009
IDM_SELECTALL equ 40010
IDM_OPTION equ 40011
IDM_UNDO equ 40012
IDM_REDO equ 40013
IDD_OPTIONDLG equ 101
IDC_BACKCOLORBOX equ 1000
IDC_TEXTCOLORBOX equ 1001
IDR_MAINACCEL equ 105
IDD_FINDDLG equ 102
IDD_GOTODLG equ 103
IDD_REPLACEDLG equ 104
IDC_FINDEDIT equ 1000
IDC_MATCHCASE equ 1001
IDC_REPLACEEDIT equ 1001
IDC_WHOLEWORD equ 1002
IDC_DOWN equ 1003
IDC_UP equ 1004
IDC_LINENO equ 1005
IDM_FIND equ 40014
IDM_FINDNEXT equ 40015
IDM_REPLACE equ 40016
IDM_GOTOLINE equ 40017
IDM_FINDPREV equ 40018
RichEditID equ 300
.data
ClassName db "IczEditClass",0
AppName db "IczEdit version 2.0",0
RichEditDLL db "riched20.dll",0
RichEditClass db "RichEdit20A",0
NoRichEdit db "Cannot find riched20.dll",0
ASMFilterString db "ASM Source code (*.asm)",0,"*.asm",0
db "All Files (*.*)",0,"*.*",0,0
OpenFileFail db "Cannot open the file",0
WannaSave db "The data in the control is modified. Want to save it?",0
FileOpened dd FALSE
BackgroundColor dd 0FFFFFFh <FONT color=#009999>; default to white</FONT>
TextColor dd 0 <FONT color=#009999> ; default to black</FONT>
hSearch dd ? <FONT color=#006666> ; handle to the search/replace dialog box</FONT>
hAccel dd ?
.data?
hInstance dd ?
hRichEdit dd ?
hwndRichEdit dd ?
FileName db 256 dup(?)
AlternateFileName db 256 dup(?)
CustomColors dd 16 dup(?)
FindBuffer db 256 dup(?)
ReplaceBuffer db 256 dup(?)
uFlags dd ?
findtext FINDTEXTEX <>
.code
start:
mov byte ptr [FindBuffer],0
mov byte ptr [ReplaceBuffer],0
invoke GetModuleHandle, NULL
mov hInstance,eax
invoke LoadLibrary,addr RichEditDLL
.if eax!=0
mov hRichEdit,eax
invoke WinMain, hInstance,0,0, SW_SHOWDEFAULT
invoke FreeLibrary,hRichEdit
.else
invoke MessageBox,0,addr NoRichEdit,addr AppName,MB_OK or MB_ICONERROR
.endif
invoke ExitProcess,eax
WinMain proc hInst:DWORD,hPrevInst:DWORD,CmdLine:DWORD,CmdShow:DWORD
LOCAL wc:WNDCLASSEX
LOCAL msg:MSG
LOCAL hwnd:DWORD
mov wc.cbSize,SIZEOF WNDCLASSEX
mov wc.style, CS_HREDRAW or CS_VREDRAW
mov wc.lpfnWndProc, OFFSET WndProc
mov wc.cbClsExtra,NULL
mov wc.cbWndExtra,NULL
push hInst
pop wc.hInstance
mov wc.hbrBackground,COLOR_WINDOW+1
mov wc.lpszMenuName,IDR_MAINMENU
mov wc.lpszClassName,OFFSET ClassName
invoke LoadIcon,NULL,IDI_APPLICATION
mov wc.hIcon,eax
mov wc.hIconSm,eax
invoke LoadCursor,NULL,IDC_ARROW
mov wc.hCursor,eax
invoke RegisterClassEx, addr wc
INVOKE CreateWindowEx,NULL,ADDR ClassName,ADDR AppName,\
WS_OVERLAPPEDWINDOW,CW_USEDEFAULT,\
CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,NULL,NULL,\
hInst,NULL
mov hwnd,eax
invoke ShowWindow, hwnd,SW_SHOWNORMAL
invoke UpdateWindow, hwnd
invoke LoadAccelerators,hInstance,IDR_MAINACCEL
mov hAccel,eax
.while TRUE
invoke GetMessage, ADDR msg,0,0,0
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -