用WIN汇编开发桌面报时工具

阅读更多

用WIN汇编开发桌面报时工具

用WIN32汇编语言开发出来的WINDOWS程序具有执行效率高、占用空间小等特点。

一、  软件简介及相应开发工具

桌面报时工具主要完成以下两个功能:

1、     F12热键语音报时、语音整点报时。

2、  在屏幕中上方直接显示时间,并不被任何窗口挡住。

3、  程序运行后最小化为任务栏图标。

笔者以WIN汇编为开发语言,以MASM32为开发工具,因为它不但可以免费获取和使用,而且自带IDE编程环境,读者可访问其官方网站下载MASM32开发包,官方网站为:http://www.movsd.com/

二、设计思路

1、软件采用标准的自定义对话框架构实现。

2、采用微软的TTS语音引擎实现语音报时。 TTS基于COM,因此开发使用Microsoft SpeechSDK,直接调用SAPI实现语音朗读。

3、使用HOOK技术,截获F12按键进行语音报时。因此,笔者单独编写了一个DLL文件,捕捉F12按键。

由此可见,该报时工具虽小,但五脏俱全,涉及对话框消息循环、HOOK技术、COM接口、DLL文件编写、主要API的调用等技术。

三、实现代码

   1、主程序代码

     .586

     .model flat, stdcall

     option casemap :none  

 

;#########################################################################

;首先声明头文件

     include windows.inc

     include user32.inc

     include kernel32.inc

     include gdi32.inc

     include masm32.inc

     include shell32.inc

     include ole32.inc

     include d:\masm32\com\include\oaidl.inc

     include timersdll.inc

     

     includelib user32.lib

     includelib kernel32.lib

     includelibgdi32.lib

     includelib masm32.lib

     includelib shell32.lib

     includelib ole32.lib

     includelib timersdll.lib

;#########################################################################     

;声明宏(主要是控件ID和菜单ID)

IDC_ALARM equ 101

 IDC_CANCEL equ 102

 IDC_OK equ 103

 IDC_ABOUT equ 104

 IDC_CHECKBS equ 105

 IDC_CHECKSHOWTIMER equ 106

 IDR_MAINFRAME equ 107

 IDR_MAINMENU equ 108

 MENUID_SYSTEMSET equ 109

 MENUID_SPEAKTIMER equ 110

 MENUID_ABOUT equ 111

 MENUID_EXIT equ 112

 DIALOG_MAIN equ 1 

 IDT_TIMER equ 1

 WM_NOTIFYICONN equ WM_USER+0

 WM_CLOCKALARM equWM_USER+100h     

;#########################################################################

;Microsoft Speech SDK没有相应的适应于汇编语言的INC头文件,笔者在代码

;文件中直接声明TTS相关虚函数。

;定义ISpVoice的虚函数接口表(根据SDK中的sapi.h中ISpVoiceVtbl的C++定义改

;成MASM32定义)

ISpVoice STRUCT DWORD

QueryInterface comethod3 ?

AddRef comethod1 ?

Release  comethod1 ?

SetNotifySink comethod2 ?

SetNotifyWindowMessage  comethod5 ?

SetNotifyCallbackFunction  comethod4 ?

SetNotifyCallbackInterface  comethod4 ?

SetNotifyWin32Event comethod1 ?

WaitForNotifyEvent comethod1 ?

GetNotifyEventHandle  comethod1 ?

SetInterest  comethod3 ?

  GetEvents  comethod4 ?

  GetInfo comethod2 ?

  SetOutput  comethod3 ?

  GetOutputObjectToken comethod2 ?

  GetOutputStream comethod2 ?

  Pause comethod1 ?

  Resume comethod1 ?

  SetVoice comethod2 ?

  GetVoice comethod2 ?

  Speak  comethod4 ?

  SpeakStream comethod4 ?

  GetStatus  comethod3 ?

  Skip comethod4 ?

  SetPriority comethod2 ?

  GetPriority comethod2 ?

  SetAlertBoundary comethod2 ?

  GetAlertBoundary comethod2 ?

  SetRate comethod2 ?

  GetRate comethod2 ?

  SetVolume comethod2 ?

  GetVolume comethod2 ?

  WaitUntilDone comethod2 ?

  SetSyncSpeakTimeout comethod2 ?

  GetSyncSpeakTimeout comethod2 ?

  SpeakCompleteEvent comethod1 ?

  IsUISupported comethod5 ?

  DisplayUI comethod6 ?

