如何禁用控制台窗口的關閉按鈕?

這是一段古老的代碼,也是我以前經常用到的代碼。雖然現在和以後基本上都不會再用到它了,但是在特定的場景中,它很好用。

使用場景

有時候,我們需要編寫一個具有一定處理邏輯的控制台程序,這比編寫 Windows 服務要簡單一些。但是,我們要防止不小心點擊到控制台窗口右上角的關閉按鈕而導致程序非正常退出。於是就有了如這篇文章標題所述的一個簡單的需求。

代碼實現

查找 Windows 窗口和禁用 Windows 窗口的按鈕,需要用到 Windows API FindWindowGetSystemMenuRemoveMenu,具體的代碼實現如下所示,可以將代碼複製到控制台項目中直接運行:

using System;
using System.Runtime.InteropServices;
using System.Threading;

namespace Demo
{
    class Program
    {
        static void Main(string[] args)
        {
            string title = $"程序 {DateTime.Now} 啟動";
            //修改控制台窗口標題
            Console.Title = title;
            //禁用控制台窗口關閉按鈕
            DisableCloseButton(title);

            //檢測指定 title 的控制台窗口是否存在
            bool isExist = IsExistsConsole(title);

            Console.WriteLine($"isExist = {isExist},窗口標題:{title}");

            Console.WriteLine("按回車鍵退出");

            Console.ReadLine();
        }

        #region 禁用控制台窗口關閉按鈕
        [DllImport("user32.dll", EntryPoint = "FindWindow")]
        static extern IntPtr FindWindow(string lpClassName, string lpWindowName);

        [DllImport("user32.dll", EntryPoint = "GetSystemMenu")]
        static extern IntPtr GetSystemMenu(IntPtr hWnd, IntPtr bRevert);

        [DllImport("user32.dll", EntryPoint = "RemoveMenu")]
        static extern IntPtr RemoveMenu(IntPtr hMenu, uint uPosition, uint uFlags);

        ///<summary>
        /// 禁用控制台窗口關閉按鈕
        ///</summary>
        ///<param name="title">窗口標題</param>
        public static void DisableCloseButton(string title)
        {
            //線程休眠,確保能夠正常 FindWindow,否則有時會 Find 失敗。
            Thread.Sleep(100);

            IntPtr windowHandle = FindWindow(null, title);
            IntPtr closeMenu = GetSystemMenu(windowHandle, IntPtr.Zero);
            const uint SC_CLOSE = 0xF060;
            RemoveMenu(closeMenu, SC_CLOSE, 0x0);
        }

        /// <summary>
        /// 檢測指定 title 的控制台窗口是否存在
        /// </summary>
        /// <param name="title">windows 窗口標題</param>
        /// <returns></returns>
        public static bool IsExistsConsole(string title)
        {
            IntPtr windowHandle = FindWindow(null, title);
            if (windowHandle.Equals(IntPtr.Zero)) return false;

            return true;
        }
        #endregion
    }
}

它的運行結果如下:

disable close button

總結

如上所述,代碼很簡單,實現的功能也很簡單。只是覺得以後基本上不會再用到它了,聊以記之,以防永久遺忘。如若恰好對您有用,不勝榮幸。

作者 : 技術譯民
出品 : 技術譯站