奇淫怪巧之给Delphi的PrintDialog增加一个页码选定范围打印的Edit 转

   在Delphi中使用PrintDialog打印对话框的时候,这个控件有三个选项,就是PrintRang那个属性的三个选项,其中有一个选项三,让我们自定义选择页码范围来打印。但是比较蛋疼的是,这个地方选中了之后啥子效果都没有。无法制定自己的页码范围,很是蛋疼。这里实际上应该要有一个Edit之类的编辑框的,这样可以让用户输入1,2,3-4之类的页码范围来整就比较人性化了。起初以为是自己没有指定某属性神马的导致,于是在控件的各个属性中找,找的我蛋都要裂开了,都没找出来似乎是隐藏了那个Edit的属性。。。。无果,整开了PrintDialog的源码看。也未发现相关的代码。咋办,咋办呢。路总是人走出来的,目的就是要在这个弹出的对话框上加一个编辑框就OK了,方式很多,可以在弹出来之后,查找句柄,然后创建一个Edit,然后SetParent就上去了。这是一方面,另外一方面,就是要配合对话框上,用户点击上面的选项来相应的联动这个Edit为可用或非可用状态。这个自然也是有办法的,我首先想到的就是Hook了,于是就用Hook整了,WH_CallWndProc消息处理过程函数的Hook,就行了。至于是要拦截神马消息,嘿嘿,这个是WinSDK的范畴,不熟悉的人,去翻番书,或者百度一番,就可以知道是WM_Command这个消息来处理窗口中的某些控件的消息处理的。首先就是要获取那个弹出的打印对话框的句柄了,我说过了可以用FindWindow来找。不过我这里要使用消息过程钩子,那么自然就不要这个了,直接从钩子中获取对话框句柄,会更加Happy。看看PrintDialog的代码,俺们可以知道,神马PrintDialog,OpenDialog,FontDialog打开都要触发WM_InitDialog这个消息,就是初始化对话框的消息。所以,第一步,俺们就拦截这个消息就 可以获取到句柄,然后创建俺们的Edit,然后将Edit设置到对话框中,俺们需要他在的位置。第二步,就是来拦截WM_Command,然后来处理和用户点击的联动处理咯。然后就是在对话框的Destroy消息中注销Hook,释放Edit。于是这个过程就OK了

首先在打开对话框之前,注册我们的过程处理钩子,必须要之前注册,因为要拦截WM_InitDialog消息嘛

procedure TForm1.Button1Click(Sender: TObject);
begin
  CEdit := TEdit.CreateParented(Application.Handle);
  CEdit.Parent := Application.MainForm;
  Hok := SetWindowsHookEx(WH_CALLWNDPROC,WNDProcHook,HInstance,GetCurrentThreadId);
  if PrintDialog1.Execute then
  begin
    ShowMessage(CEdit.Text);
  end;
  UnhookWindowsHookEx(HOK);
  FreeAndNil(CEdit);
end;

然后是钩子函数过程处理

var
  DownBtnID: Integer;
  PrntDlgHandle: THandle;
  CEdit: TEdit;
function WNDProcHook(code: Integer; wparam: WPARAM; lparam: LPARAM): LRESULT stdcall;
var
  msg: PCWPSTRUCT;
  wNotifyCode,wID: Word;
  r,WinR: TRect;
  DwonBtnHandle: THandle;
begin
  if code < 0 then
    Result := CallNextHookEx(Hok, code, wparam, lparam)
  else
  begin
    msg := PCWPSTRUCT(lparam);
    case msg.message of
    WM_INITDIALOG:
       begin
          PrntDlgHandle := msg.hwnd;
          DwonBtnHandle := FindWindowEx(PrntDlgHandle,0,'Button','打印范围');
          Windows.SetParent(CEdit.Handle,DwonBtnHandle);
          GetWindowRect(DwonBtnHandle,WinR);
          CEdit.Visible := True;
          DwonBtnHandle := FindWindowEx(PrntDlgHandle,0,'Button','选定范围(&S)');
          CEdit.Enabled := SendMessage(DwonBtnHandle,BM_GETCHECK,0,0) = 1;
          GetWindowRect(DwonBtnHandle,r);
          DownBtnID := GetDlgCtrlID(DwonBtnHandle);
          CEdit.Left := r.Right - WinR.Left;
          CEdit.Top := r.Top - WinR.Top;
       end;
    WM_COMMAND:
       begin
          if msg.hwnd = PrntDlgHandle then
          begin
            wNotifyCode := HIWORD(Msg.wparam);
            wID := LOWORD(Msg.wParam);
            if wNotifyCode = BN_CLICKED then
            begin
              CEdit.Enabled := wID = DownBtnID
            end;
          end;
       end;
    WM_DESTROY:
       begin
         //UnhookWindowsHookEx(HOK);
         //Hok := 0;
         //FreeAndNil(CEdit);
       end;
    end;
    Result := 0;
  end;
end;

  可见,我这里注销了WM_Destroy中的处理。目的是因为俺们还需要返回这个Edit中的内容嘛,所以我们直接在使用完了之后注销钩子,释放Edit就行了!

实现之后的效果,就是这样的咯

posted @ 2020-01-15 12:00  绿水青山777  阅读(333)  评论(0编辑  收藏  举报