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