空閑時間研究一個小功能:winform桌面程序如何實現動態更換桌面圖標


  今天休息在家,由於天氣熱再加上疫情原因,就在家里呆着,空閑時想着,在很早以前(約3年前),產品人員跟我提了一個需求,那就是winform桌面程序的圖標能否根據節日動態更換,這種需求在移動APP上還是比較常見,比如:淘寶、天貓、京東、360等,它們在逢節假日時除了APP內容有更新,APP ICON也是都更新了的,但PC端的應用程序(APP)則很少見到說有動態更新圖標的,故當時我是直接回絕了的,明確表示做不了,但今天我仔細想了一下,其實也是可以實現的,雖然無法直接更新桌面圖標,但我們可以更新替換掉桌面的快捷文件呀!(PC端桌面的圖標本質都是一個LINK文件)想到這里我就開始設計,最終還是實現了無感知更新PC端桌面圖標的功能。

先看實現方案的流程圖如下:

 

其中:DynamicIconApp【原生真實程序】、AppLauncher【引導啟動程序】 均是我演示的DEMO程序

 如上方案核心實現思路與步驟是:

1.桌面快捷方式連接的程序是啟動程序(即:前置程序),而非真實要打開的程序,目的是:如果要替換桌面快捷方式必需是另外進程來執行,如果快捷方式打開的是真實程序,而真實程序又來更新替換桌面快捷方式文件,會被該桌面快捷方式文件被占用; 【當然也可以不用單獨搞一個啟動程序,可以就是真實程序,但真實程序需支持傳入參,根據入參的不同的,可以開啟多個進程,也可以達到該目的,我之前就實現過類似功能:程序自己更新自己】

2.桌面快捷方式本質只是一個軟連接(LINUX中也有),故如果真實程序需要更新,只需通過獨立的更新程序(程序更新實現原理有很多,在此就不展開說明)來更新真實的程序即可,而桌面的桌面快捷方式卻不用動,仍然通過:桌面快捷方式-》啟動程序-》最新的真實程序,用戶無感知的。

3.更新桌面圖標准備工作與步驟:

3.1.創建AppLauncher【引導啟動程序】,在程序內部直接實現:執行啟動DynamicIconApp.exe【原生真實程序】,啟動時帶上額外的參數(告之來自啟動程序及自己的進程ID,如:fromlauncher:12345),然后關閉自己即可。(其實就是跳板的作用),示例代碼如下:

/// <summary>
/// 引導啟動程序
/// author:zuowenjun
/// date:2021-6-19
/// </summary>
namespace AppLauncher
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void Form1_Load(object sender, EventArgs e)
        {
            ProcessStartInfo proc = new System.Diagnostics.ProcessStartInfo();
            proc.UseShellExecute = true;
            proc.FileName =Path.Combine(Application.StartupPath, "DynamicIconApp.exe");
            proc.Arguments = "fromlauncher:" + Process.GetCurrentProcess().Id;
            proc.CreateNoWindow = false;
            //啟動進程
            Process.Start(proc);
            this.Close();
        }
    }
}

 

