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

夢在旅途發表於2021-06-19

  今天休息在家,由於天氣熱再加上疫情原因,就在家裡呆著,空閒時想著,在很早以前(約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估計直接當機),近期工作真的很忙,上班沒時間,下班又加班太晚沒有精力,生活不易,但學習也不能止。

 

相關文章