news 2026/6/10 3:31:02

汇编语言全接触-63.Win32汇编教程七

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
汇编语言全接触-63.Win32汇编教程七

在这儿下载本节的所有源程序。

有关控件子类化

说到类,大家可能马上就想到C++,的确,类首先是在C中提出的,但是,这个概念在 Win32Asm 中仍然适用,因为在类的思路是这样的:先假设某个对象有不同的属性,当一个新的对象的某个属性和上面所说的对象有些不同,而别的属性一模一样,那么实际上除了处理这个属性的代码有些不同外,别的代码完全可以使用前面的对象的代码。在具体的应用中,我举个例子,比如说我们定义一个 "edit" 控件,那么这个控件的行为是由 Windows 内定的,因为它的窗口过程是在 Windows 系统内部的,但假如我们想编一个有语法检查的 "edit" 控件,是否我们除了语法检查的代码以外,还要编写很多代码来实现老的 "edit" 控件一模一样的功能呢?答案当然是否定的,实际上,我们可以截获一个标准 "edit"控件的 WM_CHAR 消息,检查键入的键并做处理,别的消息可以传给原来的窗口过程。示意如下:

在子类化之前: Windows => edit 控件的窗口过程

在子类化之后: Windows => 我们的过程代码 => edit 控件的窗口过程

在Windows 的 API 中有个函数可以用来实现这个功能,那就是 SetWindowLong PROTO hWnd,nIndex,dwNewLong ,参数的意思是 hWnd 是你要改变的窗口句柄,nIndex 是我们要改变窗口的什么属性,它的值可以是 GWL_EXSTYLE:改变窗口风格,GWL_WNDPROC:设置窗口的新的过程,这正是我们感兴趣的,还有是 GWL_USERDATA 这是窗口自定义的一个32位的数据。dwNewLong 是新的值,还有一个 API 是用来调用原来的窗口过程的,叫 CallWindwoProc PROTO lpPrevWndFunc,hWnd,Msg,wParam,lParam。

我们在使用时有下面的过程:

用 SetWindowLong,hWnd,GWL_WNDPROC,addr _NewProcAddress 设置我们自己的代码的地址,API 返回原来的过程地址

用 SetWindowLong,hWnd,GWL_USERDATA,eax 把原来的过程地址保存在自定义数据中。

这样,所有消息会先送到我们的过程中,然后在我们自己的过程中:

对要处理的消息进行处理,如果不希望原来的过程再处理,那么返回。

对自己不处理的消息,调用原来的窗口过程处理,并把返回值返回。方法是:

用 invoke GetWindowLong,hWnd,GWL_USERDATA 取出自定义数据中保存的原过程地址

用 invoke CallWindowProc,eax,hWnd,uMsg,wParam,lParam 调用原过程 UINT uStructSize}

本节教程提供了一个源程序,它是实现对话框中的文本的 URL 连接过程,我们看到有的程序中的文本是蓝色的,有下划线,然后鼠标移动到上面会变手型,就象浏览器中的超联结一样,而且按下会自动连接到网站上,仔细想想,我们并没有一个标准的控件或 API 来实现这样一个功能,因为这首先是一个文本,所以我们可以对这个文本进行子类化,处理它的WM_LBUTTONUP 消息来实现按下自动连上网站的功能;处理 WM_SETCURSOR 消息来让鼠标移到上面改变光标,具体源程序如下:

源程序 - 资源文件

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

;************************************************

#include

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; Icon 1000 开始

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

#define IDI_MAIN 1000

#define IDC_HANDLE 2000

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; 对话框 3000 开始

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

#define DLG_ABOUT 3000

#define ID_EMAIL 3001

#define ID_HOMEPAGE 3002

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; 资源定义开始

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

IDI_MAIN ICON "Main.ico"

IDC_HANDLE CURSOR "Handle.cur"

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

DLG_ABOUT DIALOG DISCARDABLE 50, 50, 160, 30