ISpVoice ENDS

;#########################################################################

;可变数据段

   .data?

      szbufferprev db 20 dup(?)

      szbuffernow db 20 dup(?)

      hWinMain DWORD ?

      idTIMER DWORD ?

      CommandLine DWORD ?

      hInstance DWORD ?

      hicon HICON ?

      hmenu DWORD ?

      hlib DWORD ?

      timershowx DWORD ?

      timershowy DWORD ?

      showrect RECT <>

      

   ;常量数据段

   .data

      szClassName db "Timers_Class",0

            szdateformat  db"yyyy年MM月dd日",0

            sztimeformat db "hh点mm分ss秒",0

            stnidstatus NOTIFYICONDATA <>

             ;======

      isintpointalarmDWORD 1;0-表示整点不报时,1-整点报时

      isshowtimer DWORD 1;0-表示不显示,1-显示

      isintpoint  DWORD 0;0-表示不是整点,1-是整点

      ;======

             sztooltip db "桌面报时工具",0

             szabout db "桌面报时工具由WIN32汇编开发",0

      szmscap db "错误",0

      szmstext1 db "无法在桌面上显示!",0

      szmstext2 db "无法得到全屏DC!",0

      ;======

    

      szprevdate db 50 dup(?)

      szprevtime db 50 dup(?)

      sznowdate db 50 dup(?)

      sznowtime db 50 dup(?)

      

      szbegin db "桌面报时工具为您报时"

      sztext db 100 dup(?)

 

;#########################################################################

;代码段

    .code

_showtext  proto :DWORD

_ProcDlgMain proto :DWORD,:DWORD,:DWORD,:DWORD

_geterrno proto :DWORD,:DWORD

_speaktext proto :DWORD

_WinMain proto :DWORD,:DWORD,:DWORD,:DWORD

_statusicon  proto :DWORD

_configload proto :DWORD

start:

;程序的入口

       invoke GetModuleHandle, NULL

       mov hInstance, eax

 

       invoke GetCommandLine

       mov CommandLine, eax

 

       invoke _WinMain,hInstance,NULL,CommandLine,SW_SHOWDEFAULT

       invoke ExitProcess,eax

 

;#########################################################################

;得到当前时间子函数

_getnow proc

       invoke lstrcpy,addr szprevtime,addr sznowtime

       invoke lstrcpy,addr szprevdate,addr sznowdate

       invoke GetDateFormat,NULL,NULL,NULL,addr szdateformat,addrsznowdate,50

 

 invokeGetTimeFormat,LOCALE_USER_DEFAULT,TIME_FORCE24HOURFORMAT,NULL,addrsztimeformat,addr sznowtime,50

       invoke lstrcpy,addr sztext,addr sznowdate

    invokelstrcat,addr sztext,addr sznowtime

    

    invokeszLeft,addr szprevtime,addr szbufferprev,2

    invokeszLeft,addr sznowtime,addr szbuffernow,2

    invokelstrcmp,addr szbufferprev,addr szbuffernow

    testeax,eax

    je @f

    mov isintpoint,1

    @@:

    ret

_getnow endp

;#########################################################################

;显示时间子函数

;参数说明:lpsztex:要显示的文本的起始地址

_showtext proc  uses  ebxlpsztext:DWORD

    LOCAL@Desktopdc:HDC

    LOCAL@dwtextcolor,@dwbkcolor

    mov@dwtextcolor,00010000h

    mov@dwbkcolor,0000FF00h

    invokeGetWindowDC,NULL

    cmpeax,0

    jne@f

    invokeMessageBox,NULL,offset szmstext2,offset szmscap,MB_ICONERROR

    @@:

    mov@Desktopdc,eax

    invokelstrlen,lpsztext

    movebx,eax

    invokeSetBkColor,@Desktopdc,@dwbkcolor

    invokeSetTextColor,@Desktopdc,@dwtextcolor

    invokeTextOut,@Desktopdc,timershowx,timershowy,lpsztext,ebx

    cmpeax,0

    jne@f

    invokeMessageBox,NULL,offset szmstext1,offset szmscap,MB_ICONERROR

    @@:

    invokeReleaseDC,NULL,HDC

    ret

