對於一款軟件的擴展性和維護性來說,上層業務邏輯和UI表現一定要自己開發才有控制權,否則項目上線之后容易被掣肘,
而底層圖像處理,我們不需要重復造輪子,這里推薦使用fo-dicom,同樣基於Dicom3.0協議。
根據以上原則,后台影像處理完成之后,即可使用自己開發的控件來呈現。
1.先准備好自己的dcm文件,可以是單個文件或序列文件或DicomDir文件。
2.在VS編輯器里打開NUGET搜索fo-dicoman安裝即可。

3.使用fo-dicom讀取影像。
//將圖像處理模式設置為全局WPF模式 Dicom.Imaging.ImageManager.SetImplementation(Dicom.Imaging.WPFImageManager.Instance); //實例化文件處理對象並打開文件 DicomFile dicomFile = DicomFile.Open(@"C:\101\1.dcm"); //獲取dicom圖像對象 DicomImage dicomImage = new DicomImage(dicomFile.Dataset); //不使用LUT dicomImage.UseVOILUT = false; //轉換成一般圖像格式,bmp,png等, //在WPF中我們轉換成WriteableBitmap dicomImage.RenderImage().AsWriteableBitmap() :
4.設置縮放比例,通常dcm影像的大小是不固定的,由設備和技師拍攝手法決定,所以我們要將圖像大小縮小或放大到最佳尺寸。
dicomImage對象中的Scale屬性決定了縮放比例,方法參數中Border是邊框容器,也可以使用Grid或其他可以呈現圖像的容器。
/// <summary> /// 計算圖片在容器中的縮放比例 /// </summary> /// <param name="dicomImage">圖片</param> /// <param name="imgBoxDock">容器</param> /// <param name="width">實際長度</param> /// <param name="height">實際高度</param> public static void SetImageScale(DicomImage dicomImage, Border imgBoxDock, double width = 0, double height = 0) { double maxWidth = imgBoxDock.Width; double maxHeight = imgBoxDock.Height; if (double.IsNaN(imgBoxDock.Width) || double.IsNaN(imgBoxDock.Height)) { maxWidth = width; maxHeight = height; } dicomImage.Scale = Math.Min(maxWidth / dicomImage.Width, maxHeight / dicomImage.Height); }
5.設置圖像對齊方式,圖像的長寬一般是不能恰好填充容器的,尤其是設置縮放比例之后,那我們需要在容器中將圖像按照一定規則對齊。
/// <summary> /// 在圖片容器中顯示一張圖片 /// </summary> /// <param name="sourceImg">圖片源</param> /// <param name="imgBox">圖片控件</param> /// <param name="imgBoxDock">圖片容器</param> /// <param name="align">對齊方式</param> /// <param name="width">實際長度</param> /// <param name="height">實際高度</param> public static void ShowImage(WriteableBitmap sourceImg, InkCanvas imgBox, Border imgBoxDock, ImageAlignment align, double width = 0, double height = 0) { try { if (sourceImg != null) { ImageBrush ib = new ImageBrush(sourceImg); imgBox.Background = ib; } double w = 0; double h = 0; if (double.IsNaN(imgBoxDock.Width) || double.IsNaN(imgBoxDock.Height)) { w = width; h = height; } else { if (sourceImg != null) { w = imgBoxDock.Width - sourceImg.Width; h = imgBoxDock.Height - sourceImg.Height; } else return; } if (align == ImageAlignment.Center) { imgBox.Margin = new Thickness( w / 2, h / 2, w / 2, h / 2); } else if (align == ImageAlignment.Left) { imgBox.Margin = new Thickness( 0, h / 2, w, h / 2); } else if (align == ImageAlignment.Right) { imgBox.Margin = new Thickness( w, h / 2, 0, h / 2); } else if (align == ImageAlignment.Top) { imgBox.Margin = new Thickness( w / 2, 0, w / 2, h); } else if (align == ImageAlignment.Bottom) { imgBox.Margin = new Thickness( w / 2, h, w / 2, 0); } } catch (Exception e) { LogApi.WriteErrLog(e); } }
/// <summary> /// 圖像對齊方式 /// </summary> public enum ImageAlignment { /// <summary> /// 居左 /// </summary> Left = 0, /// <summary> /// 置頂 /// </summary> Top = 1, /// <summary> /// 居右 /// </summary> Right = 2, /// <summary> /// 置底 /// </summary> Bottom = 3, /// <summary> /// 居中 /// </summary> Center = 4, /// <summary> /// 跟隨父級(僅用於Cell中dicom容器) /// </summary> Parent = 5, }
最終呈現效果:

注:一些特殊的類型對齊方式是不一樣的,需要對應的掛片協議來控制,詳情請參考本系列文章的掛片協議章節。
例如:鉬靶(modality:MG)類型,是按照居左居右對齊。

