C#开发PACS医学影像处理系统(十七):2D处理之影像旋转和翻转
1.任意角度旋转
在XAML设计器中,设置RotateTransform属性
<InkCanvas x:Name="ToolInkCanvas" UseCustomCursor="True" ClipToBounds="False" MinHeight="50" MinWidth="50" EditingMode="None" Background="{x:Null}" Margin="1" >
<InkCanvas.RenderTransform>
<TransformGroup>
<RotateTransform/>
<ScaleTransform/>
</TransformGroup>
</InkCanvas.RenderTransform>
</InkCanvas>
根据鼠标按下时的位置和图片中心位置,利用WPF中Vector类计算向量夹角,得到角度,再设置元素的RotateTransform旋转属性变换。
/// <summary>
/// 图像旋转
/// </summary>
/// <param name="mouseVal">差值</param>
/// <param name="type">鼠标类型</param>
/// <param name="X">平移X轴距离</param>
/// <param name="startX">X轴起始位置</param>
/// <param name="Y">平移Y轴距离</param>
/// <param name="startY">Y轴起始位置</param>
/// <param name="isAction">是主动发起还是关联操作时被动调用</param>
private void RotateImage(double mouseVal, string type, double X, double startX, double Y, double startY, bool isAction = true)
{
double angle = ;
if (type != "wheel")
{
Vector curVector = new Vector(X - ToolInkCanvas.ActualWidth / , Y - ToolInkCanvas.ActualHeight / );
double ang = Vector.AngleBetween(downVector, curVector); angle = SetRotateTrans(ToolInkCanvas, ang, true);
ImgAngle = angle;
downVector = curVector;
}
else
{
if (mouseVal > )
{
angle = SetRotateTrans(ToolInkCanvas, , true);
}
else
{
angle = SetRotateTrans(ToolInkCanvas, -, true);
}
ImgAngle = angle;
} if (isAction && PubVal.IsStack)
{
for (int i = ; i < Cell.MedicalView.SelectedBoxList.Count; i++)
{
var box = Cell.MedicalView.SelectedBoxList[i];
if (box != this)
{
box.RotateImage(mouseVal, type, X, startX, Y, startY, false);
}
}
} //旋转所有box
for (int i = ; i < Cell.BoxList.Count; i++)
{
if (Cell.BoxList[i] != this && (Cell.BoxList[i].ImageAlign == ImageAlignment.Center || Cell.BoxList[i].ImageAlign == ImageAlignment.Parent))
{
Cell.BoxList[i].RotateImage(angle);
}
} Main.ToolBar.CheckAllText(this);
}
看效果:

2.镜像翻转
在上一篇(影像的平移和缩放)文中提到的ScaleTransform属性中,
同样还是先获取变换对象
ScaleTransform st = tgnew.Children[index] as ScaleTransform;
当 st.ScaleX的值为-1时则水平翻转,
当 st.ScaleY的值为-1时则垂直翻转
/// <summary>
/// 设置元素比例
/// </summary>
/// <param name="ui">元素</param>
/// <param name="scaleX">x比例</param>
/// <param name="scaleY">y比例</param>
/// <param name="isAppend">是否为追加比例</param>
/// <param name="index">变换属性的下标</param>
public ScaleTransform SetScaleTrans(UIElement ui, double scaleX, double scaleY, bool isAppend, int index = )
{
TransformGroup tg = ui.RenderTransform as TransformGroup;
var tgnew = tg.CloneCurrentValue();
if (tgnew != null)
{
ui.RenderTransformOrigin = new Point(0.5, 0.5);
ScaleTransform st = tgnew.Children[index] as ScaleTransform;
if (isAppend)
{
st.ScaleX *= scaleX;
st.ScaleY *= scaleY;
}
else
{
st.ScaleX = scaleX;
st.ScaleY = scaleY;
}
ui.RenderTransform = tgnew;
return st;
}
return null;
}
调用翻转:
/// <summary>
/// 图像翻转
/// </summary>
/// <param name="type">翻转类型</param>
public void FlipImage(FlipType type)
{
if (type == FlipType.Horizontal)
{
var st = SetScaleTrans(ToolInkCanvas, -, , true, );
}
else
{
var st = SetScaleTrans(ToolInkCanvas, , -, true, );
}
UpDateTag();
}
看效果:

