我如何在C#项目中收听设备的消息

时间:2019-04-05 19:06:00

标签: c# c++ windows message handle

又是我!

这个问题很难解决,因为我会尽力解释它:正如我在上一个问题中提到的那样,我正在使用提供程序发送的C ++ dll在C#上进行扫描器管理。根据API的手册,在某些条件下会发送某些消息。例如:启动扫描仪后,它应该发送消息DEVICE_CONNECTED(值为0),然后更改其状态。

这些消息值在.dll中定义

我的问题是试图在我的C#项目中获取这些消息

我一直在寻找有关邮件传输的信息,我发现有一个处理Windows邮件的WndProc,因为我尝试按照它们的示例进行操作:

private const int DEVICE_CONNECTED = 0;
/*Some code*/
[System.Security.Permissions.PermissionSet(System.Security.Permissions.SecurityAction.Demand, Name = "FullTrust")]
protected override void WndProc(ref Message m)
{
    if (m.Msg == DEVICE_CONNECTED)
       listBox1.Items.Add("Connected");
    base.WndProc(ref m);
}

当然,那个失败了。

稍后,我一直在查看API手册,我认为我可以从哪里获得消息的线索:

// This is how is defined at .dll (C++)
DWORD StartUp( HWND Handle, UINT SorterMessage )

“句柄”是应用程序的消息目标窗口的句柄。

所以我的C#导入如下:

[DllImport(path, EntryPoint = "?StartUp@@YGKPAUHWND__@@I@Z")]
public static extern int StartUp(IntPtr HWMD, uint SorterMessage);

现在我有了一个可以从中提取消息的指针。我的问题是:如何?

我在另一个论坛上找到了这个例子:

[Serializable, StructLayout(LayoutKind.Sequential)]
public struct MSG
{
    public IntPtr hwnd;
    public int message;
    public IntPtr wParam;
    public IntPtr lParam;
    public int time;
    public int pt_x;
    public int pt_y;
};

[DllImport("user32.dll", CharSet = CharSet.Ansi)]
public static extern bool GetMessage([In, Out] ref MSG msg, IntPtr hWnd, int uMsgFilterMin, int uMsgFilterMax);
[DllImport("user32.dll", CharSet = CharSet.Auto)]
public static extern IntPtr DispatchMessage([In] ref MSG msg);

MSG msg = new MSG();
while (GetMessage(ref msg, IntPtr.Zero, 0, 0))
    DispatchMessage(ref msg);

我尝试使用它,如下所示:

// Added a constructor inside of the struct:
public MSG(IntPtr hwndPtr)
{
    hwnd = hwndPtr;
    message = -1;
    wParam = new IntPtr();
    lParam = new IntPtr();
    time = 0;
    pt_x = 0;
    pt_y = 0;
}

// Left the dll imports like in their example (although I fixed the path)

// Calling the method in my main
int ID, st;
ID = Class1.StartUp(hwnd, 10); // Just used 10 like in the API's manual
Console.WriteLine("Turning on device");
MSG msg = new MSG(hwnd);
while(Class1.GetMessage(ref msg, IntPtr.Zero, 0, 0))
    Class1.DispatchMessage(ref msg);
Console.WriteLine(msg.message);

do { Class1.GetState(ID, out st); }
while (st != (int) DevStates.chgParams);
Console.WriteLine("Device on");

我期望什么?打印“打开设备”后,我应该收到消息(因为在启动过程中,并且根据手册,它会在更改状态之前发送一条消息),然后显示“设备打开”字符串。

我能得到什么?刚打印完“打开设备”程序后,除了闪烁光标外什么也没做(当然,“设备打开”字符串永远不会显示)。看起来它正在等待任何消息。尝试将消息呼叫放置在不同的位置,并且行为相同。

有什么建议吗?预先感谢。

1 个答案:

答案 0 :(得分:0)

最终解决了

这是我的方法:

  1. 使用的Windows窗体,因为它具有“消息”类
  2. 导入了我正在使用的.dll,以使其变得更容易放置 “ ScanMgr”类中的所有方法。
using ...
using APIcsharp;

class ScanMgr
{
  int ID = 0;

  public string startUp(IntPtr hwmd, uint wmApp)
  {
    int state;
    ID = NativeAPI.StartUp(hwmd, wmApp);
    if(ID != 0)
    {
      do { NativeAPI.GetState(ID, out state); }
      while(state == (int)(DevStates.StartingUp)); // DevStates is a enum
      return "Device on";
    }
    return "Error turning on";
  }

  /* Other stuff to do */
}
  1. 然后,定义消息的替代方法
public partial class Form1 : Form
{
  const uint wm_channel = 0x8000 + 1;
  ScanMgr scanner = new ScanMgr();

  public Form1()
  { InitializeComponent(); }

  private void StartBtn_Click(object sender, EventArgs e)
  { log.Items.Add(scanner.startUp(this.Handle, wm_channel)); }

  /* Other stuff yadda yadda */

  protected override void WndProc(ref Message m)
  {
    base.WndProc(ref m);
    if(m.Msg == wm_channel)
    { /* To do stuff with m.WParam and m.LParam */ }
  }
}