3.2.創建DynamicIconApp.exe【原生真實程序】,在程序內部實現:在程序啟動界面前,通過參數判斷是否來自啟動引導程序,並判斷AppLauncher【引導啟動程序】進程是否已結束,若未結束,則先嘗試直接KILL,若KILL失敗則老實等待進程退出。若進程已結束,則再判斷是否需要更新桌面快捷方式(這個看具體的情況,可以在DB表中或遠程配置中心或API中增加可獲取是否需要更新桌面快捷方式文件的邏輯),若需要更新,則將當前應用程序目錄的指定的桌面快捷方式文件(如:DynamicIconApp.Lnk,如果不在,應該從CDN獲取最新的桌面快捷方式文件)替換桌面上已有或不存的桌面快捷方式文件,替換OK后,再正常運行顯示程序界面即可,這樣就能實現桌面APP的ICON按需動態更換的效果。示例代碼如下:

  1 /// <summary>
  2 /// 原生真實程序
  3 /// author:zuowenjun
  4 /// date:2021-6-19
  5 /// </summary>
  6 namespace DynamicIconApp
  7 {
  8     static class Program
  9     {
 10         /// <summary>
 11         ///  The main entry point for the application.
 12         /// </summary>
 13         [STAThread]
 14         static void Main(String[] args)
 15         {
 16             if (args != null && args.Length > 0)
 17             {
 18                 bool fromlauncher = args[0].StartsWith("fromlauncher:");
 19                 if (fromlauncher)
 20                 {
 21                     int launcherProcId = int.Parse(args[0].Substring(args[0].IndexOf(":") + 1));
 22                     //等待AppLauncher程序完全退出后,再正式運行
 23                     //MessageBox.Show("will starting..." + launcherProcId);
 24                     Process proc = null;
 25                     try
 26                     {
 27                         proc = Process.GetProcessById(launcherProcId);
 28                         MessageBox.Show(proc.Id + "," + proc.ProcessName + "," + proc.HasExited
 29                             + "," + proc.ExitTime);
 30                     }
 31                     catch (Exception e)
 32                     {
 33                         //MessageBox.Show("Process.GetProcessById error:" + e.ToString());
 34                         if (!e.Message.Contains("has exited"))
 35                         {
 36                             return;
 37                         }
 38                         proc = null;
 39                     }
 40 
 41 
 42                     bool waitExit = false;
 43                     if (null != proc)
 44                     {
 45                         try
 46                         {
 47                             Thread.Sleep(500);
 48                             proc.Kill();
 49                             waitExit = true;
 50                         }
 51                         catch (Exception e)
 52                         {
 53                             MessageBox.Show("kill Process error:" + e.ToString());
 54                             proc.WaitForExit();
 55                             waitExit = true;
 56                         }
 57                     }
 58 
 59                     //MessageBox.Show("start run after  launcher Process exit (waitExit = " + waitExit + ") !");
 60                 }
 61             }
 62             Application.SetHighDpiMode(HighDpiMode.SystemAware);
 63             Application.EnableVisualStyles();
 64             Application.SetCompatibleTextRenderingDefault(false);
 65             Application.Run(new Form1());
 66         }
 67     }
 68 }
 69 
 70 
 71 
 72 
 73 /// <summary>
 74 /// 原生真實程序
 75 /// author:zuowenjun
 76 /// date:2021-6-19
 77 /// </summary>
 78 namespace DynamicIconApp
 79 {
 80     public partial class Form1 : Form
 81     {
 82         public Form1()
 83         {
 84             InitializeComponent();
 85         }
 86 
 87         private void Form1_Load(object sender, EventArgs e)
 88         {
 89             //TODO:這里只是示例,判斷是否需要更新桌面快捷方式文件(換圖標)取決於遠程動態配置
 90             bool needUpdateAppLink = true;
 91             //TODO:這里只是判斷應用程序根目錄有沒有快捷方式文件,而實際的可能還要增加:
 92             //若本地沒有,則去CDN下載到本地
 93             if (needUpdateAppLink && File.Exists("DynamicIconApp.lnk"))
 94             {
 95                 MessageBox.Show("will be copy DynamicIconApp.link to desktop dir!");
 96                 String desktopFilePath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.Desktop), "DynamicIconApp.lnk");
 97                 if (File.Exists(desktopFilePath))
 98                 {
 99                     File.SetAttributes(desktopFilePath, FileAttributes.Normal);
100                     File.Delete(desktopFilePath);
101                 }
102                 string linkFilePath = Path.Combine(Application.StartupPath, "DynamicIconApp.lnk");
103                 File.Copy(linkFilePath, desktopFilePath);
104             }
105         }
106     }
107 }

3.3.提前創建快捷方式文件,把圖標及鏈接目標都設置好(當然也可以使用WshShell 組件通過C#來動態創建,這個看需要,我個人覺得沒必要),放到CDN或像我示例的放到真實程序根目錄即可,注意:DynamicIconApp.lnk 快捷方式的名字雖然叫原生真實程序名,但實際鏈接執行的是:引導啟動程序,目的就是桌面的快捷方式必需是真實程序名,這樣對於普通用戶來說才是對的。

聯想一下,大家有沒有發現,原來QQ也玩的是這一套,不信你看桌面的快捷方式及實際目標,截圖為證:

桌面快捷方式:

 

QQ快捷方式的屬性(目標鏈接的是:QQScLauncher.exe,這個就是QQ的引導程序,而本身的程序是QQ.exe)

 

 

 

QQ真實應用:(它們的關系是:QQ快捷方式-》執行QQ引導程序-》QQ程序,與我們的設計是如出一轍呀!)

 

 QQ這樣做,除了我說的那個目的(可以動態改快捷圖標),也可以在啟動QQ前做各種前置驗證,比如:是否需要升級等。

 好了,回到我們的今天的主題上來,上面已講了實現方案及具體步驟,現在是見證效果的時刻了。

這是原始安裝時的桌面快捷方式:(可以看到目標是指向的引導啟動程序)

 

 

然后我在應用程序根目錄把快捷方式更新(更換圖標),如下圖示:【當然如果是真實的生產環境,應該是將快捷方式文件放到CDN,同時通過遠程配置中心或API來返回是否需要更新快捷方式文件的邏輯】

 

改后效果:

 

好了,然后我們仍然模擬用戶,是在桌面雙擊原快捷方式圖標,最后運行后,桌面的快捷方式圖標也自動更新了。如下圖示:(原生真實程序運行起來了,桌面的ICON也同步更新了,當然想改名也是OK的,甚至改快捷鏈接目標也是可以的)

 

 

 文末說一下,這篇文章只是空閑時的小研究而矣,至於技術過不過時還是看需求吧,我最近工作重點是JAVA棧的SPRING微服務體系各種研究與實戰,比如:最近我實現了基於自定義的Mybatis攔截器來實現SQL語句自動審計功能(即:自動發現SQL語句是否合規,是否存在性能問題,若審計不通過,則會報錯,這樣在開發階段就能提前發現問題,及時止損),同時也研究了關於OAUTH2.0+OIDC相關內容,后面有機會再分享(為何今天不分享,因為家里的這個筆記本電腦太差,運行VS2019都比較卡,運行IEDA估計直接死機),近期工作真的很忙,上班沒時間,下班又加班太晚沒有精力,生活不易,但學習也不能止。

 


免責聲明!

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



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