STYLE DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU

CAPTION "URL 联结文本演示 - by 罗云彬"

FONT 9, "宋体"

BEGIN

LTEXT "我的主页: ",-1, 5,5,54,9

LTEXT "http://asm.yeah.net", ID_HOMEPAGE, 55,5,80,9

LTEXT "我的E-mail: ", -1, 5,17,54,9

LTEXT "bigluo@telekbird.com.cn",ID_EMAIL, 55,17,95,9

END

源程序 - 汇编源文件

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; 是否包括调试代码

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

DEBUG = 0

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; Programmed by 罗云彬, bigluo@telekbird.com.cn

; Website: http://asm.yeah.net

; LuoYunBin's Win32 ASM page (罗云彬的编程乐园)

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; 版本信息

; 窗口子类化演示程序 Ver 1.0

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

.386

.model flat, stdcall

option casemap :none ; case sensitive

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; Include 数据

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

include windows.inc

include user32.inc

include kernel32.inc

include comctl32.inc

include comdlg32.inc

include shell32.inc

include gdi32.inc

includelib user32.lib

includelib kernel32.lib

includelib comctl32.lib

includelib comdlg32.lib

includelib shell32.lib

includelib gdi32.lib

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; Equ 数据

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

IDI_MAIN equ 1000 ;icon

IDC_HANDLE equ 2000 ;handle cursor

DLG_ABOUT equ 3000 ;dialog - about

ID_EMAIL equ 3001

ID_HOMEPAGE equ 3002

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

; 数据段

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

.data?

hInstance dd ?

hIcon dd ?

szBuffer db 256 dup (?)

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

_HyperLinkProc proto :DWORD,:DWORD,:DWORD,:DWORD

;>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>

.data

hCursorHandle dd ?

szHomePage db "http://asm.yeah.net",0

szEmail db "mailto:bigluo@telekbird.com.cn"

db "?subject=嗨!我喜欢你的程序!",0

.code

if DEBUG

include Debug.asm

endif

;********************************************************************

; 关于对话框中超级连接的窗口程序

;********************************************************************

_HyperLinkProc proc hWnd:DWORD,uMsg:DWORD,wParam:DWORD,lParam:DWORD

mov eax,uMsg

.if eax == WM_LBUTTONUP

invoke GetDlgCtrlID,hWnd

.if eax == ID_HOMEPAGE

invoke ShellExecute,0,0,offset szHomePage,0,0,0

.elseif eax == ID_EMAIL

invoke ShellExecute,0,0,offset szEmail,0,0,0

.endif

.elseif eax == WM_NCHITTEST

;将 WM_NCHITTEST 返回 TRUE 可以接收鼠标动作,实现按下功能 !

mov eax,TRUE

ret

.elseif eax == WM_SETCURSOR

invoke SetCursor,hCursorHandle

.else

invoke GetWindowLong,hWnd,GWL_USERDATA

invoke CallWindowProc,eax,hWnd,uMsg,wParam,lParam

ret

.endif

xor eax,eax

ret

_HyperLinkProc endp

;********************************************************************

; 对话框窗口主程序

;********************************************************************

AboutDialogProc proc uses ebx edi esi, \

hWnd:DWORD,uMsg:DWORD,wParam:DWORD,lParam:DWORD

local @stWindow:RECT

local @dwWidth:DWORD,@dwHeight:DWORD

local @hWinTemp:DWORD

local @stFont:LOGFONT,@hFontOutput:DWORD

mov eax,uMsg

.if eax == WM_CLOSE

invoke EndDialog,hWnd,NULL

.elseif eax == WM_INITDIALOG

invoke GetModuleHandle,NULL

invoke LoadCursor,eax,IDC_HANDLE

mov hCursorHandle,eax

invoke GetDlgItem,hWnd,ID_HOMEPAGE

mov @hWinTemp,eax

invoke SetWindowLong,@hWinTemp,GWL_WNDPROC,addr _HyperLinkProc

invoke SetWindowLong,@hWinTemp,GWL_USERDATA,eax

invoke GetDlgItem,hWnd,ID_EMAIL

mov @hWinTemp,eax

invoke SetWindowLong,@hWinTemp,GWL_WNDPROC,addr _HyperLinkProc

invoke SetWindowLong,@hWinTemp,GWL_USERDATA,eax

.elseif eax == WM_CTLCOLORSTATIC

invoke GetDlgCtrlID,lParam

.if eax == ID_HOMEPAGE || eax == ID_EMAIL

invoke SendMessage,lParam,WM_GETFONT,0,0

mov @hFontOutput,eax

invoke GetObject,@hFontOutput,sizeof LOGFONT,addr @stFont

mov @stFont.lfUnderline,TRUE

invoke CreateFontIndirect,addr @stFont

mov @hFontOutput,eax

invoke SelectObject,wParam,eax

invoke SetTextColor,wParam,Blue

invoke GetSysColor,COLOR_MENU

invoke SetBkColor,wParam,eax

invoke DeleteObject,@hFontOutput

;********************************************************************

; 注意此处一定要把StockOject的返回值返回,否则无法显示颜色

;********************************************************************

invoke GetStockObject,HOLLOW_BRUSH

.else

mov eax,FALSE

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/9 21:05:29

JS Math.floor与四舍五入的区别,别再误用了

处理数字时,很多JavaScript开发者会误用Math.floor来进行四舍五入,这是一个常见的概念混淆。Math.floor方法的功能是向下取整,即无条件舍去小数部分,而四舍五入则需要根据小数部分的值进行判断。理解这两者的根本区别,…

作者头像 李华
网站建设 2026/6/9 21:07:19

【Asyncio定时器深度解析】:掌握高效异步任务调度的5大核心技巧

第一章:Asyncio定时器的核心概念与运行机制在Python的异步编程生态中,asyncio库提供了强大的并发处理能力。虽然标准库未直接提供“定时器”类型,但开发者可通过事件循环的调度机制实现精准的延迟执行与周期性任务触发。事件循环与延迟调度 a…

作者头像 李华
网站建设 2026/6/9 22:36:22

日志文件越积越大怎么办?Python自动轮转实践方案曝光

第一章:日志文件越积越大怎么办?Python自动轮转实践方案曝光在长时间运行的服务中,日志文件会持续增长,若不加以管理,可能迅速耗尽磁盘空间,甚至导致服务异常。为解决这一问题,Python 提供了内置…

作者头像 李华
网站建设 2026/6/9 19:47:26

揭秘Gradio音频交互黑科技:3步实现在线语音识别与实时处理

第一章:Gradio音频处理功能概述Gradio 是一个轻量级的 Python 库,专为快速构建机器学习模型的交互式 Web 界面而设计。其对音频数据的支持尤为出色,能够轻松实现音频上传、播放、实时录制和预处理等功能,适用于语音识别、音频分类…

作者头像 李华
网站建设 2026/6/9 19:52:13

揭秘FastAPI中间件设计原理:如何构建高性能应用拦截机制

第一章:揭秘FastAPI中间件设计原理:如何构建高性能应用拦截机制FastAPI 作为现代 Python Web 框架的代表,其高性能特性很大程度上得益于灵活且高效的中间件系统。中间件在请求进入路由处理前和响应返回客户端前提供拦截与处理能力&#xff0c…

作者头像 李华
网站建设 2026/6/9 19:46:18

GitHub镜像gist分享VoxCPM-1.5-TTS-WEB-UI配置片段

VoxCPM-1.5-TTS-WEB-UI 配置解析:一键部署高质量语音合成的工程实践 在AI语音技术加速落地的今天,一个现实问题始终困扰着开发者:如何在有限资源下快速验证TTS(文本转语音)模型的实际效果?传统方案往往需要…

作者头像 李华