是否想為你的Windows加上一雙眼睛,察看使用者在機器上所做的各種操作(例如建立、刪除檔案;改變檔案或目錄名字)呢?
這裡介紹一種利用Windows未公開函式實現這個功能的方法。
在Windows下有一個未公開函式SHChangeNotifyRegister可以把你的視窗新增到系統的系統訊息監視鏈中,該函式在Delphi中的定義如下:
Function SHChangeNotifyRegister(hWnd,uFlags,dwEventID,uMSG,cItems:LongWord;
lpps:PIDLSTRUCT):integer;stdcall;external 'Shell32.dll' index 2;
其中引數hWnd定義了監視系統操作的視窗得控制程式碼,引數uFlags dwEventID定義監視操作引數,引數uMsg定義操作訊息,引數cItems定義附加引數,引數lpps指定一個PIDLSTRUCT結構,該結構指定監視的目錄。
當函式呼叫成功之後,函式會返回一個監視操作控制程式碼,同時系統就會將hWnd指定的視窗加入到操作監視鏈中,當有檔案操作發生時,系統會向hWnd傳送uMsg指定的訊息,我們只要在程式中加入該訊息的處理函式就可以實現對系統操作的監視了。
如果要退出程式監視,就要呼叫另外一個未公開得函式SHChangeNotifyDeregister來取消程式監視。
下面是使用Delphi編寫的具體程式實現範例,首先建立一個新的工程檔案,然後在Form1中加入一個Button控制元件和一個Memo控制元件,
程式的程式碼如下:
unit Unit1;
interface
uses
Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
StdCtrls,shlobj,Activex;
const
SHCNE_RENAMEITEM = $1;
SHCNE_CREATE = $2;
SHCNE_DELETE = $4;
SHCNE_MKDIR = $8;
SHCNE_RMDIR = $10;
SHCNE_MEDIAINSERTED = $20;
SHCNE_MEDIAREMOVED = $40;
SHCNE_DRIVEREMOVED = $80;
SHCNE_DRIVEADD = $100;
SHCNE_NETSHARE = $200;
SHCNE_NETUNSHARE = $400;
SHCNE_ATTRIBUTES = $800;
SHCNE_UPDATEDIR = $1000;
SHCNE_UPDATEITEM = $2000;
SHCNE_SERVERDISCONNECT = $4000;
SHCNE_UPDATEIMAGE = $8000;
SHCNE_DRIVEADDGUI = $10000;
SHCNE_RENAMEFOLDER = $20000;
SHCNE_FREESPACE = $40000;
SHCNE_ASSOCCHANGED = $8000000;
SHCNE_DISKEVENTS = $2381F;
SHCNE_GLOBALEVENTS = $C0581E0;
SHCNE_ALLEVENTS = $7FFFFFFF;
SHCNE_INTERRUPT = $80000000;
SHCNF_IDLIST = 0;
// LPITEMIDLIST
SHCNF_PATHA = $1;
// path name
SHCNF_PRINTERA = $2;
// printer friendly name
SHCNF_DWORD = $3;
// DWORD
SHCNF_PATHW = $5;
// path name
SHCNF_PRINTERW = $6;
// printer friendly name
SHCNF_TYPE = $FF;
SHCNF_FLUSH = $1000;
SHCNF_FLUSHNOWAIT = $2000;
SHCNF_PATH = SHCNF_PATHW;
SHCNF_PRINTER = SHCNF_PRINTERW;
WM_SHNOTIFY = $401;
NOERROR = 0;
type
TForm1 = class(TForm)
Button1: TButton;
Memo1: TMemo;
procedure FormClose(Sender: TObject; var Action: TCloseAction);
procedure Button1Click(Sender: TObject);
procedure FormCreate(Sender: TObject);
private
{ Private declarations }
procedure WMShellReg(var Message:TMessage);message WM_SHNOTIFY;
public
{ Public declarations }
end;
type PSHNOTIFYSTRUCT=^SHNOTIFYSTRUCT;
SHNOTIFYSTRUCT = record
dwItem1 : PItemIDList;
dwItem2 : PItemIDList;
end;
Type PSHFileInfoByte=^SHFileInfoByte;
_SHFileInfoByte = record
hIcon :Integer;
iIcon :Integer;
dwAttributes : Integer;
szDisplayName : array [0..259] of char;
szTypeName : array [0..79] of char;
end;
SHFileInfoByte=_SHFileInfoByte;
Type PIDLSTRUCT = ^IDLSTRUCT;
_IDLSTRUCT = record
pidl : PItemIDList;
bWatchSubFolders : Integer;
end;
IDLSTRUCT =_IDLSTRUCT;
function SHNotify_Register(hWnd : Integer) : Bool;
function SHNotify_UnRegister:Bool;
function SHEventName(strPath1,strPath2:string;lParam:Integer):string;
Function SHChangeNotifyDeregister(hNotify:integer):integer;stdcall;
external 'Shell32.dll' index 4;
Function SHChangeNotifyRegister(hWnd,uFlags,dwEventID,uMSG,cItems:LongWord;
lpps:PIDLSTRUCT):integer;stdcall;external 'Shell32.dll' index 2;
Function SHGetFileInfoPidl(pidl : PItemIDList;
dwFileAttributes : Integer;
psfib : PSHFILEINFOBYTE;
cbFileInfo : Integer;
uFlags : Integer):Integer;stdcall;
external 'Shell32.dll' name 'SHGetFileInfoA';
var
Form1: TForm1;
m_hSHNotify:Integer;
m_pidlDesktop : PItemIDList;
implementation
{$R *.DFM}
function SHEventName(strPath1,strPath2:string;lParam:Integer):string;
var
sEvent:String;
begin
case lParam of //根據引數設定提示訊息
SHCNE_RENAMEITEM: sEvent := '重新命名檔案'+strPath1+'為'+strpath2;
SHCNE_CREATE: sEvent := '建立檔案 檔名:'+strPath1;
SHCNE_DELETE: sEvent := '刪除檔案 檔名:'+strPath1;
SHCNE_MKDIR: sEvent := '新建目錄 目錄名:'+strPath1;
SHCNE_RMDIR: sEvent := '刪除目錄 目錄名:'+strPath1;
SHCNE_MEDIAINSERTED: sEvent := strPath1+'中插入可移動儲存介質';
SHCNE_MEDIAREMOVED: sEvent := strPath1+'中移去可移動儲存介質'+strPath1+' '+strpath2;
SHCNE_DRIVEREMOVED: sEvent := '移去驅動器'+strPath1;
SHCNE_DRIVEADD: sEvent := '新增驅動器'+strPath1;
SHCNE_NETSHARE: sEvent := '改變目錄'+strPath1+'的共享屬性';
SHCNE_ATTRIBUTES: sEvent := '改變檔案目錄屬性 檔名'+strPath1;
SHCNE_UPDATEDIR: sEvent := '更新目錄'+strPath1;
SHCNE_UPDATEITEM: sEvent := '更新檔案 檔名:'+strPath1;
SHCNE_SERVERDISCONNECT: sEvent := '斷開與伺服器的連線'+strPath1+' '+strpath2;
SHCNE_UPDATEIMAGE: sEvent := 'SHCNE_UPDATEIMAGE';
SHCNE_DRIVEADDGUI: sEvent := 'SHCNE_DRIVEADDGUI';
SHCNE_RENAMEFOLDER: sEvent := '重新命名資料夾'+strPath1+'為'+strpath2;
SHCNE_FREESPACE: sEvent := '磁碟空間大小改變';
SHCNE_ASSOCCHANGED: sEvent := '改變檔案關聯';
else
sEvent:='未知操作'+IntToStr(lParam);
end;
Result:=sEvent;
end;
function SHNotify_Register(hWnd : Integer) : Bool;
var
ps:PIDLSTRUCT;
begin
{$R-}
Result:=False;
If m_hSHNotify = 0 then begin
//獲取桌面資料夾的Pidl
if SHGetSpecialFolderLocation(0, CSIDL_DESKTOP,
m_pidlDesktop)<> NOERROR then
Form1.close;
if Boolean(m_pidlDesktop) then begin
ps.bWatchSubFolders := 1;
ps.pidl := m_pidlDesktop;
// 利用SHChangeNotifyRegister函式註冊系統訊息處理
m_hSHNotify := SHChangeNotifyRegister(hWnd, (SHCNF_TYPE Or SHCNF_IDLIST),
(SHCNE_ALLEVENTS Or SHCNE_INTERRUPT),
WM_SHNOTIFY, 1, ps);
Result := Boolean(m_hSHNotify);
end
Else
// 如果出現錯誤就使用 CoTaskMemFree函式來釋放控制程式碼
CoTaskMemFree(m_pidlDesktop);
End;
{$R+}
end;
function SHNotify_UnRegister:Bool;
begin
Result:=False;
If Boolean(m_hSHNotify) Then
//取消系統訊息監視,同時釋放桌面的Pidl
If Boolean(SHChangeNotifyDeregister(m_hSHNotify)) Then begin
{$R-}
m_hSHNotify := 0;
CoTaskMemFree(m_pidlDesktop);
Result := True;
{$R-}
End;
end;
procedure TForm1.WMShellReg(var Message:TMessage); //系統訊息處理函式
var
strPath1,strPath2:String;
charPath:array[0..259]of char;
pidlItem:PSHNOTIFYSTRUCT;
begin
pidlItem:=PSHNOTIFYSTRUCT(Message.wParam);
//獲得系統訊息相關得路徑
SHGetPathFromIDList(pidlItem.dwItem1,charPath);
strPath1:=charPath;
SHGetPathFromIDList(pidlItem.dwItem2,charPath);
strPath2:=charPath;
Memo1.Lines.Add(SHEvEntName(strPath1,strPath2,Message.lParam)+chr(13)+chr(10));
end;
procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
begin
//在程式退出的同時刪除監視
if Boolean(m_pidlDesktop) then
SHNotify_Unregister;
end;
procedure TForm1.Button1Click(Sender: TObject); //Button1的Click訊息
begin
m_hSHNotify:=0;
if SHNotify_Register(Form1.Handle) then begin //註冊Shell監視
ShowMessage('Shell監視程式成功註冊');
Button1.Enabled := False;
end
else
ShowMessage('Shell監視程式註冊失敗');
end;
procedure TForm1.FormCreate(Sender: TObject);
begin
Button1.Caption := '開啟監視';
end;
end.
執行程式,點選“開啟監視”按鈕,如果出現一個顯示“Shell監視程式成功註冊”的對話方塊,說明Form1已經加入到系統操作監視鏈中了,你可以試著在資源管理器中建立、刪除資料夾,移動檔案等操作,你可以發現這些操作都被記錄下來並顯示在文字框中。
在上面的程式中多次使用到了一個PItemIDList的結構,這個資料結構指定Windows下得一個“專案”,在Windows下資源實現統一管理一個“專案”可以是一個檔案或者一個資料夾,也可以是一個印表機等資源。另外一些API函式也涉及到了Shell(Windows外殼)操作,各位讀者可以參考相應的參考資料。
由於使用到了Windows的未公開函式,沒有相關得參考資料,所以有一些未知得操作(在Memo1中會顯示“未知操作”)。如果哪位讀者有興趣, http://member.netease.com/~blackcat 有實現該功能的VB程式下載。
以上程式在Windows98、Windows2000、Delphi5下執行通過。
看看這個合不合你用吧
是否想為你的Windows加上一雙眼睛,察看使用者在機器上所做的各種操作(例如建立、刪除檔案;改變檔案或目錄名字)呢?...
相關文章
- 刪除檔案或目錄提示"檔案或目錄無法刪除"的解決方法!
- linux 刪除檔案或目錄——rmLinux
- Linux rm命令:刪除檔案或目錄Linux
- 改變檔案或目錄的訪問許可權命令(轉)訪問許可權
- C語言remove()函式:刪除檔案或目錄C語言REM函式
- windows刪除檔案的批處理操作Windows
- 為什麼刪除記錄表檔案不會減小?(記錄的插入與刪除在磁碟上的變化)
- Linux改變檔案或目錄的訪問許可權命令Linux訪問許可權
- git的忽略檔案和刪除檔案操作Git
- 電腦上怎麼徹底刪除一個檔案?兩種可以直接徹底刪除檔案的操作方法
- 刪除github上的檔案Github
- 刪除指定目錄下指定字尾的檔案
- Linux rm(刪除檔案/目錄) 命令Linux
- Python 刪除目錄中特定檔案Python
- Linux命令:改變檔案或目錄的訪問許可權(轉)Linux訪問許可權
- 無法建立目錄或檔案 無法建立目錄的解決辦法
- Linux學習筆記--rm命令(刪除檔案或目錄)Linux筆記
- python基礎之刪除檔案及刪除目錄的方法Python
- 在WINDOWS下刪除幾天前的檔案Windows
- 兩個目錄中,刪除其中一個目錄中同名檔案的做法
- 刪除目錄及目錄下所有檔案與子目錄 (轉)
- 資料檔案丟失的恢復(改變目錄)
- 刪除git已經跟蹤的檔案或者目錄Git
- linux 刪除問題 一次刪除多個目錄下的相同副檔名的檔案Linux
- thinkphp對檔案的上傳,刪除,下載操作PHP
- linux rm 命令(刪除檔案和目錄)Linux
- php檔案操作之提取檔案/目錄的名稱PHP
- PHP 檔案操作的各種姿勢PHP
- c# 對檔案的各種操作C#
- 配置檔案的建立分配及刪除
- linux下刪除目錄及其子目錄下某種型別檔案Linux型別
- IOS中獲取各種檔案的目錄路徑的方法iOS
- windows下刪除過期的檔案Windows
- linux伺服器目錄檔案的命令操作Linux伺服器
- [python]如何刪除目錄下指定格式的檔案Python
- 刪除當前目錄下的所有可執行檔案
- 檔案的刪除
- windows 下刪除.svn檔案Windows