纳金网

标题: unity3d 实现windows 消息 [打印本页]

作者: yeu1233    时间: 2012-9-13 11:39
标题: unity3d 实现windows 消息
先前提到可以用 Hooks 的方法在 Unity 裡監控 Windows Message,但是使用 Hooks 這個方法我們沒辦法更改 Message 的內容。因此這邊提出第二個方式。Windows 傳送 Message 給 Unity 時,會呼叫 Unity 預設的 Message 處理函數,但透過函數:



pOldWndProc =(WNDPROC)SetWindowLong(hWnd, GWL_WNDPROC, (LONG)SubWndProc);



我們可以將原本 Windows 呼叫 Unity 預設的 Message 處理函數改為呼叫我們指定的函數,指定的函數把我們想要處理的 Message 處理完,再把剩下的 Message 丟回給 Unity 來處理。



原本實作是使用 dllimport 讓 SetWindowLong function 可以在 C# 裡呼叫, 並把整個 callback function 及流程實作出來,且測試時運作都相當正常,但程式在關閉時會出現 Access Violation 的錯誤,後來將整個實作改成 C DLL 之後錯誤才沒有出現,不知道是什麼原因?底下是 DLL 部分的原始碼:



#include "stdafx.h"


LRESULT CALLBACK SubWndProc(
HWND hWnd,
UINT nMessage,
WPARAM wParam, LPARAM lParam);

WNDPROC  gOldWndProc = NULL;
HWND  gUnityWnd = NULL;

#ifdef    __cplusplus
extern "C" {
#endif    /*    __cplusplus    */

__declspec(dllexport) bool __stdcall init(HWND hWnd)
{
  gOldWndProc =(WNDPROC)SetWindowLong(hWnd, GWL_WNDPROC, (LONG)SubWndProc);
  gUnityWnd =hWnd;

  if (gOldWndProc !=NULL)
   return ***e;

  return false;
}

__declspec(dllexport) void __stdcall release()
{
  SetWindowLong(gUnityWnd, GWL_WNDPROC, (LONG)gOldWndProc);
  gOldWndProc =0;
  gUnityWnd =0;
}

#ifdef    __cplusplus
}
#endif    /*    __cplusplus    */


LRESULT CALLBACK SubWndProc(
HWND hWnd,
UINT nMessage,
WPARAM wParam, LPARAM lParam)
{
switch(nMessage)
{
  case WM_IME_SETCONTEXT:
  case WM_IME_STARTCOMPOSITION:
  case WM_IME_ENDCOMPOSITION:
  case WM_IME_COMPOSITION:
  case WM_IME_REQUEST:
   {
    //...
   }
   break;
}
return CallWindowProc(gOldWndProc, hWnd, nMessage, wParam, lParam);
}


Unity 可以透過呼叫 DLL 提供的 init() 函數,讓 Windows 改為呼叫我們指定的函數 (SubWndProc) 來處理 Message,透過 release() 函數讓 Message 處理流程復原。底下是 Unity 部分的原始碼(DLL 檔名為 UnityIMEDLL.dll 且檔案放在 Assets/Plugins 目錄下)

using UnityEngine;

using System;
using System.Collections;
using System.Collections.Generic;
using System.Runtime.InteropServices;

public class IMEInputBox : MonoBehaviour
{
    //-----------------------------------------------------------
    [DllImport("UnityIMEDLL")]
    protected static extern bool init(IntPtr hWnd);

    [DllImport("UnityIMEDLL")]
    protected static extern void release();

    [DllImport("user32")]
    protected static extern IntPtr GetActiveWindow();

    //-----------------------------------------------------------
// Use this for initialization
void Start ()
{
        Debug.Log("init UnityIMEDLL.");
        try
        {
            init(GetActiveWindow());
        }
        catch (Exception e)
        {
            Debug.Log(e.ToString());
        }
}

    void OnDisable()
    {
        Debug.Log("release UnityIMEDLL.");
        try
        {
            release();
        }
        catch (Exception e)
        {
            Debug.Log(e.ToString());
        }
    }
}






欢迎光临 纳金网 (http://old.narkii.com/club/) Powered by Discuz! X2.5