绘制箭头

 

1,直接用平台库

Pen arrowPen = new Pen(Color.Blue);
           arrowPen.Width = 4;
           arrowPen.EndCap = System.Drawing.Drawing2D.LineCap.ArrowAnchor;

Graphics gfx = this.pCanvas.CreateGraphics();
           gfx.DrawLine(arrowPen,
               5, 5,
               5, 100);

结果:

2,设置箭头样式

System.Drawing.Drawing2D.AdjustableArrowCap lineCap =
                new System.Drawing.Drawing2D.AdjustableArrowCap(6, 6, true);
            Pen redArrowPen = new Pen(Color.Red, 4);
            redArrowPen.CustomEndCap = lineCap;

Graphics gfx = this.pCanvas.CreateGraphics();
            gfx.DrawLine(redArrowPen,
                25, 5,
                25, 100);

结果:

3,手动绘制

注意:代码中应用了,只用了其中的Vector2

//线的起点

PointF startPt = new PointF(100, 300);

//线的终点

PointF endPt = new PointF(200, 200);

//箭头的宽

float width = 10;

//箭头夹角

double angle = 60.0 / 180 * Math.PI;

 

//求BC长度

double widthBE = width / 2 / (Math.Tan(angle / 2));

 

//直线向量

Vector2 lineVector = new Vector2(endPt.X - startPt.X, endPt.Y - startPt.Y);

//单位向量

lineVector.Normalize();

 

//求BE向量

Vector2 beVector = (float)widthBE * -lineVector;

 

//求E点坐标

PointF ePt=new PointF();

//ePt - endPt = bcVector

ePt.X = endPt.X + beVector.X;

ePt.Y = endPt.Y + beVector.Y;

 

//因为CD向量和AB向量垂直,所以CD方向向量为

Vector2 cdVector = new Vector2(-lineVector.Y, lineVector.X);

//求单位向量

cdVector.Normalize();

 

//求CE向量

Vector2 ceVector = width / 2 * cdVector;

//求C点坐标,ePt - cPt = ceVector;

PointF cPt = new PointF();

cPt.X = ePt.X - ceVector.X;

cPt.Y = ePt.Y - ceVector.Y;

 

//求DE向量

Vector2 deVector = width / 2 * -cdVector;

//求D点,ePt-dPt = deVector;

PointF dPt = new PointF();

dPt.X = ePt.X - deVector.X;

dPt.Y = ePt.Y - deVector.Y;

 

//开始绘制

Graphics gfx = this.pCanvas.CreateGraphics();

 

//绘制线

gfx.DrawLine(Pens.Blue, startPt, endPt);

//绘制箭头

//gfx.DrawPolygon(Pens.Green,

//    new PointF[]{

//        cPt,dPt,endPt});

gfx.FillPolygon(Brushes.Green,

    new PointF[]{

        cPt,dPt,endPt});

 
结果:

手动绘制原理:

分为两部分绘制:

绘制直线较为简单,麻烦在绘制箭头(多边形)。

已知:aP(A点坐标),bP(B点坐标),θ角(angle表示),CD长度width.

求:点C坐标,点D坐标。

解:

1,求出向量AB    abVector=bP-aP;,并且单位向量化。此时abVector代表了方向

2,BE的长度求解 widthBE= width / 2 / (Math.Tan(angle / 2)); (三角形BEC为直角三角形)。

3,求解向量BE     beVector = (float)widthBE * -abVector; (由长度和AB单位向量决定)。

4,求解E点坐标   eP – bP = beVector ,所以 eP = bP + beVector .

5,4中求出了E点坐标,如果能够求出向量CE和向量ED,那么点C和点D的坐标就知道了。

因为向量CD和向量AB垂直,所以向量CD为:

cdVector = ( –abVector.Y , abVector.X);

同时对cdVector 单位化,此时cdVector 代表了CD方向。

6,则CE向量:

ceVector = ceWidth*cdVector ;

则点C eP – cP = ceVector ; 即 cP = eP – ceVector

同理, dP =  eP – deVector;

7,此时求出了点 C和点D,加上点B可以构成一个Polygon了。