3.复合操作与标记文字角度自适应
在2D操作过程中,除了会同时做平移、缩放和旋转之外,图像上通常有测量标记,当影像发生2D变化时,需要将测量文字自适应角度回正,与旋转方向相反。
/// <summary>
/// 重新设置文本大小和旋转角度
/// <param name="txt">文本控件</param>
/// </summary>
public void ReSetMeasureSize(TextBlock txt)
{
if (cvsWidth < cvsHeight)
{
txt.FontSize = (cvsWidth / w_num);
}
else
{
txt.FontSize = (cvsHeight / h_num);
} if (txt.RenderTransform.Dispatcher == null)
{
TransformGroup tf = new TransformGroup();
tf.Children.Add(new RotateTransform());
tf.Children.Add(new ScaleTransform());
txt.RenderTransform = tf;
} TransformGroup cvsTf = inkCanvas.RenderTransform as TransformGroup;
var cvsTg = cvsTf.CloneCurrentValue();
RotateTransform rtf = cvsTg.Children[] as RotateTransform; //设置翻转
ScaleTransform stf = cvsTg.Children[] as ScaleTransform;
TransformGroup tfg = txt.RenderTransform as TransformGroup;
var tfgnew = tfg.CloneCurrentValue();
if (tfgnew != null)
{
ScaleTransform st = tfgnew.Children[] as ScaleTransform; if (stf.ScaleX * st.ScaleX < )
{
st.ScaleX *= -;
}
if (stf.ScaleY * st.ScaleY < )
{
st.ScaleY *= -;
}
txt.RenderTransform = tfgnew;
} //设置旋转
TransformGroup tg = txt.RenderTransform as TransformGroup;
var tgnew = tg.CloneCurrentValue();
if (tgnew != null)
{
txt.RenderTransformOrigin = new Point(, );
RotateTransform rt = tgnew.Children[] as RotateTransform;
if (rtf.Angle != rt.Angle)
{
rt.Angle = - * rtf.Angle;
}
ScaleTransform st = tfgnew.Children[] as ScaleTransform;
if (((st.ScaleX < && st.ScaleY > ) || (st.ScaleY < && st.ScaleX > )) && ((rt.Angle > && rtf.Angle <= ) || (rt.Angle < && rtf.Angle >= )))
{
rt.Angle *= -;
}
else if (st.ScaleY > && st.ScaleX > && ((rt.Angle < && rtf.Angle < )||(rt.Angle >= && rtf.Angle >= )))
{
rt.Angle *= -;
}
else if (st.ScaleY < && st.ScaleX < && ((rt.Angle < && rtf.Angle < ) || (rt.Angle >= && rtf.Angle >= )))
{
rt.Angle *= -;
}
txt.RenderTransform = tgnew;
}
}
看效果:

