TApplicationEvents控件的使用方法

Combobox拦截滚轮事件和响应滚动事件

OnMessage方法:Application中的OnMessage方法可以处理属于Application的除WM_QUIT以外的所有消息。但该方法用于处理个别方法时,效率会很低,因为需要对WINDOWS中的众多的消息中筛选其中之一。

在滚动鼠标滚轮时,原意是想让整个页面向下Scroll,但是此时恰巧页面焦点聚焦在Combobox的文本框内,滚动滚轮却是把Combobox内的内容给改变了。正常情况下这个也不影响什么,有时也方便操作,但是在此时的情况下无疑是不合适的,因为用户可能在不知情的情况下会把Combobox内的内容给改变,这无疑是不能发生的,所以,我就自己写段代码,来阻止这种事情发生。

RzCombobox有滚轮事件,但是Delphi自带的Combobox还有Dev Express的cxCombobox没有此事件,那么我们就可以自己写代码来拦截滚轮事件,以下是代码:

unit Unit1;
interface
uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, StdCtrls, ExtCtrls;
type
  TForm1 = class(TForm)
    Panel1: TPanel;
    ComboBox1: TComboBox;
    Label1: TLabel;
    procedure FormCreate(Sender: TObject);
  private
    { Private declarations }
    Procedure OnMouseWheel(Var Msg: TMsg; var Handled: Boolean);
  public
    { Public declarations }
  end;
var
  Form1: TForm1;
implementation
{$R *.dfm}
procedure TForm1.FormCreate(Sender: TObject);
begin
  Application.OnMessage := OnMouseWheel;
end;
procedure TForm1.OnMouseWheel(var Msg: TMsg; var Handled: Boolean);
begin
  if Msg.message = WM_MOUSEWHEEL then //多个过程都接收这一消息时会产生冲突,只会其中一个有效
  begin
    //ShowMessage(ActiveControl.ClassName);
    if ActiveControl IS TComboBox then //Delphi自带Combobox
    //if ActiveControl.ClassName = 'TcxCustomComboBoxInnerEdit' then //CxCombobox
    //if ActiveControl is TcxCustomComboBoxInnerEdit then //表格不响应滚轮
    //if ActiveControl is TStringGrid then //ActiveControl:指定以窗体为焦点的控件;不响应滚轮
    begin
      Handled := True;
    end;
  end;
end;
procedure TForm1.FormDestroy(Sender: TObject);
begin
  Application.OnMessage := nil;
end;
end.

响应滚动事件,添加控件ApplicationEvents1: TApplicationEvents;

procedure TFrm2.ApplicationEvents1Message(var Msg: tagMSG; var Handled: Boolean);
var
  sColumnCaption: string;
begin
  inherited;
   //窗体的滚动条响应鼠标滚轮
  if g_sEmbryoAutoDropDown = '2' then
  begin
    if Msg.message = WM_MOUSEWHEEL then
    begin
      if self.Visible then
      begin
        sColumnCaption := cxGrid1DBBandedTableView1.Controller.FocusedColumn.DataBinding.DefaultCaption;
        if Msg.wParam > 0 then
        begin
          cxGrid1DBBandedTableView1.DataController.DataSource.DataSet.MoveBy(-1);
          cxGrid1DBBandedTableView1.GetColumnByFieldName(sColumnCaption).Focused := True;
        end
        else
        begin
          cxGrid1DBBandedTableView1.DataController.DataSource.DataSet.MoveBy(1);
          cxGrid1DBBandedTableView1.GetColumnByFieldName(sColumnCaption).Focused := True;
        end;
        Handled := True;
      end
      else
        Handled := false;
    end;
  end;
end;

以下是OnMessage事件,用来阻止WebBrowser的右键菜单:

procedure TfrmSearch.ApplicationEvents1Message(var Msg: tagMSG;
  var Handled: Boolean);
begin
  //如果是Webbrowser,则不许使用右键菜单
  if IsChild(WebBrowser1.Handle,Msg.Hwnd) and ((Msg.Message = WM_RBUTTONDOWN) or (Msg.Message=WM_RBUTTONUP)) then
  begin
    Handled:=True;
  end;
end;

dbgrid中假如记录很多,使用鼠标滚动条滚动的时候,只会在dbgrid当前页面数据内滚动,而滚动条是不会动的,这样操作很不方便,需要再拖动滚动条去滚动数据。加了TApplicationEvents就可以去判断事件触发对应的方法。

