【題外話】
最近在做一個調用某實驗儀器的程序,這個儀器提供了Windows上COM的接口。調用儀器的時候需要傳輸圖片,提供的接口里使用了IPicture這個接口,由於以前沒接觸過,所以查找了一些資料,整理了一下與.NET中System.Drawing.Image的互轉的方式。
【文章索引】
- IPicture和IPictureDisp是什么
- 使用AxHost實現與System.Drawing.Image的互轉
- 使用VB6 compatibility library實現互轉
- Alpha通道的問題
根據MSDN上對IPicture和IPictureDisp的說明來看,IPicture與IPictureDisp提供了與語言無關的接口,這個接口用來提供對位圖(Bitmap)、圖標(Icon)、圖元文件(Metafile)的抽象,其中后者還實現了IDispatch接口以實現COM的自動化接口。總之,如果通過COM接口傳輸圖像的話,可能會接觸到這兩個接口。
【二、使用AxHost實現與System.Drawing.Image的互轉】
.NET在System.Windows.Forms下提供了一個叫AxHost的類來實現與ActiveX控件進行訪問,不過這里用到的只是在AxHost里的protected的靜態方法而已。由於是protected的方法,所以沒有辦法直接調用,好在AxHost不是密封的類,所以我們還可以通過集成AxHost來實現調用,例如以下的代碼:
1 using System.Drawing; 2 using System.Windows.Forms; 3 4 public sealed class IPictureConverter : AxHost 5 { 6 private IPictureConverter() : base("") { } 7 8 #region IPictureDisp 9 public static stdole.IPictureDisp ImageToIPictureDisp(Image image) 10 { 11 return (stdole.IPictureDisp)GetIPictureDispFromPicture(image); 12 } 13 14 public static Image IPictureDispToImage(stdole.IPictureDisp pictureDisp) 15 { 16 return GetPictureFromIPictureDisp(pictureDisp); 17 } 18 #endregion 19 20 #region IPicture 21 public static stdole.IPicture ImageToIPicture(Image image) 22 { 23 return (stdole.IPicture)GetIPictureFromPicture(image); 24 } 25 26 public static Image IPictureToImage(stdole.IPicture picture) 27 { 28 return GetPictureFromIPicture(picture); 29 } 30 #endregion 31 }
【三、使用VB6 compatibility library實現互轉】
除了AxHost,其實微軟也提供了另外一個庫提供托管的Image與IPicture等互轉,那就是Microsoft.VisualBasic.Compatibility.dll,其中有一個叫Support的類提供了很多向后兼容的方法。對於IPicture或IPictureDisp的轉換,我們可以寫如下代碼:
1 using System.Drawing; 2 using Microsoft.VisualBasic.Compatibility.VB6; 3 4 public static class IPictureConverter 5 { 6 #region IPictureDisp 7 public static stdole.IPictureDisp ImageToIPictureDisp(Image image) 8 { 9 return (stdole.IPictureDisp)Support.ImageToIPictureDisp(image); 10 } 11 12 public static Image IPictureDispToImage(stdole.IPictureDisp pictureDisp) 13 { 14 return Support.IPictureDispToImage(pictureDisp); 15 } 16 #endregion 17 18 #region IPicture 19 public static stdole.IPicture ImageToIPicture(Image image) 20 { 21 return (stdole.IPicture)Support.ImageToIPicture(image); 22 } 23 24 public static Image IPictureToImage(stdole.IPicture picture) 25 { 26 return Support.IPictureToImage(picture); 27 } 28 #endregion 29 }
仔細看其實與上一段代碼非常類似,本着好奇的態度,我們Relector一下這些方法實現的代碼。

看起來兩者幾乎是一樣的,不過有意思的是,雖然兩者的很多方法如GetPICTDESCFromPicture等都不是同一個方法,甚至IPicture等接口都不是在一個庫里定義的(AxHost是在System.Windows.Forms.UnsafeNativeMethods中定義的,而VB6 compatibility library則是在單獨的一個stdole.dll中定義的),但是其調用的方法里執行的內容基本都相同,IPicture等接口也都是ComImport的同一個Guid,而兩個方法實現的源頭,更都是DllImport的oleaut32.dll,調用其中的“OleCreatePictureIndirect”方法,所以上述兩種方法是完全一樣的。
不過在4.0的CLR下,提供的Microsoft.VisualBasic.Compatibility.dll的版本為10.0.0.0,Support類以及相應的方法都被標記為過時的(Obsolete),所以編譯的時候提示的警告也蠻讓人惡心的(2.0的CLR下提供的8.0.0.0的dll沒有這個問題),所以倒不妨采用第一種方法。
如果你的圖片包含Alpha通道的話,上述轉換可能會導致顏色有些問題,由於IPicture沒有辦法支持Alpha通道,所以妥協的辦法只能是要么不用Alpha通道,要么在轉換為IPicture前在圖片底下墊上一個純色的背景(比如對方程序中要顯示圖片的位置的背景),比如How to Convert a System.Drawing.Image to an IPictureDisp with Alpha Transparency這篇文章就是這么做的。
【相關鏈接】
- IPicture interface:http://msdn.microsoft.com/en-us/library/ms680761.aspx
- IPictureDisp interface:http://msdn.microsoft.com/en-us/library/ms680762.aspx
- Converting between IPictureDisp and System.Drawing.Image:http://blogs.msdn.com/b/andreww/archive/2007/07/30/converting-between-ipicturedisp-and-system-drawing-image.aspx
