Карта сайта Kansoftware
НОВОСТИУСЛУГИРЕШЕНИЯКОНТАКТЫ
KANSoftWare

Ловить события мышки вне вашего приложения

Delphi , ОС и Железо , Мышка и Курсор

Ловить события мышки вне вашего приложения

Оформил: DeeCo
Автор: http://www.swissdelphicenter.ch

unit Unit1;

 interface

 uses
   Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms,
   Dialogs, AppEvnts, StdCtrls;

 type
   TForm1 = class(TForm)
     ApplicationEvents1: TApplicationEvents;
     Button_StartJour: TButton;
     Button_StopJour: TButton;
     ListBox1: TListBox;
     procedure ApplicationEvents1Message(var Msg: tagMSG;
       var Handled: Boolean);
     procedure Button_StartJourClick(Sender: TObject);
     procedure Button_StopJourClick(Sender: TObject);
     procedure FormClose(Sender: TObject; var Action: TCloseAction);
   private
     { Private declarations }
     FHookStarted : Boolean;
   public
     { Public declarations }
   end;

 var
   Form1: TForm1;


 implementation

 {$R *.dfm}

 var
   JHook: THandle;

 // The JournalRecordProc hook procedure is an application-defined or library-defined callback 
// function used with the SetWindowsHookEx function. 
// The function records messages the system removes from the system message queue. 
// A JournalRecordProc hook procedure does not need to live in a dynamic-link library. 
// A JournalRecordProc hook procedure can live in the application itself. 

// WH_JOURNALPLAYBACK Hook Function 

//Syntax 

// JournalPlaybackProc( 
// nCode: Integer;  {a hook code} 
// wParam: WPARAM;  {this parameter is not used} 
// lParam: LPARAM  {a pointer to a TEventMsg structure} 
// ): LRESULT;  {returns a wait time in clock ticks} 


function JournalProc(Code, wParam: Integer; var EventStrut: TEventMsg): Integer; stdcall;
 var
   Char1: PChar;
   s: string;
 begin
   {this is the JournalRecordProc}
   Result := CallNextHookEx(JHook, Code, wParam, Longint(@EventStrut));
   {the CallNextHookEX is not really needed for journal hook since it it not 
  really in a hook chain, but it's standard for a Hook}
   if Code < 0 then Exit;

   {you should cancel operation if you get HC_SYSMODALON}
   if Code = HC_SYSMODALON then Exit;
   if Code = HC_ACTION then
   begin
     { 
    The lParam parameter contains a pointer to a TEventMsg 
    structure containing information on 
    the message removed from the system message queue. 
    }
     s := '';

     if EventStrut.message = WM_LBUTTONUP then
       s := 'Left Mouse UP at X pos ' +
         IntToStr(EventStrut.paramL) + ' and Y pos ' + IntToStr(EventStrut.paramH);

     if EventStrut.message = WM_LBUTTONDOWN then
       s := 'Left Mouse Down at X pos ' +
         IntToStr(EventStrut.paramL) + ' and Y pos ' + IntToStr(EventStrut.paramH);

     if EventStrut.message = WM_RBUTTONDOWN then
       s := 'Right Mouse Down at X pos ' +
         IntToStr(EventStrut.paramL) + ' and Y pos ' + IntToStr(EventStrut.paramH);

     if (EventStrut.message = WM_RBUTTONUP) then
       s := 'Right Mouse Up at X pos ' +
         IntToStr(EventStrut.paramL) + ' and Y pos ' + IntToStr(EventStrut.paramH);

     if (EventStrut.message = WM_MOUSEWHEEL) then
       s := 'Mouse Wheel at X pos ' +
         IntToStr(EventStrut.paramL) + ' and Y pos ' + IntToStr(EventStrut.paramH);

     if (EventStrut.message = WM_MOUSEMOVE) then
       s := 'Mouse Position at X:' +
         IntToStr(EventStrut.paramL) + ' and Y: ' + IntToStr(EventStrut.paramH);

     if s <> '' then
        Form1.ListBox1.ItemIndex :=  Form1.ListBox1.Items.Add(s);
   end;
 end;

 procedure TForm1.Button_StartJourClick(Sender: TObject);
 begin
   if FHookStarted then
   begin
     ShowMessage('Mouse is already being Journaled, can not restart');
     Exit;
   end;
   JHook := SetWindowsHookEx(WH_JOURNALRECORD, @JournalProc, hInstance, 0);
   {SetWindowsHookEx starts the Hook}
   if JHook > 0 then
   begin
     FHookStarted := True;
   end
   else
     ShowMessage('No Journal Hook availible');
 end;

 procedure TForm1.Button_StopJourClick(Sender: TObject);
 begin
   FHookStarted := False;
   UnhookWindowsHookEx(JHook);
   JHook := 0;
 end;

 procedure TForm1.ApplicationEvents1Message(var Msg: tagMSG;
   var Handled: Boolean);
 begin
   {the journal hook is automaticly camceled if the Task manager 
  (Ctrl-Alt-Del) or the Ctrl-Esc keys are pressed, you restart it 
  when the WM_CANCELJOURNAL is sent to the parent window, Application}
   Handled := False;
   if (Msg.message = WM_CANCELJOURNAL) and FHookStarted then
     JHook := SetWindowsHookEx(WH_JOURNALRECORD, @JournalProc, 0, 0);
 end;

 procedure TForm1.FormClose(Sender: TObject; var Action: TCloseAction);
 begin
   {make sure you unhook it if the app closes}
   if FHookStarted then
     UnhookWindowsHookEx(JHook);
 end;

 end.

Программный код на Delphi, демонстрирующий установку хука для захвата событий мыши вне области приложения.

Функция JournalProc - это callback-процедура, которая будет вызываться каждый раз, когда происходит событие мыши вне области приложения. В этой функции проверяется тип события (нажатие левой кнопки, отпускание левой кнопки, нажатие правой кнопки, отпускание правой кнопки, движение колеса мыши или перемещение мыши) и строится строка для отображения в списке.

Процедуры Button_StartJourClick и Button_StopJourClick устанавливают и останавливают хук соответственно, вызывая функцию SetWindowsHookEx с константой WH_JOURNALRECORD и адресом функции JournalProc. Процедура ApplicationEvents1Message используется для перезапуска хука в случае его отмены из-за системного события, такого как открытие задачмана. Процедура FormClose обеспечивает остановку хука при закрытии приложения.

Вот некоторые потенциальные улучшения:

  • Обработка ошибок: код не обрабатывает ошибки должным образом. Например, если функция SetWindowsHookEx fails, она вернет ноль и код не будет уведомлен о этом. Лучше было бы проверять возвращаемое значение этой функции и отображать сообщение об ошибке, если оно не успешно.

  • Организация кода: код мог быть организован в отдельные модули для основной логики приложения и процедуры хука. Это сделало бы код более легко поддерживаемым и повторно используемым.

  • Выполнение: процедура хука вызывается каждый раз, когда происходит событие мыши вне области приложения. Если приложение обрабатывает много таких событий, это может привести к проблемам с производительностью. Возможно, нужно оптимизировать обработку этих событий или использовать другой подход.

  • Совместимость: код может не работать корректно на всех версиях Windows. Например, он использует константу WH_JOURNALRECORD, которая является специфичной для Windows NT и поздних версий.

Вот альтернативное решение с использованием встроенного компонента TMouseHook Delphi:

unit Unit1;

interface

uses
  Winapi.Windows, Vcl.Controls, Vcl.Forms, Vcl.Dialogs;

type
  TForm1 = class(TForm)
    procedure FormCreate(Sender: TObject);
    procedure FormDestroy(Sender: TObject);
  private
    FMouseHook: TMouseHook;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);