代码:新建应用程序工程,直接在界面上放个dbgrid,高度不要太高,写form的oncreate事件和TApplicationEvents的onmessage事件。不放和放TApplicationEvents

nit Unit1;

interface

uses

Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,

Dialogs,DBClient,DB, Grids, DBGrids, AppEvnts;

type

TForm1 = class(TForm)

DBGrid1: TDBGrid;

ApplicationEvents1: TApplicationEvents;

procedure FormCreate(Sender: TObject);

procedure ApplicationEvents1Message(var Msg: tagMSG;

var Handled: Boolean);

private

{ Private declarations }

cdstemp:TClientDataSet;

ds:TDataSource;

public

{ Public declarations }

end;

var

Form1: TForm1;

implementation

{$R *.dfm}

procedure TForm1.FormCreate(Sender: TObject);

var

i:Integer;

begin

//创建对象

cdstemp:=TClientDataSet.Create(Application);

ds:=TDataSource.Create(Application);

//创建dataset列

with cdstemp do

begin

Active:=False;

FieldDefs.Clear;

FieldDefs.Add('id',ftString,32);

CreateDataSet;

Active:=True;

end;

//生成测试数据

for i:=0 to 20 do

begin

cdstemp.Append;

cdstemp.FieldByName('id').AsInteger:=i;

cdstemp.Post;

end;

//绑定

ds.DataSet:=cdstemp;

DBGrid1.DataSource:=ds;

end;

procedure TForm1.ApplicationEvents1Message(var Msg: tagMSG;

var Handled: Boolean);

begin

//先判断焦点在不在DBGrid上

if DBGrid1.Focused then

begin

//判断是不是鼠标滚轮事件

if Msg.message  =  WM_MOUSEWHEEL then

begin

if  Msg.wParam  >  0  then

//滚动条往上滚

SendMessage(DBGrid1.Handle,  WM_KEYDOWN,  VK_UP,  0)

else

//滚动条往下滚

SendMessage(DBGrid1.Handle,  WM_KEYDOWN,  VK_DOWN,  0);

Handled  :=  True;

end;

end;

end;

end.

TApplication.OnIdle   当应用程序变为空闲时,OnIdle事件发生。   使用OnIdle事件,可以写一个应用程序空闲时执行特定处理的事件处理程序。当应用程序不处理代码时,称为应用程序空闲。例如,当应用程序等待来自用户的输入时,应用程序为空闲。   TIdleEvent类型是OnIdle事件的类型,它指向一个应用程序空闲时运行的方法。TIdleEvent类型有一个布尔型参数Done,默认时该参数为True。若参数Done为True,当OnIdle事件返回时,调用Windows API WaitMessage函数。只有在应用程序消息队列中出现一个新消息时,WaitMessage函数才放弃对其他应用程序的控制。参数Done为False时,即使应用程序不忙,也不放弃对其他应用程序的控制。   当应用程序转移到空闲状态时,只调用一次OnIdle事件。除非参数Done设置为False,否则不连续调用OnIdle事件。将参数Done设置为False的应用程序,将消耗过多的CPU时间,从而影响整个系统性能。 在delphi中, 当在一个窗口上放置一个ApplicationEvents控件时,Application将会把所有的事件都转寄到ApplicationEvents; 也就是说,ApplicationEvents可以拦截到应用程序的全部事件,包括OnActivate\OnHelp\OnIdle\OnRestore\OnShortCut等等, 甚至可能通过OnMessage事件,在其中截取所有post到应用程序中所有窗口的消息,如WM_PAINT,WM_KEYDOWN, WM_KEYUP等常见的windows消息; 所以当有消息到来的时候就会触发它的OnMessage事件,在OnMessage中监视消息就可以了。 Action的事件有OnExecute和OnUpdate,OnExecute事件在控制被触发时响应,比如说按钮被按下,菜单被按下,而OnUpdate事件是在应用程序空闲时被调用, APPLICATIONEVENTS是用来捕获程序级事件的 ApplicationEvents1Message(var Msg: tagMSG;var Handled: Boolean); {通过 Perform 向窗体发送 消息; OnMessage 收不到} {通过 SendMessage 向窗体发送 消息; OnMessage 收不到} {通过 PostMessage 向窗体发送  消息; OnMessage 可以收到}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值