_showtext endp

;#########################################################################

_showtime proc

       invoke _showtext,offset sztext

     ret

_showtime endp

;#########################################################################



;语音朗读文本子函数

;参数说明: pszspeakansi:要朗读的文本起始地址

_speaktext proc uses edx ebx pszspeakansi:DWORD

       .data

          szspeaktext db 200dup(?),0

    szerrtext db "错误号:"

    szerrtextno db 20 dup(?),0

      ppVoice  DWORD ?

     

   

          CLSID_SpVoice GUID<96749377H,3391H,11D2H,<9EH,0E3H,00H,0C0H,4FH,79H,73H,96H>>

          IID_ISpVoice GUID<6C44DF74H,72B9H,4992H,<0A1H,0ECH,0EFH,99H,6EH,04H,22H,0D4H>>

      

 

  .code 

               

    invoke lstrlen,pszspeakansi

    mov ebx,eax

          invokeMultiByteToWideChar,CP_ACP,MB_PRECOMPOSED,pszspeakansi,-1,addrszspeaktext,ebx

      invoke CoInitialize,NULL

     ;#######################################################################

     ;使用pVoice接口

      invoke CoCreateInstance,addr CLSID_SpVoice,NULL,CLSCTX_ALL,addrIID_ISpVoice,addr ppVoice

     .IF_FAILED

         invoke _geterrno,eax,addr szerrtextno

         invoke MessageBox,NULL,addr szerrtext,addr szmscap,MB_ICONERROR

         jmp@f

     .endif

             mov eax,ppVoice

                      mov edx,[eax]

           invoke (ISpVoice PTR[edx]).SetVoice,ppVoice,NULL

           .IF_SUCCEEDED

                              mov eax,ppVoice

                              mov edx,[eax]

               invoke(ISpVoice PTR[edx]).SetRate,ppVoice,3

               .IF_FAILED

                 invoke _geterrno,eax,addr szerrtextno

                 invoke MessageBox,NULL,addr szerrtext,addr szmscap,MB_ICONERROR

              .endif

                    mov eax,ppVoice

                              mov edx,[eax]

               invoke(ISpVoice PTR[edx]).Speak,ppVoice,addr szspeaktext,0,NULL

               .IF_FAILED

                 invoke _geterrno,eax,addr szerrtextno

                 invoke MessageBox,NULL,addr szerrtext,addr szmscap,MB_ICONERROR

              .endif

           .endif

    

    mov eax,ppVoice

      mov edx,[eax]

      invoke (ISpVoice PTR[edx]).Release,ppVoice

    @@:

    call CoUninitialize

       ret

_speaktext endp

;#########################################################################

;对话框消息处理函数

_ProcMain proc uses ebx hWnd,uMsg,wParam,lParam

  LOCAL @stpos:POINT

  .if uMsg==WM_TIMER

      mov eax,wParam

     .if eax==IDT_TIMER

                call _getnow

                .if isshowtimer==1

                   call_showtime

               .endif

                .if isintpointalarm==1 &&isintpoint==1

                     invoke _speaktext,addr szbegin

                     mov isintpoint,0

         .endif

      .endif      

.elseif uMsg==WM_NOTIFYICONN

     moveax,wParam

     .if eax== IDR_MAINFRAME

                      mov eax,lParam

                      movzx eax,ax

                      .if eax== WM_RBUTTONUP

                         invoke GetCursorPos,addr @stpos

                         invokeTrackPopupMenu,hmenu,TPM_LEFTALIGN,@stpos.x,@stpos.y,NULL,hWnd,NULL

                 .endif

              .endif