begin
  FMouseHook := TMouseHook.Create(nil);
  FMouseHook.OnMouseDown := MouseHookMouseDown;
  FMouseHook.OnMouseMove := MouseHookMouseMove;
  FMouseHook.OnMouseUp := MouseHookMouseUp;
end;

procedure TForm1.FormDestroy(Sender: TObject);
begin
  FreeAndNil(FMouseHook);
end;

procedure TForm1.MouseHookMouseDown(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  // Обрабатываем событие нажатия мыши
  Memo1.Lines.Add('Мышка нажата в (' + IntToStr(X) + ', ' + IntToStr(Y) + ')');
end;

procedure TForm1.MouseHookMouseMove(Sender: TObject; X, Y: Integer;
  Shift: TShiftState);
begin
  // Обрабатываем событие перемещения мыши
  Memo1.Lines.Add('Мышка переместилась в (' + IntToStr(X) + ', ' + IntToStr(Y) + ')');
end;

procedure TForm1.MouseHookMouseUp(Sender: TObject; Button: TMouseButton;
  Shift: TShiftState; X, Y: Integer);
begin
  // Обрабатываем событие отпускания мыши
  Memo1.Lines.Add('Мышка отпущена в (' + IntToStr(X) + ', ' + IntToStr(Y) + ')');
end;

end.

В этом коде компонент TMouseHook используется для захвата событий мыши вне области приложения. События OnMouseDown, OnMouseMove и OnMouseUp используются для обработки этих событий. Координаты события мыши отображаются в текстовом поле.

Ловить события мышки вне вашего приложения: статья рассказывает о способе поймать события мыши в Windows, не используя традиционных окон и форм, а также предоставляет пример на Delphi.


Комментарии и вопросы

Получайте свежие новости и обновления по Object Pascal, Delphi и Lazarus прямо в свой смартфон. Подпишитесь на наш Telegram-канал delphi_kansoftware и будьте в курсе последних тенденций в разработке под Linux, Windows, Android и iOS




Материалы статей собраны из открытых источников, владелец сайта не претендует на авторство. Там где авторство установить не удалось, материал подаётся без имени автора. В случае если Вы считаете, что Ваши права нарушены, пожалуйста, свяжитесь с владельцем сайта.


:: Главная :: Мышка и Курсор ::


реклама


©KANSoftWare (разработка программного обеспечения, создание программ, создание интерактивных сайтов), 2007
Top.Mail.Ru

Время компиляции файла: 2024-08-19 13:29:56
2024-11-21 13:19:33/0.0061891078948975/1