色女孩综合网_天天做天天添婷婷我也去 _日韩视频高清_草草影院免费

完善主體資料,免費贈送VIP會員!
* 主體類型
* 企業名稱
* 信用代碼
* 所在行業
* 企業規模
* 所在職位
* 姓名
* 所在行業
* 學歷
* 工作性質
請先選擇行業
您還可以選擇以下福利:
行業福利,領完即止!

下載app免費領取會員

NULL

ad.jpg

Revit二次開發教程:Revit API Hook 之 攔截鼠標雙擊元素事件

發布于:2019-06-22 17:55:56

網友投稿

更多

HOOK(鉤子,掛鉤)是一種實現Windows平臺下類似于中斷的機制。HOOK機制允許應用程序攔截并處理Windows消息或指定事件,當指定的消息發出后,HOOK程序就可以在消息到達目標窗口之前將其捕獲,從而得到對消息的控制權,進而可以對該消息進行處理或修改,加入我們所需的功能。鉤子按使用范圍分,可分為線程鉤子和系統鉤子,其中,系統鉤子具有相當大的功能,幾乎可以實現對所有Windows消息的攔截、處理和監控。這項技術涉及到兩個重要的API,一個是SetWindowsHookEx,安裝鉤子;另一個是UnHookWindowsHookEx,卸載鉤子。


本文使用的HOOK API技術,是指截獲系統或進程對某個API函數的調用,使得API的執行流程轉向我們指定的代碼段,從而實現我們所需的功能。Windows下的每個進程均擁有自己的地址空間,并且進程只能調用其地址空間內的函數,因此HOOK API尤為關鍵的一步是,設法將自己的代碼段注入到目標進程中,才能進一步實現對該進程調用的API進行攔截。然而微軟并沒有提供HOOK API的調用接口,這就需要開發者自己編程實現。

 一般來說,HOOK API由兩個組成部分,即實現HOOK API的DLL文件,和啟動注入的主調程序。本文采用HOOK API 技術對剪切板相關的API 函數進行攔截,從而實現對剪切板內容的監控功能,同樣使用該技術實現進程防終止功能。其中DLL文件支持HOOK API的實現,而主調客戶端程序將在初始化時把帶有HOOK API功能的DLL隨著鼠標鉤子的加載注入到目標進程中,這里的鼠標鉤子屬于系統鉤子。

下面介紹在Revit中,如何應用Hook對鼠標雙擊元素事件進行攔截。


第一步,先封裝HookBase抽象類,因所有Hook的都具有注冊、卸載邏輯,且注冊、卸載大同小易。如下

