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