如何禁用控制檯視窗的關閉按鈕?

技術譯民發表於2021-01-06

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

使用場景

有時候,我們需要編寫一個具有一定處理邏輯的控制檯程式,這比編寫 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("按Enter鍵退出");

            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

總結

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


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

相關文章