Событие WindowChanging окна в WPF - PullRequest
12 голосов
/ 29 мая 2009

Мне нужно обработать приложение WPF прямо перед тем, как оно будет свернуто. Не тогда, когда оно уже есть. Я обнаружил в Window объект StateChanged, но он срабатывает, когда объект Window уже находится в состоянии «Свернуть», тогда уже слишком поздно.

Итак, мне нужно что-то вроде события StateChanging для обработки, пока объект Window все еще находится в предыдущем состоянии.

Можно ли создать такое событие?

Ответы [ 2 ]

13 голосов
/ 30 мая 2009

Найденные сообщения Windows вызываются в окно прямо перед сворачиванием с помощью Spy ++. Первый из них называется WM_WINDOWPOSCHANGING. Я не знал, что Windows перемещает окно на точку -32000, -32000, когда минимизирует вдову, и это были параметры WM_WINDOWPOSCHANGING. Хотя я тестировал только на Vista. http://blogs.msdn.com/oldnewthing/archive/2004/10/28/249044.aspx

код, использованный здесь, был опубликован Nir здесь

вот пример кода

XAML :

<Window x:Class="WindowStateTest2.Window1"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
Title="Window1" Height="300" Width="300">
<Grid>
    <Grid.RowDefinitions>
        <RowDefinition Height="Auto"></RowDefinition>

        <RowDefinition Height="*"></RowDefinition>
    </Grid.RowDefinitions>
        <Button Click="btnClear_Click" Grid.Row="0" x:Name="btnClear">Clear</Button>            

        <TextBox Name="txt" VerticalScrollBarVisibility="Visible" Grid.Row="2"></TextBox>
</Grid>
</Window>

C #

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Windows.Interop;
using System.Runtime.InteropServices;

namespace WindowStateTest2
{
/// <summary>
/// Interaction logic for Window1.xaml
/// </summary>
public partial class Window1 : Window
{
    public Window1()
    {
        InitializeComponent();

        this.StateChanged += new EventHandler(Window1_StateChanged);
        this.SourceInitialized += new EventHandler(Window1_SourceInitialized);

    }

    #region Event handlers

    void btnClear_Click(object sender, RoutedEventArgs e)
    {
        this.txt.Text = string.Empty;
    }
    void Window1_SourceInitialized(object sender, EventArgs e)
    {
        AttachWndProc();
    }

    void Window1_StateChanged(object sender, EventArgs e)
    {
        if (this.WindowState == WindowState.Minimized)
            Console.WriteLine("SC: " + this.WindowState);
    } 

    #endregion

    #region Const

    private int SYSCOMMAND = 0x0112;
    private int SC_MINIMIZE = 0xf020;
    private int WINDOWPOSCHANGING = 0x0046;

    #endregion

    private void AttachWndProc()
    {
        HwndSource source = HwndSource.FromHwnd(new WindowInteropHelper(this).Handle);
        source.AddHook(new HwndSourceHook(WndProc));
    }

    [StructLayout(LayoutKind.Sequential)]
    internal struct WINDOWPOSPARAMS
    {
        public IntPtr hwnd;
        public IntPtr hwndInsertAfter;
        public int x;
        public int y;
        public int cx;
        public int cy;
        public int flags;
    }


    private IntPtr WndProc(IntPtr hwnd, int msg, IntPtr wParam, IntPtr lParam, ref bool handled)
    {
        if (msg == WINDOWPOSCHANGING)               
        {
            WINDOWPOSPARAMS param = (WINDOWPOSPARAMS)Marshal.PtrToStructure(lParam, typeof(WINDOWPOSPARAMS));
            if (param.x == -32000 && param.y == -32000)
            {
                Output("");

                                    // EVENT WOULD BE RAISED HERE

                Output("State before minimize:");
                Output(string.Format("CurrentState: {0}", this.WindowState));
                Output(string.Format("Location {0} {1}: ", this.Top, this.Left));
                Output("");
            }
        }

        // process minimize button
        if (msg == SYSCOMMAND && SC_MINIMIZE == wParam.ToInt32())
        {
            Output("Minimize clicked");             
        }

        handled = false;
        return IntPtr.Zero;
    }

    public void Output(object output)
    {
        this.txt.Text += output.ToString();
        this.txt.Text += Environment.NewLine;           
    }       

}
}
0 голосов
/ 30 мая 2009

Не думаю, что вы сможете сделать это напрямую.

A Минимизация вызова окна может происходить из нескольких мест, а не только из-за кнопки минимизации в окне Chrome (например, щелчок правой кнопкой мыши на панели задач или из диспетчера задач Windows) и AFAIK, прямого способа нет обрабатывать события кнопок, запускаемые из Window Chrome (если кто-то знает, как это сделать, пожалуйста, дайте мне знать!).

Хорошая новость заключается в том, что вы можете подделать ее, но это не тривиально, поэтому вам придется решить, стоит ли это того. Во-первых, вам придется заменить стандартный Window Chrome своим собственным. Вы можете узнать, как это сделать здесь .

Во-вторых, вам нужно будет создать свои собственные кнопки «Развернуть / Свернуть / Закрыть» и связать события с соответствующим поведением. Поскольку это ваш собственный пользовательский интерфейс, вы можете слушать и отменять события Button по своему выбору.

Имейте в виду, что вы все равно не сможете обнаружить или помешать пользователям свернуть с помощью панели задач / диспетчера задач Windows, поэтому это может быть не совсем то, что вы ищете.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...