C# 画箭头的更多相关文章

  1. 菱形实现气泡Bubble,菱形画箭头,菱形画三角形

    菱形实现气泡Bubble,菱形画箭头,菱形画三角形 >>>>>>>>>>>>>>>>>>&g ...

  2. android 使用Canvas画箭头

    public class MyCanvas extends View{        private Canvas myCanvas;    private Paint myPaint=new Pai ...

  3. Swift实时画箭头的实现

    iOS上实现画箭头,如果是指定了坐标点,那是很简单的,但如果需要做到实时绘制,就需要计算一下了 需求: 在白板上,根据手势落下点和移动点,实时绘制一条箭头直线(如下图) 实现代码: /// 获取箭头的 ...

  4. SVG 使用marker画箭头(一)

    一.使用Marker画箭头 1.定义一个箭头的marker引用 <defs> <marker id='markerArrow' markerWidth='13' markerHeig ...

  5. WPF画箭头

    简介 参考Using WPF to Visualize a Graph with Circular Dependencies的基础上写了一个WPF画箭头的库. 效果图如下: 使用的XAML代码如下: ...

  6. canvas画箭头demo

    效果图: 代码: <!DOCTYPE html> <html> <title>canvas画箭头demo</title> <body> &l ...

  7. D2D画箭头的例子

    原文:D2D画箭头的例子 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/sunnyloves/article/details/50830102 用处 ...

  8. 如何用CorelDRAW画箭头?

    CorelDRAW,简称为cdr,是一款专业的矢量绘图软件,在设计界也是常用的专业设计之一,在日常的设计工作中,我们常常需要绘制一些特殊的图形,比如箭头.很多对cdr不是特别熟练的小伙伴不知道如何用c ...

  9. 在matlab 画箭头

    [转载]在matlab 画箭头 原文地址:在matlab 画箭头作者:纯情小郎君 完整见链接http://www.mathworks.com/matlabcentral/fx_files/14056/ ...

  10. Workflow:采用坐标变换(移动和旋转)画箭头

    背景 流程设计器的连线部分需要画一个箭头代表连接的方向,下图是期望的效果: 刚开始我准备采用三角函数(sin和cos)来计算三角的坐标,实现的过程真不爽(有兴趣的朋友可以试试),就在完工的时候,突然想 ...

随机推荐

  1. 【UVA】11992 - Fast Matrix Operations(段树模板)

    主体段树,要注意,因为有set和add操作,当慵懒的标志下推.递归优先set,后复发add,每次运行set行动add马克清0 WA了好几次是由于计算那一段的时候出问题了,可笑的是我对着模板找了一个多小 ...

  2. Android正在使用Handler实现消息分发机制(零)

    演讲前,AsyncTask文章.我们在最后谈到.AsyncTask它是利用Handler异步消息处理机制,操作结果.使用Message回到主线程,从而执行UI更新线程. 而在我们的日常开发工作,Han ...

  3. Harden the Hacker Thinking (Updating)

    录制自己的最新思考harden过程.通过记录,反射,加强管理,发现缺陷. 等一下design,等一下coding,三十分钟rethinking. 2015-02-26 : 不要在一件事上停留太久: 歇 ...

  4. springboot 使用日志

    spring boot 默认使用日志打印到console 添加application.properties文件在src/main/resoures文件夹下 logging.file=my.log 将日 ...

  5. 在实现视频播放器的步骤client(三)风行网络电影列表

    (三) 今日热门电影实现这个功能.主要从server获取数据.然后显示在屏幕上.虽然说是从这个server获取电影信息数据,但,不实际的http相关知识,我们直接sdk包(56网络提供api),你将能 ...

  6. c#中的GetUpperBound,GetLowerBound方法

    今天使用数组的时候,用到了几个数组的属性,总结如下: Array的Rank 属性:语法:public int Rank { get; } 得到Array的秩(维数).Array的GetUpperBou ...

  7. 图像处理中的跨度(stride)

    原文:图像处理中的跨度(stride) 使用C#的BitmapData 最近要转开发平台,正研究C#.C#好是好,不过处理图片时一个像素一个像素的操作像素不是一般的慢.其实Delphi也一样,但好在D ...

  8. C# Oracle数据库操作类

    using System; using System.Data; using System.Collections.Generic; using System.Configuration; using ...

  9. chrome 里面js提示Provisional headers are shown错误

    参考:http://stackoverflow.com/questions/21177387/caution-provisional-headers-are-shown-in-chrome-debug ...

  10. virtualbox 安装 extension pack 方法以及出现 "The installer failed with exit code 1: VBoxExtPackHelperApp.exe: error: Failed to rename the temporary directory to the final one"的解决办法

    virtualbox 的版本:5.1.26    下载地址:https://www.virtualbox.org/wiki/Downloads extension pack 的版本:5.1.26   ...