.elseif uMsg==WM_CLOCKALARM

     invoke_speaktext,addr szbegin 

  .elseif uMsg==WM_COMMAND

    moveax,wParam

    movzxeax,ax

     .ifeax==MENUID_EXIT

                 invoke InvalidateRect,NULL,NULL,NULL

       invoke    UnInstallHook,hWnd

                invoke KillTimer,hWnd,idTIMER

                invoke _statusicon,NIM_DELETE

                            invoke DestroyWindow,hWnd

              .elseif eax==MENUID_SYSTEMSET

                            invoke _configload,0

                            invoke ShowWindow,hWinMain,SW_SHOWNORMAL

              .elseif eax==MENUID_ABOUT

                  invokeMessageBox,NULL,addr szabout,addr sztooltip,NULL

              .elseif eax==MENUID_SPEAKTIMER

        invoke_speaktext,addr szbegin

    .elseifeax==IDC_CANCEL

          invokeShowWindow,hWinMain,SW_HIDE      

              .elseif eax==IDC_OK

                      call _configsave

                      invokeShowWindow,hWinMain,SW_HIDE      

    .endif

      .elseif uMsg==WM_DESTROY

              invoke PostQuitMessage,NULL

.else

         invoke DefDlgProc,hWnd,uMsg,wParam,lParam

         ret

  .endif

  xor eax,eax

  ret

_ProcMain endp

;#########################################################################   ;取得COM接口错误号函数,可以根据错误号在winerror.h中查到错误类型

_geterrno proc szerrhex:DWORD,pszerrtext:DWORD

   push eax

   invokedw2hex,szerrhex,pszerrtext

   pop eax

   ret

_geterrno endp 

;#########################################################################   ; 状态栏图标操作

_statusicon  proc operation:DWORD

invoke Shell_NotifyIcon,operation,addr stnidstatus

ret

_statusiconendp     

;#########################################################################   ;主窗口消息循环

_WinMain proc hInst:DWORD,hPrevInst:DWORD,CmdLine:DWORD,CmdShow:DWORD

       LOCAL @stwc:WNDCLASSEX

       LOCAL @stmsg:MSG

       LOCAL @CommandLine:DWORD

       [email protected],        sizeof WNDCLASSEX

       [email protected],         CS_HREDRAW or CS_VREDRAW

       [email protected],   offset _ProcMain

       [email protected],    NULL

       [email protected],    DLGWINDOWEXTRA

       push hInst

       pop @stwc.hInstance

       mov @stwc.hbrBackground,  COLOR_BTNFACE+1

       mov @stwc.lpszMenuName,  NULL

       mov @stwc.lpszClassName,  offset szClassName

       invoke LoadIcon,hInst, IDR_MAINFRAME

       mov hicon,eax

       push hicon

       pop @stwc.hIcon

       push hicon

       pop @stwc.hIconSm

       invoke LoadCursor,NULL,IDC_ARROW

       mov @stwc.hCursor,eax

       invoke RegisterClassEx,addr @stwc

       invoke CreateDialogParam,hInst,DIALOG_MAIN,NULL,NULL,NULL

       mov   hWinMain,eax

       invoke LoadMenu,hInst,IDR_MAINMENU

       mov hmenu,eax

       invoke GetSubMenu,hmenu,0

       mov hmenu,eax

       mov stnidstatus.cbSize,sizeof NOTIFYICONDATA

         push hWinMain

         pop stnidstatus.hwnd

         mov stnidstatus.uID,IDR_MAINFRAME

         push hicon

         pop  stnidstatus.hIcon

         mov stnidstatus.uFlags,NIF_ICON or NIF_TIP or NIF_MESSAGE

         mov stnidstatus.uCallbackMessage,WM_NOTIFYICONN

       invoke lstrcpy,addr stnidstatus.szTip,addr sztooltip

       invoke ShowWindow,hWinMain,SW_HIDE

       invoke UpdateWindow,hWinMain

       invoke _configload,1

    .while TRUE

         invoke GetMessage,addr @stmsg,NULL,0,0

         .BREAK .IF eax==0

            invokeIsDialogMessage,hWinMain,addr @stmsg

            .ifeax==FALSE

                    invoke TranslateMessage,addr @stmsg

                           invoke DispatchMessage,addr @stmsg

              .endif

     .endw 

      moveax,@stmsg.wParam

      ret       

_WinMain endp

