Listening to another window resize events in C#

后端 未结 4 686
Happy的楠姐
Happy的楠姐 2020-12-04 20:01

I am implementing a small application (observer) that needs to \"attach\" itself to the bottom of another window (observed). The latter is not a window inside the applicatio

4条回答
  •  时光取名叫无心
    2020-12-04 20:24

    I ran into the same thing in some code I was working with and discovered I couldn't inject a managed .DLL into the process.

    Not wanting to write a C++ unmanaged DLL that used SetWindowsHook, I went with a combination of SetWinEventHook, which signals when a window starts and ends a move event and a Timer to poll the window while it's moving to give it the appearance of following the window while it moves.

    Here's a (very simplified) version of a class that can do that (just replace the TODO's with code to move the window or raise an event).

    public class DockingHelper
    {
        private readonly uint m_processId, m_threadId;
    
        private readonly IntPtr m_target;
    
        // Needed to prevent the GC from sweeping up our callback
        private readonly WinEventDelegate m_winEventDelegate;
        private IntPtr m_hook;
    
        private Timer m_timer;
    
        public DockingHelper(string windowName, string className)
        {
            if (windowName == null && className == null) throw new ArgumentException("Either windowName or className must have a value");
    
            m_target = FindWindow(className, windowName);
            ThrowOnWin32Error("Failed to get target window");
    
            m_threadId = GetWindowThreadProcessId(m_target, out m_processId);
            ThrowOnWin32Error("Failed to get process id");
    
            m_winEventDelegate = WhenWindowMoveStartsOrEnds;
        }
    
        [DllImport("user32.dll", SetLastError = true)]
        private static extern bool GetWindowRect(IntPtr hwnd, out RECT lpRect);
    
        [DllImport("user32.dll", SetLastError = true)]
        private static extern bool UnhookWinEvent(IntPtr hWinEventHook);
    
        [DllImport("user32.dll")]
        private static extern IntPtr SetWinEventHook(uint eventMin, uint eventMax, IntPtr hmodWinEventProc, WinEventDelegate lpfnWinEventProc, uint idProcess, uint idThread, uint dwFlags);
    
        [DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
        private static extern IntPtr FindWindow(string lpClassName, string lpWindowName);
    
        [DllImport("user32.dll", SetLastError = true)]
        private static extern uint GetWindowThreadProcessId(IntPtr hWnd, out uint lpdwProcessId);
    
        private void ThrowOnWin32Error(string message)
        {
            int err = Marshal.GetLastWin32Error();
            if (err != 0)
                throw new Win32Exception(err, message);
        }
    
        private RECT GetWindowLocation()
        {
            RECT loc;
            GetWindowRect(m_target, out loc);
            if (Marshal.GetLastWin32Error() != 0)
            {
                // Do something useful with this to handle if the target window closes, etc.
            }
            return loc;
        }
    
        public void Subscribe()
        {
            // 10 = window move start, 11 = window move end, 0 = fire out of context
            m_hook = SetWinEventHook(10, 11, m_target, m_winEventDelegate, m_processId, m_threadId, 0);
        }
    
        private void PollWindowLocation(object state)
        {
            var location = GetWindowLocation();
            // TODO: Reposition your window with the values from location (or fire an event with it attached)
        }
    
        public void Unsubscribe()
        {
            UnhookWinEvent(m_hook);
        }
    
        private void WhenWindowMoveStartsOrEnds(IntPtr hWinEventHook, uint eventType, IntPtr hwnd, int idObject, int idChild, uint dwEventThread, uint dwmsEventTime)
        {
            if (hwnd != m_target) // We only want events from our target window, not other windows owned by the thread.
                return;
    
            if (eventType == 10) // Starts
            {
                m_timer = new Timer(PollWindowLocation, null, 10, Timeout.Infinite);
                // This is always the original position of the window, so we don't need to do anything, yet.
            }
            else if (eventType == 11)
            {
                m_timer.Dispose();
                m_timer = null;
                var location = GetWindowLocation();
                // TODO: Reposition your window with the values from location (or fire an event with it attached)
            }
        }
    
        [StructLayout(LayoutKind.Sequential)]
        private struct RECT
        {
            public int Left, Top, Right, Bottom;
        }
    
        private delegate void WinEventDelegate(IntPtr hWinEventHook, uint eventType, IntPtr hwnd, int idObject, int idChild, uint dwEventThread, uint dwmsEventTime);
    }
    

提交回复
热议问题