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