C#开发PACS、RIS医学影像处理系统
目录整理:
(一)PACS客户端:
C#开发PACS医学影像处理系统(七):读取影像Dicom信息
C#开发PACS医学影像处理系统(十):Dicom影像下载策略与算法
C#开发PACS医学影像处理系统(十一):Dicom影像挂片协议
C#开发PACS医学影像处理系统(十二):绘图处理之图形标记
C#开发PACS医学影像处理系统(十三):绘图处理之病灶测量
C#开发PACS医学影像处理系统(十四):处理Dicom影像窗宽窗位
C#开发PACS医学影像处理系统(十五):Dicom影像交叉定位线算法
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影像处理系统宣传视频
![]() |
QQ:1850969244 近10年开发经验,主攻C#、ASP MVC,HTML5, B/S C/S 皆可,目前研究医疗领域医学影像相关技术, 任何技术问题欢迎加QQ交流。 |
C#开发PACS医学影像处理系统(十七):2D处理之影像旋转和翻转的更多相关文章
- C#开发PACS医学影像处理系统(十五):Dicom影像交叉定位线算法
1.定位线概念:某个方位的影像在另一个方向的影像上的投影相交线,例如横断面(从头到脚的方向)在矢状面(从左手到右手)上的影像投影面交线. 举个例子:右边的是MR(核磁共振)的某一帧切片,这是从头开始扫 ...
- C#开发PACS医学影像处理系统(十九):Dicom影像放大镜
在XAML代码设计器中,添加canvas画布与圆形几何对象,利用VisualBrush笔刷来复制画面内容到指定容器: <Canvas x:Name="CvsGlass" Wi ...
- C#开发PACS医学影像处理系统(十六):2D处理之影像平移和缩放
1.平移,利用WPF中控件边距来控制位移: /// <summary> /// 平移图像 /// </summary> /// <param name="X&q ...
- C#开发PACS医学影像处理系统(二):界面布局之菜单栏
在菜单栏布局上,为了使用自定义窗体样式和按钮,我们需要先将窗体设置为无边框,然后添加一个Grid作为菜单栏并置顶,VerticalAlignment="Top" logo图片和标题 ...
- C#开发PACS医学影像处理系统(三):界面布局之工具栏
工具栏布局采用WPF中Grid作为容器,按钮采用自定义样式和图标,并采用Separator分割线: XAML设计器代码: 其中 Style="{StaticResource ButtonS ...
- C#开发PACS医学影像处理系统(六):加载Dicom影像
对于一款软件的扩展性和维护性来说,上层业务逻辑和UI表现一定要自己开发才有控制权,否则项目上线之后容易被掣肘, 而底层图像处理,我们不需要重复造轮子,这里推荐使用fo-dicom,同样基于Dicom3 ...
- C#开发PACS医学影像处理系统(十二):绘图处理之图形标记
在医生实际使用过程中,对于有病灶的影像需要一些2D绘图操作,例如对于病灶的标记和测量, 这就牵涉到在WPF中的2D绘图操作技术,一般的思路是监听鼠标的按下和抬起以及运动轨迹,目前整理出的常用绘图和测量 ...
- C#开发PACS医学影像处理系统(十三):绘图处理之病灶测量
接上一篇文章,当我们可以绘制图形标记后,就可以在此操作类上面进行扩展, 比如测量类工具,目前整理出的常用绘图和测量功能如下: 测量工具类:(图形标记类请参考本系列文章:绘图处理之图形标记) 功能 说明 ...
- C#开发PACS医学影像处理系统(十四):处理Dicom影像窗宽窗位
概念解释(网络资料): 窗宽: 窗宽指CT图像所显示的CT 值范围.在此CT值范围内的组织结构按其密度高低从白到黑分为16 个灰阶以供观察对比.例如,窗宽选定为100 Hu ,则人眼可分辨的CT值为1 ...
随机推荐
- Gitlab安装使用
Gitlab安装使用 1. 为什么要使用gitlab Git的优点多多这里就不详细介绍了: Git是版本控制系统,Github是在线的基于Git的代码托管服务: Github有个小缺陷 (也不能算是缺 ...
- latex三种标准文类book, report, article的章节命令与层次深度
Latex有三种标准文类:book, report, article. 每种文类的章节命令和层次深度如下: 三种标准文类的章节命令与层次深度 层次深度 层次名 book report article ...
- Arduboy基本用法(一)
Arduboy基本用法(一) 一.第一个Arduboy程序 在arduino中新建项目 示例代码 #include<Arduboy.h> Arduboy arduboy; void set ...
- 更好地使用google
精确搜索:双引号 精确搜索就是在你要搜索的词上,加上双引号,这时google就会完全的匹配你所要搜索的字符串 "今日黄瓜" 站内搜索:site 例如我想在stackoverflow ...
- Java面试题(Java基础篇)
Java 基础 1.JDK 和 JRE 有什么区别? JDK:Java Development Kit 的简称,java 开发工具包,提供了 java 的开发环境和运行环境. JRE:Java Run ...
- 为什么网站URL需要设置为静态化
http://www.wocaoseo.com/thread-95-1-1.html 为什么网站URL需要静态化?网站url静态化的好处是什么?现在很多网站的链接都是静态规的链接,但是网站 ...
- go语言之接口
一:接口的基本概念 1 接口声明 接口字面量,接口命名类型,接口声明使用interface关键字. 1)接口字面量类型声明语法如下: interface{ methodSignature1 metho ...
- 2020.08.31 Unit 10(暂未完成)
[重点短语] 01.at night 在晚上 02.in a more natural environment 在一个更加自然的环境中 03.all year round 一年到头,终年 04.be ...
- 关于js中循环遍历中顺序执行ajax的问题(vue)
js里的循环,每次都是自顾自的走,它不等ajax执行好走完到success代码,就继续循环下一条数据了,这样数据就全乱了. 后来,想到试试ajax里async这个属性,async默认是true,即为异 ...
- Java实现的二叉堆以及堆排序详解
一.前言 二叉堆是一个特殊的堆,其本质是一棵完全二叉树,可用数组来存储数据,如果根节点在数组的下标位置为1,那么当前节点n的左子节点为2n,有子节点在数组中的下标位置为2n+1.二叉堆类型分为最大堆( ...
