1. 問題 假設我在Windows10的環境新建一個4.6的WPF項目,添加一個ComboBox,並用Blend在這個ComboBox上右鍵“編輯模板” “編輯副本”,Blend不僅幫我創建了模板,還會自動引用PresentationFramework.Aero2這個DLL,即使用Aero2這個主題 ...
1. 問題
假設我在Windows10的環境新建一個4.6的WPF項目,添加一個ComboBox,並用Blend在這個ComboBox上右鍵“編輯模板”->“編輯副本”,Blend不僅幫我創建了模板,還會自動引用PresentationFramework.Aero2
這個DLL,即使用Aero2這個主題的資源文件。一切看起來很簡單,直接,純真,善良,但將這個項目放到Windows7環境下運行就會報這樣的錯誤:
“System.IO.FileNotFoundException: 未能載入文件或程式集“PresentationFramework.Aero2, PublicKeyToken=31bf3856ad364e35”或它的某一個依賴項。系統找不到指定的文件。”
既沒做什麼喪盡天良的事,也沒做什麼泯滅人性的操作,然而程式出錯了。
2. 原因
先來說說什麼是Aero2。
WPF提供了以下幾種主題:
主題文件 | 桌面主題 |
---|---|
Classic.xaml | Windows XP 操作系統上的經典 Windows 外觀(Windows 95、Windows 98 和 Windows 2000)。 |
Luna.NormalColor.xaml | Windows XP 上的預設藍色主題。 |
Luna.Homestead.xaml | Windows XP 上的橄欖色主題。 |
Luna.Metallic.xaml | Windows XP 上的銀色主題。 |
Royale.NormalColor.xaml | Windows XP Media Center Edition 操作系統上的預設主題。 |
Aero.NormalColor.xaml | Windows Vista 操作系統上的預設主題。 |
Windows 8 之後WPF更新了Aero2和AeroLite兩種主題,關於Aero、Aero2、AeroLite的區別具體可見這個網頁。再之後微軟就沒有更新WPF主題了。
Aero
Aero2
WPF程式啟動時大概就是用這段代碼確定主題,也就是說預設是Aero,如果在Windows 8 或以上自動轉為Aero2:
_themeName = themeName.ToString();
_themeName = Path.GetFileNameWithoutExtension(_themeName);
if(String.Compare(_themeName, "aero", StringComparison.OrdinalIgnoreCase) == 0 && Utilities.IsOSWindows8OrNewer)
{
_themeName = "Aero2";
}
所以在Windows 10上使用Blend獲取控制項模板的副本時Blend識別出當前使用Aero2的主題並主動引用了Aero2相關的資源。
那麼為什麼在WIndows 7 中使用Aero2會出錯呢?用Bing搜一搜答案就出來了:
Problem with assembly PresentationFramework.Aero2
The assembly PresentationFramework.Aero2 in your project is metadata only assembly, which is used in dev time. You can get the full assembly under:
C:\Windows\Microsoft.NET\Framework\v4.0.30319\WPF
Please try to replace with the correct assembly. It should work.
簡單來說就是在C:\Program Files (x86)\Reference Assemblies\Microsoft\Framework.NETFramework\v4.6\PresentationFramework.Aero2.dll
這個位置的是個假貨(大小為161K)。真貨在C:\Windows\Microsoft.NET\Framework\v4.0.30319\WPF
這個目錄下(大小為248K),但WIN7下同個目錄找不到Aero2這個DLL。
3. 解決方案
知道問題原因後,要解決這個問題就很簡單了,隨隨便便都能想到3個:
在Windows10電腦上找到
C:\Windows\Microsoft.NET\Framework\v4.0.30319\WPF\PresentationFramework.Aero2.dll
這個DLL,放到項目中,併在項目中引用這個DLL。在Nuget上搜搜
Aero2
,找個上去像那麼回事的,例如這個。用ILSpy反編譯Aero2.dll,把需要的樣式複製粘貼到自己的項目中。
簡單測試了看上去都沒問題,不過,其實,可是我都沒有用這三個方案。
4. 實際上根本不需要Aero2?
回到最開始的問題,ComboBox的樣式用到Aero2的地方只有Themes:SystemDropShadowChrome
這個部分,這用於給彈出菜單提供陰影。而這個類在Aero(不是2)中也有提供,在我記憶里兩個DLL中這個類的實現完全一致,將Aero2的引用替換成Aero就可以解決這個問題了。甚至反編譯後獲取SystemDropShadowChrome
的源碼自己創建一個也可以。
5. 結語
程式員的開發環境總是用最新的,但客戶環境不受控制,最近還聽到人抱怨要相容XP的電腦。我以前面對的客戶群體都比較單一所以沒有太多相容性方面的經驗,所以這次才踩了這麼明顯的坑,不知道有沒有這方面的完整的指南?
6. 參考
Problem with assembly PresentationFramework.Aero2
Getting Started PresentationTheme Aero
我的博客即將同步至騰訊雲+社區,邀請大家一同入駐:https://cloud.tencent.com/developer/support-plan?invite_code=2fe5qd7iobmso