public abstract class HookBase : IHook

    {

        private static Dictionary<int, IHook> m_Hooks;

        private IntPtr m_ProcessId;

        private int m_ThreadId;

        private HookType m_HookType;

        private HookProc m_HookProc; 

        protected internal int m_HookId; 

        static HookBase(){

            m_Hooks = new Dictionary<int, IHook>();

        } 

        private HookBase(HookType hookType){

            m_HookType = hookType;

            m_HookProc = HookProc;

        } 

        protected HookBase(IntPtr processId, HookType hookType):this(hookType){

            m_ProcessId = processId;

            if (m_ProcessId == IntPtr.Zero)

            {

                m_ProcessId = HookHelper.GetCurrentProcessId();

            }

        } 

        protected HookBase(int threadId, HookType hookType):this(hookType){

            m_ThreadId = threadId;

            if (m_ThreadId == 0)

            {

                m_ThreadId = HookHelper.GetCurrentThreadId();

            }

        } 

        public void Install(){

            if (m_ThreadId != 0)

            {

                m_HookId = HookHelper.SetWindowsHookEx(m_HookType, m_HookProc, IntPtr.Zero, m_ThreadId);

            }

            else

            {

                if (m_ProcessId == IntPtr.Zero)

                {

                    return;

                }

                m_HookId = HookHelper.SetWindowsHookEx(m_HookType, m_HookProc, m_ProcessId,0);

            }


            if (m_HookId == 0)

            {

                return;

            }


            if (!m_Hooks.ContainsKey(m_HookId))

            {

                m_Hooks.Add(m_HookId, this);

            }

        } 

        public void Uninstall()

        {

            if (m_HookId == 0)

            {

                return;

            }


            var flag = HookHelper.UnhookWindowsHookEx(m_HookId);

            if (flag)

            {

                if (m_Hooks.Remove(m_HookId))

                {

                    m_HookId = 0;

                }

            }

        } 

        protected abstract int HookProc(int nCode, IntPtr wParam, IntPtr lParam);


第二步 ,因鼠標Hook分為線程鼠標Hook以及全局鼠標Hook兩種,僅注冊方式有點區別。為使用方便,將其封裝為事件注冊方式。如下

public abstract class MouseHookBase : HookBase

    {

        protected MouseHookBase(IntPtr processId)

            : base(processId, HookType.WH_MOUSE_LL)

        {


        }


        protected MouseHookBase(int threadId)

            : base(threadId, HookType.WH_MOUSE)

        {


        }

        public event HookHandler<MouseEventArgs> MouseDoubleClick;

        public event HookHandler<MouseEventArgs> MouseMove;

        public event HookHandler<MouseEventArgs> MouseDown;

        public event HookHandler<MouseEventArgs> MouseUp;


        protected override int HookProc(int nCode, IntPtr wParam, IntPtr lParam)

        {

            if (nCode < 0)

            {

                return HookHelper.CallNextHookEx(m_HookId, nCode, wParam, lParam);

            }


            var mouseMsg = (MouseMessage)wParam.ToInt32();

            var mouseHookStruct = lParam.ToStruct<MOUSEHOOKSTRUCT>();


            var button = this.GetMouseButtons(mouseMsg);


            switch (mouseMsg)

            {

                case MouseMessage.WM_LBUTTONDOWN:

                case MouseMessage.WM_RBUTTONDOWN:

                case MouseMessage.WM_MBUTTONDOWN:


                    return this.OnRaiseMouseDown(button, 1, mouseHookStruct.pt.X, mouseHookStruct.pt.Y, mouseHookStruct.mouseData);


                case MouseMessage.WM_LBUTTONUP:

                case MouseMessage.WM_MBUTTONUP:

                case MouseMessage.WM_RBUTTONUP:


                    return this.OnRaiseMouseUp(button, 1, mouseHookStruct.pt.X, mouseHookStruct.pt.Y, mouseHookStruct.mouseData);


                case MouseMessage.WM_LBUTTONDBLCLK:

                case MouseMessage.WM_RBUTTONDBLCLK:

                case MouseMessage.WM_MBUTTONDBLCLK:


                    return this.OnRaiseMouseDoubleClick(button, 2, mouseHookStruct.pt.X, mouseHookStruct.pt.Y, mouseHookStruct.mouseData);


                case MouseMessage.WM_MOUSEMOVE:


                    return this.OnRaiseMouseMove(MouseButtons.None, 0, mouseHookStruct.pt.X, mouseHookStruct.pt.Y, mouseHookStruct.mouseData);

                default:

                    return HookHelper.CallNextHookEx(m_HookId, nCode, wParam, lParam);

            }

        }


        private MouseButtons GetMouseButtons(MouseMessage mouseMsg)

        {

            MouseButtons result = MouseButtons.None;

            switch (mouseMsg)

            {

                case MouseMessage.WM_LBUTTONDBLCLK:

                case MouseMessage.WM_LBUTTONDOWN:

                case MouseMessage.WM_LBUTTONUP:

                    result = MouseButtons.Left;

                    break;

                case MouseMessage.WM_MBUTTONDBLCLK:

                case MouseMessage.WM_MBUTTONDOWN:

                case MouseMessage.WM_MBUTTONUP:

                    result = MouseButtons.Middle;

                    break;

                case MouseMessage.WM_RBUTTONDBLCLK:

                case MouseMessage.WM_RBUTTONDOWN:

                case MouseMessage.WM_RBUTTONUP:

                    result = MouseButtons.Right;

                    break;

            }

            return result;

        }


        private int OnRaiseMouseDoubleClick(MouseButtons button, int clicks, int x, int y, intdelta)

        {

            if (this.MouseDoubleClick != null)

            {

                return this.MouseDoubleClick(this, new MouseEventArgs(button, clicks, x, y, delta));

            }

            return 0;

        }


        private int OnRaiseMouseDown(MouseButtons button, int clicks, int x, int y, int delta)

        {

            if (this.MouseDown != null)

            {

                return this.MouseDown(this, new MouseEventArgs(button, clicks, x, y, delta));

            }

            return 0;

        }


        private int OnRaiseMouseUp(MouseButtons button, int clicks, int x, int y, int delta)

        {

            if (this.MouseUp != null)

            {

                return this.MouseUp(this, new MouseEventArgs(button, clicks, x, y, delta));

            }

            return 0;

        }


        private int OnRaiseMouseMove(MouseButtons button, int clicks, int x, int y, int delta)

        {

            if (this.MouseMove != null)

            {

                return this.MouseMove(this, new MouseEventArgs(button, clicks, x, y, delta));

            }

            return 0;

        }

}

第三步,依次實現線程鼠標Hook以及全局鼠標Hook.

 public class MouseHook : MouseHookBase{

        public MouseHook(int threadId = 0)

            : base(threadId)

        { 

        }

    }

    public class GlobalMouseHook : MouseHookBase

    {

        public GlobalMouseHook(IntPtr processId)

            : base(processId)

        { 

        }

 }

第四步,有了鼠標Hook,我們如果在Revit內使用并且攔截鼠標雙擊元素事件呢?我們繼續封裝一個元素監控類 ,如下:

public class ElementMonitor

    {

        private static ElementMonitor m_Instance;

        private MouseHook m_MouseHook;

        private bool m_IsMonitor;

        private UIApplication m_UIApplication;


        private ElementMonitor(UIApplication uiApp)

        {

            m_Instance = this;

            m_UIApplication = uiApp;


            m_MouseHook = new MouseHook();

            m_MouseHook.Install();


            m_MouseHook.MouseDoubleClick += OnRaiseMouseDoubleClick;

        }


        /// <summary>

        /// 靜態實例,可在入口類判斷此實例是否為null,防止重復注冊.

        /// </summary>

        public static ElementMonitor Instance

        {

            get

            {

                return m_Instance;

            }

        }


        /// <summary>

        /// 當鼠標雙擊元素時觸發此事件.

        /// </summary>

        public event HookHandler<DoubleClickElementEventArgs> DoubleClickElement;


        /// <summary>

        /// 注冊元素監控,并指定是否立即監控.

        /// </summary>

        public static void Register(UIApplication uiApp, bool immediatelyMonitor = true)

        {

            if (uiApp == null)

            {

                throw new ArgumentNullException(nameof(uiApp));

            }


            new ElementMonitor(uiApp)

            {

                m_IsMonitor = immediatelyMonitor

            };

        }


        /// <summary>

        /// 注冊元素監控,并指定是否立即監控.

        /// </summary>

        public static void Register(UIControlledApplication uiControllApp, bool immediatelyMonitor = true)

        {

            if (uiControllApp == null)

            {

                throw new ArgumentNullException(nameof(uiControllApp));

            }


            var flag = System.Reflection.BindingFlags.NonPublic | System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.InvokeMethod;


            var uiApp = (UIApplication)uiControllApp.GetType().InvokeMember("getUIApplication", flag, Type.DefaultBinder, uiControllApp, null);


            Register(uiApp, immediatelyMonitor);

        }


        /// <summary>

        /// 返回1,則攔截鼠標消息,返回0則傳遞給真正消息接收者.

        /// </summary>

        private int OnRaiseMouseDoubleClick(object sender, System.Windows.Forms.MouseEventArgs e)

        {

            if (!m_IsMonitor || e.Button != MouseButtons.Left || e.Clicks != 2)

            {

                return 0;

            }


            var uiDoc = m_UIApplication.ActiveUIDocument;


            if (uiDoc == null)

            {

                return 0;

            }


            var elemIds = uiDoc.Selection.GetElementIds();


            if (elemIds.Count == 1)

            {

                var elem = uiDoc.Document.GetElement(elemIds.First());


                if (elem == null)

                {

                    return 0;

                } 

                if (this.DoubleClickElement == null)

                {

                    return 0;

                } 

                return this.DoubleClickElement(this, new DoubleClickElementEventArgs(elem));

            }


            return 0;

        }

    }

第五步,調用測試,如下

[Transaction(TransactionMode.Manual)]

    public class MouseHookTest : IExternalCommand

    {

        Result IExternalCommand.Execute(ExternalCommandData commandData, ref string message, ElementSet elements)

        {

            if (ElementMonitor.Instance == null)

            {

                ElementMonitor.Register(commandData.Application);

            }


            ElementMonitor.Instance.DoubleClickElement += OnRaiseDoubleClickElement;


            return Result.Succeeded;

        }


        private int OnRaiseDoubleClickElement(object sender, DoubleClickElementEventArgs e)

        {

            if (e.Element == null)

            {

                return 0;

            }


            System.Windows.Forms.MessageBox.Show(string.Format("雙擊擊元素Id: {0}", e.Element.Id)); 

            return 1;


        }

    }



本文版權歸腿腿教學網及原創作者所有,未經授權,謝絕轉載。

未標題-1.jpg

上一篇:Revit二次開發教程:Revit族參數可見性設置

下一篇:Revit二次開發教程:注冊Revit插件

60acb4e0ef112.png
色女孩综合网_天天做天天添婷婷我也去 _日韩视频高清_草草影院免费

    9000px;">

      欧美精品日韩精品| 中文字幕一区免费在线观看| 欧美激情资源网| 亚洲与欧洲av电影| 久久精品国产精品亚洲精品| www.66久久| 欧美一区二区三区四区视频 | 在线精品亚洲一区二区不卡| 亚洲精品一区二区三区精华液| 亚洲欧美日韩国产手机在线| 韩国一区二区视频| 欧美电影一区二区| 国产精品免费观看视频| 久久超碰97中文字幕| 欧美日韩国产综合视频在线观看| 国产精品久线在线观看| 美女一区二区久久| 欧美另类一区二区三区| 亚洲精选视频在线| 99久久久精品| 中文一区二区在线观看| 国产一区二区伦理| 精品欧美乱码久久久久久1区2区| 天堂va蜜桃一区二区三区漫画版| 日本道在线观看一区二区| 国产精品久久看| 成人理论电影网| 久久亚洲春色中文字幕久久久| 天天影视网天天综合色在线播放| 在线免费观看不卡av| 中文字幕日韩一区| 成人国产电影网| 18涩涩午夜精品.www| a级精品国产片在线观看| 久久精品男人的天堂| 国产裸体歌舞团一区二区| 欧美r级在线观看| 精品亚洲欧美一区| 久久久久久久久久久久电影| 狠狠色狠狠色综合日日91app| 精品福利在线导航| 寂寞少妇一区二区三区| 精品美女一区二区| 国产一区二区主播在线| 精品国产乱码久久久久久夜甘婷婷| 另类调教123区| 欧美精品一区视频| 国产suv精品一区二区883| 日本一区二区三区国色天香| av不卡一区二区三区| 成人免费视频在线观看| 欧美揉bbbbb揉bbbbb| 日本v片在线高清不卡在线观看| 日韩欧美成人激情| 国产成人夜色高潮福利影视| 欧美国产丝袜视频| 在线看一区二区| 美女视频一区二区三区| 久久久久久久久久久久久夜| 成人免费高清视频在线观看| 亚洲精品乱码久久久久久日本蜜臀| 欧美曰成人黄网| 麻豆久久一区二区| 久久欧美一区二区| 91丝袜高跟美女视频| 亚洲一区二区高清| 精品国产91洋老外米糕| 成人精品高清在线| 午夜激情一区二区三区| 久久先锋影音av| 91电影在线观看| 国产综合色产在线精品| 国产精品福利av| 7777精品伊人久久久大香线蕉完整版 | 亚洲精品日韩专区silk| 欧美一级国产精品| 大陆成人av片| 视频一区视频二区在线观看| 久久九九全国免费| 欧美日本乱大交xxxxx| 国产精品18久久久久| 亚洲成人av电影| 亚洲国产岛国毛片在线| 欧美年轻男男videosbes| 成人爽a毛片一区二区免费| 首页亚洲欧美制服丝腿| 国产精品成人网| 日韩精品一区二区三区四区视频 | 美女看a上一区| 亚洲综合图片区| 国产片一区二区| 制服丝袜av成人在线看| 91网站在线播放| 国产麻豆9l精品三级站| 亚州成人在线电影| 亚洲六月丁香色婷婷综合久久| 337p粉嫩大胆色噜噜噜噜亚洲| 欧美色综合网站| 91碰在线视频| 成人一区在线看| 喷水一区二区三区| 一区二区三区免费看视频| 欧美精彩视频一区二区三区| 精品欧美黑人一区二区三区| 欧美性三三影院| 91麻豆蜜桃一区二区三区| 国产精品91xxx| 麻豆国产欧美日韩综合精品二区 | 亚洲最新在线观看| 亚洲色图19p| 国产精品免费视频网站| 亚洲国产精品精华液ab| 精品成人免费观看| 欧美一区二区精品在线| 91精品国产综合久久香蕉的特点| 欧美日本在线一区| 欧美性受xxxx黑人xyx| 91视频精品在这里| 一本久久综合亚洲鲁鲁五月天| 成人精品高清在线| 成人白浆超碰人人人人| hitomi一区二区三区精品| 国产精品伊人色| 国产不卡高清在线观看视频| 丰满岳乱妇一区二区三区 | 久久99在线观看| 国产一区二区三区蝌蚪| 久久国产欧美日韩精品| 美女被吸乳得到大胸91| 国产在线不卡视频| 成人美女在线视频| 91国在线观看| 欧美午夜免费电影| 91精品午夜视频| 精品国产一区二区国模嫣然| 337p日本欧洲亚洲大胆色噜噜| 久久蜜桃av一区精品变态类天堂 | 男人的天堂久久精品| 精品一区二区三区蜜桃| 国产成人小视频| 一本大道综合伊人精品热热 | 色综合久久久久久久| 欧美日本一区二区在线观看| 日韩一区二区精品在线观看| 久久伊人蜜桃av一区二区| 国产精品免费免费| 亚洲成人资源在线| 狠狠色狠狠色综合日日91app| 国产成+人+日韩+欧美+亚洲| 在线欧美日韩国产| 日韩欧美激情一区| 一区精品在线播放| 天天综合天天综合色| 国产成人在线免费| 欧美午夜不卡在线观看免费| 精品理论电影在线观看| 亚洲欧美激情小说另类| 五月天一区二区三区| 成人免费电影视频| 欧美精品九九99久久| 欧美韩国日本综合| 奇米精品一区二区三区在线观看一| 成人丝袜18视频在线观看| 欧美日韩精品电影| 中文成人综合网| 青青草原综合久久大伊人精品 | 国产精品每日更新在线播放网址| 亚洲国产精品一区二区www | 免费观看久久久4p| 日本福利一区二区| 亚洲国产精品二十页| 青娱乐精品视频在线| 色婷婷综合久久| 国产精品福利一区| 国产一区在线视频| 欧美二区三区91| 一区二区国产盗摄色噜噜| 粉嫩绯色av一区二区在线观看| 91精品国产91久久久久久一区二区| 日韩伦理av电影| 国产91精品一区二区| 精品99999| 日韩高清在线电影| 欧美精品在线一区二区| 一区二区在线观看免费视频播放| 国产成人免费xxxxxxxx| www日韩大片| 精品一区二区三区在线播放| 欧美精品乱码久久久久久按摩| 亚洲一区二区三区在线看| jlzzjlzz亚洲日本少妇| 国产精品欧美精品| 国产成人精品亚洲日本在线桃色| 欧美一区二区福利视频| 亚洲高清免费在线| 91在线视频网址| 亚洲精品菠萝久久久久久久| 99精品久久99久久久久| 日韩一区有码在线| 色成人在线视频|