WPF中窗體最大化問題處理


遇到的問題信息

  • 問題:當WindowStyle=None時,窗口最大化,不顯示任務欄 —— 即窗體是全屏效果。
  • 解決中遇到的問題列表【主要涉及到任務欄發生改變后的一些問題處理】:
    • 最大化時,任務欄被遮蓋;
    • 最大化后,拖動任務欄,無法自適應窗體;
    • 最大化后,拖動任務欄,窗體還原,還原數據丟失,始終顯示最大;
    • 最大化后,拖動任務欄,窗體還原,設置之前保存的窗體位置數據,再次設置,由於和之前一樣,窗體位置信息不生效;

解決方案

  • 思路:窗體最大化時,將窗體透明化,設置內部元素Grid的Margin屬性,從而顯示出任務欄
  • 步驟:

1、 設置窗體相關屬性:WindowStyle="None" AllowsTransparency="True" Background="Transparent" ResizeMode="CanMinimize"

窗體需要支持透明,並將窗體設置為透明;設置ResizeMode,否則最大化時,邊框會有影響。

2、 添加窗體最大化/還原代碼如下:

    double normaltop;
    double normalleft;
    double normalwidth;
    double normalheight;
    /// <summary>
    /// 最大化/還原處理
    /// </summary>
    /// <param name="sender"></param>
    /// <param name="e"></param>
    private void Btn_maximize_Click(object sender, RoutedEventArgs e)
    {
        //wpf最大化 全屏顯示任務欄處理
        if (this.WindowState == WindowState.Normal)
        {
            normaltop = this.Top;
            normalleft = this.Left;
            normalwidth = this.Width;
            normalheight = this.Height;

            double top = SystemParameters.WorkArea.Top;
            double left = SystemParameters.WorkArea.Left;
            double right = SystemParameters.PrimaryScreenWidth - SystemParameters.WorkArea.Right;
            double bottom = SystemParameters.PrimaryScreenHeight - SystemParameters.WorkArea.Bottom;
            gd_main.Margin = new Thickness(left, top, right, bottom);

            this.WindowState = WindowState.Maximized;               
        }
        else
        {
            this.WindowState = WindowState.Normal;

            //必須先設置為0,在重新設值,若前后值一樣,會失效 --拖動任務欄后,還原-始終顯示在屏幕最左上方
            this.Top = 0;
            this.Left = 0;
            this.Width = 0;
            this.Height = 0;

            this.Top = normaltop;
            this.Left = normalleft;
            this.Width = normalwidth;
            this.Height = normalheight;

            gd_main.Margin = new Thickness(0);
        }
    }

3、添加任務欄變化處理

注意:此節實現僅適用於.Net Framework 4.5及以上。因為4.0及以前的版本中不包含StaticPropertyChanged事件。
不過可以通過WndProc來自己實現,其中會有一個問題:利用WndProc來監聽,結果是比較實時的,而SystemParameters中的值,實時性可能會跟不上,從而獲取到的值仍然是舊的。

對此有兩種解決方案:

  • 1、添加一個Timer 或者 直接Sleep,等待一下在SystemParameters的值【此法相對簡單,但無法完全保證有效,畢竟SystemParameters中值更新的時間,還是要看.Net Framework】;
  • 2、利用Windows API讀取系統值,用方法SystemParametersInfo獲取SPI_GETWORKAREA【微軟官方其實就是用這個獲取的,這樣比我下面的運行效率還要高些】

另外:下面的方法可能會有系統兼容性問題,我在Windows 10上是通過的,但在Windows 8.1上,邊界存在問題【不確定是不是Framework在系統上有bug】

注冊事件:SystemParameters.StaticPropertyChanged += SystemParameters_StaticPropertyChanged;
添加如下代碼:

    private void SystemParameters_StaticPropertyChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e)
    {
        if (e.PropertyName == "WorkArea")
        {
            if (this.WindowState == WindowState.Maximized)
            {
                double top = SystemParameters.WorkArea.Top;
                double left = SystemParameters.WorkArea.Left;
                double right = SystemParameters.PrimaryScreenWidth - SystemParameters.WorkArea.Right;
                double bottom = SystemParameters.PrimaryScreenHeight - SystemParameters.WorkArea.Bottom;
                gd_main.Margin = new Thickness(left, top, right, bottom);
            }
        }
    }

相關下載

點擊查看完整源代碼

解:奇葩史


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM