对于一款软件的扩展性和维护性来说,上层业务逻辑和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 = , double height = )
{
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 = , double height = )
{
try
{
if (sourceImg != null)
{
ImageBrush ib = new ImageBrush(sourceImg);
imgBox.Background = ib;
} double w = ;
double h = ;
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 / ,
h / ,
w / ,
h / );
}
else if (align == ImageAlignment.Left)
{
imgBox.Margin = new Thickness(
,
h / ,
w,
h / );
}
else if (align == ImageAlignment.Right)
{
imgBox.Margin = new Thickness(
w,
h / ,
,
h / );
}
else if (align == ImageAlignment.Top)
{
imgBox.Margin = new Thickness(
w / ,
,
w / ,
h);
}
else if (align == ImageAlignment.Bottom)
{
imgBox.Margin = new Thickness(
w / ,
h,
w / ,
);
}
}
catch (Exception e)
{
LogApi.WriteErrLog(e);
}
}
/// <summary>
/// 图像对齐方式
/// </summary>
public enum ImageAlignment
{
/// <summary>
/// 居左
/// </summary>
Left = , /// <summary>
/// 置顶
/// </summary>
Top = , /// <summary>
/// 居右
/// </summary>
Right = , /// <summary>
/// 置底
/// </summary>
Bottom = , /// <summary>
/// 居中
/// </summary>
Center = , /// <summary>
/// 跟随父级(仅用于Cell中dicom容器)
/// </summary>
Parent = , }

最终呈现效果:

注:一些特殊的类型对齐方式是不一样的,需要对应的挂片协议来控制,详情请参考本系列文章的挂片协议章节。

例如:钼靶(modality:MG)类型,是按照居左居右对齐。

C#开发PACS、RIS医学影像处理系统

目录整理:

(一)PACS客户端:

C#开发PACS医学影像处理系统(一):开发背景和功能预览

C#开发PACS医学影像处理系统(二):界面布局之菜单栏

C#开发PACS医学影像处理系统(三):界面布局之工具栏

C#开发PACS医学影像处理系统(四):界面布局之状态栏

C#开发PACS医学影像处理系统(五):查询病人信息列表

C#开发PACS医学影像处理系统(六):加载Dicom影像

C#开发PACS医学影像处理系统(七):读取影像Dicom信息

C#开发PACS医学影像处理系统(八):单元格变换

C#开发PACS医学影像处理系统(九):序列列表

C#开发PACS医学影像处理系统(十):Dicom影像下载策略与算法

C#开发PACS医学影像处理系统(十一):Dicom影像挂片协议

C#开发PACS医学影像处理系统(十二):绘图处理之图形标记

C#开发PACS医学影像处理系统(十三):绘图处理之测量工具

C#开发PACS医学影像处理系统(十四):处理Dicom影像窗宽窗位

C#开发PACS医学影像处理系统(十五):基于体位图交叉定位线算法

C#开发PACS医学影像处理系统(十六):2D处理之平移和缩放

C#开发PACS医学影像处理系统(十七):2D处理之任意角度旋转与镜像翻转

C#开发PACS医学影像处理系统(十八):Dicom影像色彩增强(伪彩)

C#开发PACS医学影像处理系统(十九):Dicom影像反色处理(负片)

C#开发PACS医学影像处理系统(二十):Dicom影像放大镜功能

(二)PACS三维:MRP、MIP、VR

C#开发PACS医学影像三维重建(一):使用VTK三维重建Dicom影像

(三)PACS网页端:开发Web版本的PACS

C#开发Web端PACS(一):基于PACS客户端思想重写Web端

(四)PACS移动端:开发基于HTML5移动端版本的PACS

C#开发移动端PACS(一):使用HTML5和CSS3开发PACS手机端页面

C#开发移动端PACS(二):使用 .Net MVC 开发手机端PACS服务端

(五)PACS服务端:

C#开发PACS医学影像处理系统服务端(一):医疗设备的连接与收图

C#开发PACS医学影像处理系统服务端(二):高并发架构

(六)PACS与RIS系统的通信与集成

在RIS系统中调起PACS并打开Dicom影像

(七)云PACS与远程会诊

C#开发PACS医学影像处理系统之云PACS(区域PACS)(一):架构概述

C#开发PACS医学影像处理系统之云PACS(区域PACS)(二):远程会诊与双向转诊

(八)科幻级视频特效:使用Adobe After Effects 制作PACS影像处理系统宣传视频

C#开发PACS医学影像处理系统(六):加载Dicom影像的更多相关文章

  1. C#开发PACS医学影像处理系统(十五):Dicom影像交叉定位线算法

    1.定位线概念:某个方位的影像在另一个方向的影像上的投影相交线,例如横断面(从头到脚的方向)在矢状面(从左手到右手)上的影像投影面交线. 举个例子:右边的是MR(核磁共振)的某一帧切片,这是从头开始扫 ...

  2. C#开发PACS医学影像处理系统(十九):Dicom影像放大镜

    在XAML代码设计器中,添加canvas画布与圆形几何对象,利用VisualBrush笔刷来复制画面内容到指定容器: <Canvas x:Name="CvsGlass" Wi ...

  3. C#开发PACS医学影像处理系统(十八):Dicom使用LUT色彩增强和反色

    在医生阅片确诊的过程中,当发线疑似病灶时在灰度显示下有时并不清晰,这时候就需要色彩增强效果来使灰度图像变为彩色图像. LUT可以简单的理解为0-255的颜色映射值,例如:彩虹编码,将其打包成LUT格式 ...

  4. C#开发PACS医学影像处理系统(十六):2D处理之影像平移和缩放

    1.平移,利用WPF中控件边距来控制位移: /// <summary> /// 平移图像 /// </summary> /// <param name="X&q ...

  5. C#开发PACS医学影像处理系统(二):界面布局之菜单栏

    在菜单栏布局上,为了使用自定义窗体样式和按钮,我们需要先将窗体设置为无边框,然后添加一个Grid作为菜单栏并置顶,VerticalAlignment="Top" logo图片和标题 ...

  6. C#开发PACS医学影像处理系统(三):界面布局之工具栏

    工具栏布局采用WPF中Grid作为容器,按钮采用自定义样式和图标,并采用Separator分割线: XAML设计器代码: 其中  Style="{StaticResource ButtonS ...

  7. C#开发PACS医学影像处理系统(十二):绘图处理之图形标记

    在医生实际使用过程中,对于有病灶的影像需要一些2D绘图操作,例如对于病灶的标记和测量, 这就牵涉到在WPF中的2D绘图操作技术,一般的思路是监听鼠标的按下和抬起以及运动轨迹,目前整理出的常用绘图和测量 ...

  8. C#开发PACS医学影像处理系统(十三):绘图处理之病灶测量

    接上一篇文章,当我们可以绘制图形标记后,就可以在此操作类上面进行扩展, 比如测量类工具,目前整理出的常用绘图和测量功能如下: 测量工具类:(图形标记类请参考本系列文章:绘图处理之图形标记) 功能 说明 ...

  9. C#开发PACS医学影像处理系统(十七):2D处理之影像旋转和翻转

    1.任意角度旋转 在XAML设计器中,设置RotateTransform属性 <InkCanvas x:Name="ToolInkCanvas" UseCustomCurso ...

随机推荐

  1. Window C盘 占满原因之一

    最近一段时间,突然C盘 莫名奇妙的满了 ,也没有中毒. 最后查找是因为安装了SQL Reporting  的原因 C:\Program Files\Microsoft SQL Server Repor ...

  2. 前端面试?这份手撸Promise请你收下

    前言 现在很多大厂面试前端都会要求能够手动的写出一个Promise,所以这里整理了一份手写的Promise. 绝对详细,功能绝对强大.如果你不了解Promise的基本使用,那么本篇文章可能不太适合你, ...

  3. sql server 查询表字段的说明备注信息

    SELECT 表名 = case when a.colorder= then d.name else '' end, 表说明 = case when a.colorder= then isnull(f ...

  4. unity探索者之protobuf的序列化和反序列化导致unity崩溃的问题研究

    版权声明:本文为原创文章,转载请声明http://www.cnblogs.com/unityExplorer/p/7574569.html 这两天博主在接微信支付SDK的时候碰到一个非常恶心又诡异的问 ...

  5. 《Java从入门到失业》第二章:Java环境(四):IDE集成环境

    2.4IDE集成环境 在掌握了编写.编译和运行Java程序的基本步骤以后,你肯定就在想,这太麻烦了,有没有更好的工具?当然有了,那就是IDE.IDE就是专业的集成开发环境(Integrated Dev ...

  6. 边缘计算、区块链、5G,哪个能走的更远

    频繁出现的新词汇5G.区块链.边缘计算,这些都代表了什么,又能给我们的生活带来什么巨大的改变么?抉择之时已至,能够走向未来的真的只有一个吗? "没有什么能够阻挡,你对自由的向往....&qu ...

  7. 使用hexo+github搭建博客

    https://blog.csdn.net/qq_36667170/article/details/105789610这一篇已经写得很详细了,下面的内容是我操作的时候遇到的问题及解决方法. 1.下载N ...

  8. Jmeter 常用函数(18)- 详解 __isDefined

    如果你想查看更多 Jmeter 常用函数可以在这篇文章找找哦 https://www.cnblogs.com/poloyy/p/13291704.htm 作用 判断 Jmeter 变量是否存在,1 就 ...

  9. Flink的应用场景和架构

    Flink的应用场景 Flink项目的理念就是:Flink是为分布式,高性能,随时可用以及准确的流处理应用程序打造的开源流处理框架.自2019年开源以来,迅速成为大数据实时计算领域炙手可热的技术框架. ...

  10. 容器生态.png

    https://www.cnblogs.com/CloudMan6/p/6706546.html