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医学影像处理系统(一):开发背景和功能预览

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医学影像处理系统(十五):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处理之影像旋转和翻转的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

    对于一款软件的扩展性和维护性来说,上层业务逻辑和UI表现一定要自己开发才有控制权,否则项目上线之后容易被掣肘, 而底层图像处理,我们不需要重复造轮子,这里推荐使用fo-dicom,同样基于Dicom3 ...

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

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

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

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

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

    概念解释(网络资料): 窗宽: 窗宽指CT图像所显示的CT 值范围.在此CT值范围内的组织结构按其密度高低从白到黑分为16 个灰阶以供观察对比.例如,窗宽选定为100 Hu ,则人眼可分辨的CT值为1 ...

随机推荐

  1. 【算法•日更•第三十一期】KMP算法

    ▎前言 这次要讲的HMP算法KMP算法很简单,是用于处理字符串的,之前一直以为很难,其实也不过如此(说白了就是优化一下暴力). ▎处理的问题 通常处理的问题是这样的:给定两个字符串s1和s2,其中s1 ...

  2. 利用 Github Actions 自动更新 docfx 文档

    利用 Github Actions 自动更新 docfx 文档 Intro docfx 是微软出品一个 .NET API 文档框架,有一个理念是代码即文档,会根据项目代码自动生成 API 文档,即使没 ...

  3. 非确定性有穷状态决策自动机练习题Vol.1 A.扭动的回文串

    非确定性有穷状态决策自动机练习题Vol.1 A.扭动的回文串 题目描述 \(JYY\)有两个长度均为\(N\)的字符串\(A\)和\(B\). 一个"扭动字符串\(S(i,j,k)\)由\( ...

  4. 使用CrashHandler获取应用crash信息

      Android应用不可避免会发生crash,也称之为崩溃.发生原因可能是由于Android系统底层的bug,也可能是由于不充分的机型适配或者是糟糕的网络情况.当crash发生时,系统会kill掉正 ...

  5. 如何理解算法时间复杂度的表示法O(n²)、O(n)、O(1)、O(nlogn)等?

    先从 来说,理论上哈希表就是O(1).因为哈希表是通过哈希函数来映射的,所以拿到一个关键字,用哈希函数转换一下,就可以直接从表中取出对应的值.和现存数据有多少毫无关系,故而每次执行该操作只需要恒定的时 ...

  6. linux驱动之jiffies的理解

    本文主要借鉴了http://blog.csdn.net/allen6268198/article/details/7270194,非常感谢该网友的分享. 在LINUX的时钟中断中涉及至二个全局变量一个 ...

  7. go module 获取码云私有仓库代码

    因为码云免费组织有5人限制,其他人想获得代码 只能通过别的方式 go mod 底层使用的git 获取代码, 所以首先解决如何通过git clone代码 思路为通过ssh密钥的方式获取 首先在码云仓库部 ...

  8. CentOS7上lvm分区调整(resize2fs: Bad magic number in super-block while trying to open ...)

    问题描述:根目录空间占用率100% 首先用传统resize2fs方式调整分区: resize2fs -p /dev/mapper/centos-home 50Gresize2fs 1.42.9 (28 ...

  9. Robot Framework(6)——案例分层

    上一篇以gogomall的登录为例,记录了一个简单的登录流程的脚本,在实际应用中,当流程发生一些小的变动,或者前端代码改变,导致元素选择器失效,我们维护脚本的成本就比较大,需要一个一个去修改,所以,有 ...

  10. 从一个小需求感受Redis的独特魅力

    分享一个简单的小需求应该怎么设计实现以及有关Redis的使用 Redis在实际应用中使用的非常广泛,本篇文章就从一个简单的需求说起,为你讲述一个需求是如何从头到尾开始做的,又是如何一步步完善的. 需求 ...