;#########################################################################   ;加载配置

_configload proc isfirstrun:DWORD

.code

.if isfirstrun==1 ;第一次运行,需要读取文件和初始化

       ;初始化

       invoke GetDateFormat,NULL,NULL,NULL,addr szdateformat,addrsznowdate,50

    invokeGetTimeFormat,LOCALE_USER_DEFAULT,TIME_FORCE24HOURFORMAT,NULL,addrsztimeformat,addr sznowtime,50

     invoke SetTimer,hWinMain,IDT_TIMER,1000,NULL      

  mov idTIMER,eax

       invoke _statusicon,NIM_ADD;创建状态栏图标

        invokeInstallHook,hWinMain,WM_CLOCKALARM;安装热键钩子

        invoke GetSystemMetrics,SM_CXSCREEN

    shreax,1

    movtimershowx, eax

    movtimershowy, 1

.endif

 .if isintpointalarm==0

              invokeCheckDlgButton,hWinMain,IDC_CHECKBS,BST_UNCHECKED

 .else

           invoke CheckDlgButton,hWinMain,IDC_CHECKBS,BST_CHECKED

 .endif

 .if isshowtimer==0

              invokeCheckDlgButton,hWinMain,IDC_CHECKSHOWTIMER,BST_UNCHECKED

 .else

                 invoke CheckDlgButton,hWinMain,IDC_CHECKSHOWTIMER,BST_CHECKED

 .endif

  ret

_configload endp

;#########################################################################   ;保存配置

_configsave proc

 invoke IsDlgButtonChecked,hWinMain,IDC_CHECKBS

 .if eax==BST_CHECKED

        mov isintpointalarm,1

 .else

                   mov isintpointalarm,0

 .endif

 invokeIsDlgButtonChecked,hWinMain,IDC_CHECKSHOWTIMER 

 .if eax==BST_CHECKED

                     mov isshowtimer,1

 .else

                     mov isshowtimer,0

                     invoke InvalidateRect,NULL,NULL,NULL

 .endif

       ret

_configsave endp

;#########################################################################   end start

2、  DLL文件代码

.586

 .model flat, stdcall

 option casemap:none  

 

;#########################################################################

 

 include windows.inc

 include user32.inc

 include kernel32.inc

 

 includelib user32.lib

 includelib kernel32.lib

.data?

  hHook DWORD ?

  hMainWnd DWORD ?

  hWmessage DWORD ?

.data

   hInstance DWORD 0

   sztooltip db "桌面报时工具",0

;#########################################################################  

.code

;#########################################################################

;DLL入口

  DllEntry prochInst:HINSTANCE,reason:DWORD,userreserved:DWORD

            push hInst

     pop hInstance

              mov eax,TRUE

     ret

  DllEntry endp

;#########################################################################

;键盘消息处理函数

KeyProc proc uses ebx dwCode:DWORD,wParam:DWORD,lParam:DWORD

  invokeCallNextHookEx,hHook,dwCode,wParam,lParam

  .if dwCode == HC_ACTION&&wParam==VK_F12

        mov ebx,1

        shl ebx,30

        test lParam,ebx

        jne @f

        invoke SendMessage,hMainWnd,hWmessage,0,0

        @@:

  .endif

  xor eax,eax

  ret

KeyProc endp

;#########################################################################

;安装HOOK函数

InstallHook proc hwnd:DWORD,dwmessage:DWORD

   push hwnd

   pop hMainWnd

   push dwmessage

   pop hWmessage

   invokeSetWindowsHookEx,WH_KEYBOARD,addr KeyProc,hInstance,NULL

   mov hHook,eax

   ret

InstallHook endp

;#########################################################################

;卸载HOOK函数

UnInstallHook proc hwnd:DWORD

  invoke UnhookWindowsHookEx,hHook

  ret

UnInstallHook endp

;#########################################################################  

end DllEntry

3、  DLL导出文件内容(声明DLL的外部函数)

KeyProc  proto :DWORD,:DWORD,:DWORD

InstallHook proto :DWORD,:DWORD

UnInstallHook proto :DWORD

 

你可能感兴趣的:(Windows,Microsoft,编程,F#,IDE)