從任務(wù)通知區(qū)打開(kāi)屏幕保護(hù)程序
發(fā)表時(shí)間:2024-02-28 來(lái)源:明輝站整理相關(guān)軟件相關(guān)文章人氣:
[摘要]作者:朱志強(qiáng)本文通過(guò)一個(gè)快速啟動(dòng)屏幕保護(hù)程序的小程序SSLaunch,來(lái)介紹應(yīng)用程序如何向任務(wù)欄通知區(qū)加入圖標(biāo)、如何禁止多個(gè)Win32實(shí)例以及屏幕保護(hù)程序的有關(guān)內(nèi)容! SLaunch用C語(yǔ)言編寫(xiě),用Visual C++ 5.0編譯,是一個(gè)基于無(wú)模式對(duì)話框的程序,同時(shí)禁止多個(gè)實(shí)例,即一次只能有一...
作者:朱志強(qiáng)
本文通過(guò)一個(gè)快速啟動(dòng)屏幕保護(hù)程序的小程序SSLaunch,來(lái)介紹應(yīng)用程序如何向任務(wù)欄通知區(qū)加入圖標(biāo)、如何禁止多個(gè)Win32實(shí)例以及屏幕保護(hù)程序的有關(guān)內(nèi)容。
SSLaunch用C語(yǔ)言編寫(xiě),用Visual C++ 5.0編譯,是一個(gè)基于無(wú)模式對(duì)話框的程序,同時(shí)禁止多個(gè)實(shí)例,即一次只能有一個(gè)實(shí)例運(yùn)行。任務(wù)欄通知區(qū)圖標(biāo)在對(duì)話框初始化時(shí)加入,對(duì)話框響應(yīng)程序定義的回調(diào)消息,當(dāng)鼠標(biāo)左鍵按下時(shí),彈出一由屏幕保護(hù)程序名填充的上下文菜單。對(duì)話框關(guān)閉(即程序退出)時(shí)刪除任務(wù)欄通知區(qū)圖標(biāo)。如果讀者有興趣可以很容易地把它移植成基于 MFC 的程序。
1、任務(wù)欄通知區(qū)
Windows 95的任務(wù)欄中有一個(gè)通知區(qū), 應(yīng)用程序可以把一個(gè)圖標(biāo)放入其中,以表示操作狀態(tài),并可以有與之相關(guān)聯(lián)的工具用作說(shuō)明控制。當(dāng)鼠標(biāo)出現(xiàn)在此圖標(biāo)的矩形邊界內(nèi)時(shí),向相應(yīng)的應(yīng)用程序發(fā)送應(yīng)用程序定義的回調(diào)消息。應(yīng)用程序通過(guò)發(fā)送消息增加、修改、刪除任務(wù)欄圖標(biāo)。消息的發(fā)送通過(guò)調(diào)用函數(shù)Shell_NotifyIcon來(lái)完成,如果調(diào)用成功,則返回TRUE;否則,返回FALSE。Shell_NotifyIcon函數(shù)原形如下:
WINSHELLAPI BOOL WINAPI Shell_NotifyIcon(
DWORD dwMessage, // 消息標(biāo)識(shí)符
PNOTIFYICONDATA pnid // NOTIFYICONDATA 結(jié)構(gòu)
);
消息標(biāo)識(shí)符可以是 :
NIM_ADD 向任務(wù)欄通知區(qū)加入圖標(biāo)
NIM_DELETE 從任務(wù)欄通知區(qū)刪除圖標(biāo)
NIM_MODIFY 改變?nèi)蝿?wù)欄通知區(qū)圖標(biāo)
NOTIFYICONDATA 結(jié)構(gòu):
typedef struct _NOTIFYICONDATA {
DWORD cbSize;
HWND hWnd;
UINT uID;
UINT uFlags;
UINT uCallbackMessage;
HICON hIcon;
char szTip[64];
} NOTIFYICONDATA, *PNOTIFYICONDATA;
其中:
cbSize NOTIFYICONDATA 結(jié)構(gòu)大小
hWnd 接收回調(diào)消息窗口句柄
uID 任務(wù)欄通知區(qū)圖標(biāo)標(biāo)識(shí)
uFlags 指定該結(jié)構(gòu)中那些成員有效
uCallbackMessage 應(yīng)用程序定義的回調(diào)消息
hIcon 任務(wù)欄通知區(qū)圖標(biāo)句柄
szTip 任務(wù)欄通知區(qū)提示字符串
參數(shù)uFlags可以是下列值的組合:
NIF_ICON 任務(wù)欄通知區(qū)圖標(biāo)有效
NIF_MESSAGE 應(yīng)用程序定義的回調(diào)消息有效
NIF_TIP 任務(wù)欄通知區(qū)提示字符串有效
a.任務(wù)欄通知區(qū)圖標(biāo)的加入
BOOL SSLaunch_OnInitDialog(HWND hwnd, HWND hwndFocus, LPARAM lParam)
{
// Add an notification icon to the taskbar
NOTIFYCONDATA nid;
NOTIFYICONDATA nid;
nid.cbSize = sizeof(nid);
nid.hWnd = hwnd;
nid.uID = IDI_SSLAUNCH;
nid.uFlags = NIF_MESSAGE NIF_ICON NIF_TIP;
nid.uCallbackMessage = WM_SSLAUNCHICONNOTIFY;
nid.hIcon=LoadIcon(GetWindowInstance(hwnd),
KEINTRESOURCE(IDI_SSLAUNCH));
lstrcpyn(nid.szTip,g_szAppName,sizeof(nid.szTip) /sizeof(nid.szTip[0]));
return(Shell_NotifyIcon(NIM_ADD, &nid))
}
b.任務(wù)欄通知區(qū)圖標(biāo)的刪除
應(yīng)用程序退出時(shí),應(yīng)該刪除任務(wù)通知區(qū)上相應(yīng)的圖標(biāo):
void SSLaunch_OnDestroy(HWND hwnd)
{
// Remove the notification icon from the taskbar
NOTIFYICONDATA nid;
nid.cbSize = sizeof(nid);
nid.hWnd = hwnd;
nid.uID = IDI_SSLAUNCH;
Shell_NotifyIcon(NIM_DELETE, &nid);
}
c.應(yīng)用程序定義回調(diào)消息的接收
若為任務(wù)欄通知區(qū)指定了回調(diào)消息,則系統(tǒng)會(huì)于鼠標(biāo)事件在此區(qū)域發(fā)生時(shí)
向應(yīng)用程序發(fā)送此消息,其中wParam是任務(wù)欄通知區(qū)圖標(biāo)標(biāo)識(shí),lParam
是鼠標(biāo)事件發(fā)生后的鼠標(biāo)信息。
void SSLaunch_OnIconNotify(WPARAM wParam, LPARAM lParam)
{
UINT uID = (UINT)wParam;
UINT uMsg = (UINT)lParam;
if(uID == IDI_SSLAUNCH){
switch(uMsg){
case WM_LBUTTONDOWN :
//Do something
break;
case WM_LBUTTONUP :
//Do something
break;
default :
break;
}
}
}
2.禁止多個(gè)Win32實(shí)例
在討論禁止多個(gè)Win32實(shí)例之前,我們先討論一下WinMain函數(shù)。我們知道,任何一個(gè)基于GDI的Windows程序以WinMain函數(shù)作為入口被系統(tǒng)調(diào)用。在Win16中,hPrevInstance指向前一個(gè)實(shí)例的句柄,但在Win32中,每一個(gè)進(jìn)程都有一個(gè)獨(dú)立的4G地址空間,從0到2G屬于進(jìn)程私有,對(duì)其他進(jìn)程來(lái)說(shuō)是不可見(jiàn)的。所以,在Win32中,hPrevInstance總是為NULL。
int WINAPI WinMain(
HINSTANCE hInstance, // handle to current instance
HINSTANCE hPrevInstance, // handle to previous instance
LPSTR lpCmdLine, // pointer to command line
int nCmdShow // show state of window
);
因而,在Win32下不能通過(guò)判斷hPrevInstance是否為NULL來(lái)判斷一個(gè)程序的另一個(gè)實(shí)例是否存在,要用其他的方法來(lái)判斷。
方法一
用FindWindow 函數(shù)查找指定窗口,如果成功,則返回要找的窗口的句柄,否則返回NULL,由此可判斷是否有程序的另一個(gè)實(shí)例存在。
下圖的代碼片段演示如何使用FindWindow函數(shù):
TCHAR szClassName[] = _TEXT("My Wnd Class");
TCHAR szWndName[] = _TEXT("My Wnd");
HWND hWnd = FindWindow(szClassName,szWndName);
if(hWnd){
MessageBox(NULL, _TEXT("Another Instance is already running."), _TEXT("Information"),
MB_OK MB_ICONINFORMATION);
}
需要注意的是,很可能程序的各個(gè)實(shí)例有不同的窗口名,如果象下面這樣調(diào)用FindWindow
HWND hWnd = FindWindow(szClassName,NULL);
則查找所有的窗口并匹配窗口類(lèi)名,如果你能保證你的窗口類(lèi)名是唯一的,那么你可以信賴FindWindow,否則,你需要用更好的方法。
方法二
通過(guò)在EXE之間共享數(shù)據(jù)段從而共享數(shù)據(jù)來(lái)判斷是否有程序的另一個(gè)實(shí)例存在。
每個(gè)EXE或DLL都是由段的集合組成,在Win32程序中,每個(gè)段以點(diǎn)(.)開(kāi)頭。例如,當(dāng)編譯程序是編譯器時(shí),則將所有代碼放入一個(gè)叫.text的段、將所有未初始化的數(shù)據(jù)放入.bss段、將所有初始化的數(shù)據(jù)放入.data段。
可以給每個(gè)段賦予一個(gè)或多個(gè)屬性(以下為常用的一些段屬性):
READ 段中的數(shù)據(jù)可讀
WRITE 段中的數(shù)據(jù)可寫(xiě)
SHARED 段中的數(shù)據(jù)可被多個(gè)實(shí)例共享
EXECUTE 段中的數(shù)據(jù)可被執(zhí)行
可以用以下指令生成段:
#pragma data_seg("Shared")
static LONG g_lInstanceCount = -1;
#pragma data_seg()
編譯器生成這段代碼時(shí),產(chǎn)生一個(gè)新段,并把它所在#pragma data_seg("Shared")指令后的初始化數(shù)據(jù)放入新段Shared,未初始化的數(shù)據(jù)放入.bss段。#pragma data_seg()以后的數(shù)據(jù)放回缺省數(shù)據(jù)段。
僅告訴編譯器把特定數(shù)據(jù)放入自己的段內(nèi)還不足以共享它們,還要告訴鏈接器在某一特定段內(nèi)變量要共享?梢栽阪溄訒r(shí)指定這個(gè)段的屬性。
/section:Shared,rws
段名 屬性
程序初始化時(shí),例如調(diào)用WinMain函數(shù)時(shí),調(diào)用InterlockedIncrement函數(shù)使共享段內(nèi)變量加1,就可以通過(guò)判斷共享段內(nèi)變量的值來(lái)判斷一個(gè)程序有幾個(gè)實(shí)例在運(yùn)行。以下代碼演示了如何判斷一個(gè)正在運(yùn)行的程序?qū)嵗沁@個(gè)程序的第一個(gè)實(shí)例。
BOOL bIsFirstInstance = (InterlockedIncrement(&g_lInstanceCount) == 0);
if(!bIsFirstInstance){
MessageBox(NULL, _TEXT("Screen Saver Launcher is already running."), g_szAppName,
MB_OK MB_ICONINFORMATION);
}
使共享段內(nèi)變量加1,沒(méi)使用 g_lInstanceCount ++,而是使用InterlockedIncrement(&g_lInstanceCount),因?yàn)镮nterlockedIncrement函數(shù)對(duì)變量的訪問(wèn)進(jìn)行同步(Synchronize),阻止多個(gè)線程同時(shí)訪問(wèn)同一個(gè)變量。有關(guān)線程同步的內(nèi)容請(qǐng)參閱有關(guān)Win32 SDK的文檔。
禁止多個(gè)Win32實(shí)例的方法很多,如Win32核心對(duì)象(Mutex, Semaphore)、全局原子等都可以用來(lái)禁止多個(gè)Win32實(shí)例,在這里我們只簡(jiǎn)單地介紹以上兩種方法。
3.Screen Saver Launch:
屏幕保護(hù)程序是以scr為擴(kuò)展名的標(biāo)準(zhǔn)Windows可執(zhí)行程序。當(dāng)編輯可用屏幕保護(hù)程序的列表時(shí),Control Panel Desktop Applet在Windows啟動(dòng)目錄(Windows目錄和系統(tǒng)目錄)下查找擴(kuò)展名是scr的基于Windows的可執(zhí)行程序,如果Windows目錄和系統(tǒng)目錄下同時(shí)存在相同文件名的屏幕保護(hù)程序,則忽略Windows目錄下的那一個(gè)。任何蓄意的搗亂(如將文本文件或是基于DOS的可執(zhí)行文件擴(kuò)展名改為scr)Window95都不予理睬,但是將標(biāo)準(zhǔn)Windows可執(zhí)行程序的擴(kuò)展名改為scr時(shí),Windows95及NT將不會(huì)察覺(jué)。這只是很極端的情況,相信用戶不會(huì)采用這種做法來(lái)"測(cè)試"你的Windows.
標(biāo)準(zhǔn)的基于Win32的屏幕保護(hù)程序必須按照嚴(yán)格的標(biāo)準(zhǔn)編寫(xiě),有關(guān)詳細(xì)介紹請(qǐng)參閱有關(guān)Win32 SDK文檔。這里需要提到的一點(diǎn)是所有的基于Win32的屏幕保護(hù)程序都要求有一個(gè)不超過(guò)25個(gè)字符的說(shuō)明字符串。在屏幕保護(hù)程序的資源字符串表中,這個(gè)說(shuō)明字符串的標(biāo)識(shí)必須是1。
但我們發(fā)現(xiàn)在Windows 95下的屏幕保護(hù)程序不完全是嚴(yán)格按照標(biāo)準(zhǔn)編寫(xiě)的,當(dāng)編輯可用屏幕保護(hù)程序的列表時(shí),Control Panel Desktop Applet只是簡(jiǎn)單地把屏幕保護(hù)程序的文件名加入列表,而不是加入上面提及的說(shuō)明字符串。而在Windows NT下,系統(tǒng)嚴(yán)格區(qū)分標(biāo)準(zhǔn)的和非標(biāo)準(zhǔn)的屏幕保護(hù)程序。對(duì)于標(biāo)準(zhǔn)的屏幕保護(hù)程序,系統(tǒng)取得它的說(shuō)明字符串并將其顯示在屏幕保護(hù)程序的列表中;對(duì)于非標(biāo)準(zhǔn)的屏幕保護(hù)程序,系統(tǒng)只把它的文件名加入列表。
由于Windows 95和Windows NT下屏幕保護(hù)程序的列表顯示略有不同,所以這里分別加以說(shuō)明。為區(qū)別起見(jiàn),Windows 95下的SSLaunch用SSLaunch95表示,Windows NT下的SSLaunch用SSLaunchNT表示。
SSLaunch95 采用Window 95調(diào)用屏幕保護(hù)程序的方法,在Windows95的啟動(dòng)目錄下搜索屏幕保護(hù)程序,把文件名加到任務(wù)欄通知區(qū)圖標(biāo)上下文菜單中,單擊鼠標(biāo)即可啟動(dòng)相應(yīng)的屏幕保護(hù)程序。Windows 95把用戶選中的屏幕保護(hù)程序名保存在 System.ini文件中\(zhòng)boot\SCRNSAVE.EXE 下。SSLaunch95比較系統(tǒng)保存的用戶選中的屏幕保護(hù)程序名和搜索到的屏幕保護(hù)程序名,如果相同,則在任務(wù)欄通知區(qū)圖標(biāo)上下文菜單的相應(yīng)菜單項(xiàng)設(shè)置檢查標(biāo)志,以表示這個(gè)屏幕保護(hù)程序是否是當(dāng)前用戶選中的。SSLaunch95沒(méi)有判斷Windows啟動(dòng)目錄下的屏幕保護(hù)程序是否是真正的屏幕保護(hù)程序,因?yàn)閃indows 95下的Win32不能輕易地判斷一個(gè)scr文件是否是基于GDI的Windows可執(zhí)行文件(NE 或PE格式)。作者找到了兩個(gè)可用于判斷文件類(lèi)型的函數(shù):SHGetFileInfo,GetBinaryType。SHGetFileInfo可以判斷出.exe、.com、.bat幾種文件類(lèi)型,但認(rèn)為.scr文件不是可執(zhí)行文件;GetBinaryType可以輕易地判斷出文件類(lèi)型,但Windows 95不支持,只是簡(jiǎn)單地返回ERROR_NOT_IMPLEMENT,而Win32卻支持它。
點(diǎn)擊示意圖
SSLaunch95也可以在Windows NT 下運(yùn)行,不過(guò)彈出的上下文菜單不能用屏幕保護(hù)程序說(shuō)明字符串填充,并且不能判斷scr是否是基于GDI的Windows可執(zhí)行程序。
下面介紹SSLaunchNT在Windows NT下對(duì)scr文件的判別,以及從scr文件資源中取得屏幕保護(hù)程序描述字符串的方法。
a.對(duì)scr文件的判別
Windows NT提供了對(duì)GetBinaryType函數(shù)的支持,因此,可用此函數(shù)判斷一個(gè)scr文件是否是Windows可執(zhí)行程序,并判斷出它是基于Win16還是 Win32的可執(zhí)行程序。這一點(diǎn)很重要,因?yàn),?duì)基于Win32的scr文件,我們?cè)诤竺嬉〉盟淖址Y源中的一個(gè)重要信息,及對(duì)屏幕保護(hù)程序的描述字符串。還應(yīng)注意的是,lpApplicationName應(yīng)給出全路徑,否則,它只在進(jìn)程所在的路徑下尋找文件,這樣會(huì)導(dǎo)致錯(cuò)誤,從而不能返回在Windows啟動(dòng)目錄下的.scr文件的信息。
BOOL GetBinaryType(
LPCTSTR lpApplicationName,
LPDWORD lpBinaryType
);
GetBinaryType調(diào)用成功后,lpBinaryType指向的DWORD返回以下值:
SCS_32BIT_BINARY 基于Win32的應(yīng)用程序
SCS_DOS_BINARY 基于MS-DOS的應(yīng)用程序
SCS_OS216_BINARY 基于16位OS/2的應(yīng)用程序
SCS_PIF_BINARY MS-DOS應(yīng)用程序的PIF 文件
SCS_POSIX_BINARY 基于POSIX的應(yīng)用程序
SCS_WOW_BINARY 基于16位Windows的應(yīng)用程序
b.從scr文件字符串資源中取得屏幕保護(hù)文件描述字符串
當(dāng)我們判斷出了一個(gè)基于Win32的scr文件后,就可以著手取得它的字符串。在Win32中,有一種簡(jiǎn)單有效的方法:把一個(gè)EXE或DLL文件以數(shù)據(jù)文件方式加載,調(diào)用LoadLibraryEx函數(shù)。
HINSTANCE LoadLibraryEx(
LPCTSTR lpLibFileName, // EXE或DLL文件名
HANDLE hFile, // 保留參數(shù),必須為NULL
DWORD dwFlags // 函數(shù)入口標(biāo)志
);
dwFlags可以是0或以下標(biāo)志的組合:
DON'T_RESOLVE_DLL_REFERENCES 系統(tǒng)將DLL映射到進(jìn)程的地址空間而不調(diào)用DllMain函數(shù)。
LOAD_LIBRARY_AS_DATAFILE 系統(tǒng)將DLL象一個(gè)數(shù)據(jù)文件那樣映射到進(jìn)程的地址空間,而不調(diào)用DllMain函數(shù)。如果要取得EXE中的資源,也可調(diào)用LoadLibraryEx函數(shù)把EXE映射到進(jìn)程地址空間。
LOAD_WITH_ALTERED_SEARCH_PATH 將改變LoadLibraryEx在定位DLL文件時(shí)所采用的方法。
當(dāng)以LOAD_LIBRARY_AS_DATAFILE的方式調(diào)用LoadLibraryEx時(shí),系統(tǒng)只是簡(jiǎn)單地創(chuàng)建一個(gè)文件映象對(duì)象,把DLL(EXE)映射到本進(jìn)程的地址空間,并不調(diào)用DllMain(WinMain)。如果調(diào)用成功,則函數(shù)返回一個(gè)HINSTANCE,即被映射到本進(jìn)程地址空間的DLL(EXE)的裝入地址,這樣,就可以調(diào)用LoadString函數(shù),從DLL(EXE)文件的字符串資源表中取得指定的字符串。
點(diǎn)擊示意圖
這里仍需指出的是,必須判斷LoadString函數(shù)調(diào)用是否成功,因?yàn)橛行﹕cr文件(即使是基于Win32的)也有可能是非標(biāo)準(zhǔn)的(如Windows 95下的大多數(shù)scr文件),如果LoadString調(diào)用失敗,則SSLaunchNT用文件名取代scr的描述字符串填入SSLaunchNT上下文菜單的菜單項(xiàng)。