Как обрабатывать сообщения WndProc в WPF? - PullRequest
104 голосов
/ 09 марта 2009

Нахождение WPF крутой кривой обучения.

В старых добрых Windows Forms я просто переопределяю WndProc и начинаю обрабатывать сообщения по мере их поступления.

Может кто-нибудь показать мне пример того, как добиться того же в WPF?

Ответы [ 8 ]

127 голосов
/ 18 декабря 2009

Вы можете сделать это через пространство имен System.Windows.Interop, которое содержит класс с именем HwndSource.

Пример использования

using System;
using System.Windows;
using System.Windows.Interop;

namespace WpfApplication1
{
    public partial class Window1 : Window
    {
        public Window1()
        {
            InitializeComponent();
        }

        protected override void OnSourceInitialized(EventArgs e)
        {
            base.OnSourceInitialized(e);
            HwndSource source = PresentationSource.FromVisual(this) as HwndSource;
            source.AddHook(WndProc);
        }

        private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
        {
            // Handle messages...

            return IntPtr.Zero;
        }
    }
}

Полностью взято из отличного поста в блоге: Использование пользовательского WndProc в приложениях WPF от Стива Рэндса

55 голосов
/ 09 марта 2009

На самом деле, насколько я понимаю, такая вещь действительно возможна в WPF с использованием HwndSource и HwndSourceHook. См. этот поток на MSDN в качестве примера. (Соответствующий код указан ниже)

// 'this' is a Window
HwndSource source = HwndSource.FromHwnd(new WindowInteropHelper(this).Handle);
source.AddHook(new HwndSourceHook(WndProc));

private static IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
{
    //  do stuff

    return IntPtr.Zero;
}

Теперь я не совсем уверен, почему вы хотите обрабатывать сообщения Windows Messaging в приложении WPF (если это не самая очевидная форма взаимодействия для работы с другим приложением WinForms). Идеология дизайна и природа API в WPF сильно отличаются от WinForms, поэтому я бы посоветовал вам просто больше ознакомиться с WPF, чтобы точно узнать , почему не имеет эквивалента WndProc.

14 голосов
/ 14 июля 2009
HwndSource src = HwndSource.FromHwnd(new WindowInteropHelper(this).Handle);
src.AddHook(new HwndSourceHook(WndProc));


.......


public IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
{

  if(msg == THEMESSAGEIMLOOKINGFOR)
    {
      //Do something here
    }

  return IntPtr.Zero;
}
2 голосов
/ 02 февраля 2017

Если вы не возражаете против ссылки на WinForms, вы можете использовать более MVVM-ориентированное решение, которое не связывает сервис с представлением. Вам необходимо создать и инициализировать System.Windows.Forms.NativeWindow, которое представляет собой облегченное окно для приема сообщений.

public abstract class WinApiServiceBase : IDisposable
{
    /// <summary>
    /// Sponge window absorbs messages and lets other services use them
    /// </summary>
    private sealed class SpongeWindow : NativeWindow
    {
        public event EventHandler<Message> WndProced;

        public SpongeWindow()
        {
            CreateHandle(new CreateParams());
        }

        protected override void WndProc(ref Message m)
        {
            WndProced?.Invoke(this, m);
            base.WndProc(ref m);
        }
    }

    private static readonly SpongeWindow Sponge;
    protected static readonly IntPtr SpongeHandle;

    static WinApiServiceBase()
    {
        Sponge = new SpongeWindow();
        SpongeHandle = Sponge.Handle;
    }

    protected WinApiServiceBase()
    {
        Sponge.WndProced += LocalWndProced;
    }

    private void LocalWndProced(object sender, Message message)
    {
        WndProc(message);
    }

    /// <summary>
    /// Override to process windows messages
    /// </summary>
    protected virtual void WndProc(Message message)
    { }

    public virtual void Dispose()
    {
        Sponge.WndProced -= LocalWndProced;
    }
}

Используйте SpongeHandle, чтобы зарегистрироваться для сообщений, которые вас интересуют, а затем переопределите WndProc для их обработки:

public class WindowsMessageListenerService : WinApiServiceBase
{
    protected override void WndProc(Message message)
    {
        Debug.WriteLine(message.msg);
    }
}

Единственным недостатком является то, что вы должны включить ссылку System.Windows.Forms, но в остальном это очень инкапсулированное решение.

Подробнее об этом можно прочитать здесь

0 голосов
/ 08 сентября 2017

Вы можете присоединиться к классу 'SystemEvents' встроенного класса Win32:

using Microsoft.Win32;

в классе окон WPF:

SystemEvents.PowerModeChanged += SystemEvents_PowerModeChanged;
SystemEvents.SessionSwitch += SystemEvents_SessionSwitch;
SystemEvents.SessionEnding += SystemEvents_SessionEnding;
SystemEvents.SessionEnded += SystemEvents_SessionEnded;

private async void SystemEvents_PowerModeChanged(object sender, PowerModeChangedEventArgs e)
{
    await vm.PowerModeChanged(e.Mode);
}

private async void SystemEvents_PowerModeChanged(object sender, PowerModeChangedEventArgs e)
{
    await vm.PowerModeChanged(e.Mode);
}

private async void SystemEvents_SessionSwitch(object sender, SessionSwitchEventArgs e)
{
    await vm.SessionSwitch(e.Reason);
}

private async void SystemEvents_SessionEnding(object sender, SessionEndingEventArgs e)
{
    if (e.Reason == SessionEndReasons.Logoff)
    {
        await vm.UserLogoff();
    }
}

private async void SystemEvents_SessionEnded(object sender, SessionEndedEventArgs e)
{
    if (e.Reason == SessionEndReasons.Logoff)
    {
        await vm.UserLogoff();
    }
}
0 голосов
/ 09 марта 2009

Существуют способы обработки сообщений с помощью WndProc в WPF (например, с использованием HwndSource и т. Д.), Но обычно эти методы зарезервированы для взаимодействия с сообщениями, которые не могут напрямую обрабатываться через WPF. Большинство элементов управления WPF даже не являются окнами в смысле Win32 (и расширением Windows.Forms), поэтому у них не будет WndProcs.

0 голосов
/ 09 марта 2009

Короткий ответ: ты не можешь. WndProc работает, передавая сообщения HWND на уровне Win32. Окна WPF не имеют HWND и, следовательно, не могут участвовать в сообщениях WndProc. Базовый цикл сообщений WPF находится поверх WndProc, но абстрагирует их от базовой логики WPF.

Вы можете использовать HWndHost и получить WndProc для него. Однако это почти наверняка не то, что вы хотите сделать. Для большинства целей WPF не работает с HWND и WndProc. Ваше решение почти наверняка основано на внесении изменений в WPF, а не в WndProc.

0 голосов
/ 09 марта 2009

WPF не работает с WinForms типа wndprocs

Вы можете разместить HWndHost в соответствующем элементе WPF, а затем переопределить wndproc Hwndhost, но AFAIK это так близко, как вы собираетесь получить.

http://msdn.microsoft.com/en-us/library/ms742522.aspx

http://blogs.msdn.com/nickkramer/archive/2006/03/18/554235.aspx

...