上篇博客《iOS可视化动态绘制八种排序过程》可视化了一下一些排序的过程,本篇博客就来聊聊图的东西。在之前的博客中详细的讲过图的相关内容,比如《图的物理存储结构与深搜、广搜》。当然之前写的程序是比较抽象的。上篇博客我们以可视化的方式看了一下各种排序的过程,今天博客中我们就来可视化的看一下图的相关部分,今天我们要画的图是无向图,并且每个点到其他点都有直接的连线。今天我们就基于此图来做一些事情。当然本篇博客在画图时我们使用的是Bezier曲线来画的,因为之前也聊过关于Bezier的相关东西,所以今天就不对Bezier做过多赘述了。

今天的博客我们有易到难大致分为三个部分。第一部分我们会画出相应的图,并该图是可以对每个点进行拖动的,在拖动的过程中,我们对其进行重绘。第二部分会取消拖动,使用UIView自带的动画来让其自己变换,当然本部分你也可以使用Timer或者GCD的TimerSource让其运动。第三部分则是第二部分的升级,再第二部分的基础上我们稍作改进,此部分我们使用的是DispatchSourceTimer来让每个点进行运动的。在第三部分我们让局部范围的点进行连线,也就是在运动的过程中,我们需要找出在当前点的规定范围内有哪些点,然后将这些点进行连接。

上述这三部分的内容下方会详细的进行介绍,并会附有相应的运行结果图。接下来就进入我们的主题部分。

一、图的绘制

在本篇博客的第一部分我们要按照要求先把图给绘制出来,我们会随机的生成几个坐标点,然后在这些坐标点上添加上View,然后再将这些坐标点使用Bezier进行连接。当然,在连接时我们使用的是邻接矩阵来记录的每两点之间的关系。在绘制的过程中,我们会随机的为每个点每条边分配颜色。

当相应的图绘制好后,我们需要为每个点添加上Move事件,在对每个点进行拖动时,我们会及时的重新绘制整个图的关系。下方就是我们本部分要实现内容的运行效果,如下所示:

  

如果理解了数据结构中图的构建,实现上述效果,并不困难。解析来我们就来看一下实现上述效果的核心代码。

1、图的节点View的封装

首先我们来封装上述图的节点View,当然此节点View的封装比较简单。核心就在于给每个节点View添加一个TouchesMoved事件,然后在TouchesMoved事件执行时,将触摸的移动点设置成当前View的Center即可。这样我们就可以拖动每个节点View了。在拖动节点View时,我们还需要将拖动的事件回调到节点View的父视图上,让父视图知道当前用户拖动的是哪个View。接下来我们就来看一下节点View的核心代码。

下方这段代码的上一部分就是我们定义的一个闭包类型,用来将节点View的触摸事件回调给父视图。该闭包类型需要传一个参数,该参数就是当前View的Tag, 这样父视图就知道当前用户拖动的是哪个节点了。

而randomColor()函数则是用来负责随机生成颜色的,上面每次颜色的变化都是使用的下方这个函数所随机生成的UIColor对象。

  

下方这段就是节点View的TouchesMoved事件,在该事件中我们获取到当前用户触摸移动的坐标点,然后将该点赋值给当前节点View的Center,然后调用更新父视图的闭包回调对象即可。如下所示:

  

2、图View的封装

接下来我们要实现画图的View了,也就是上述节点View的父视图了。父视图主要负责的工作内容就是创建上述的节点View,然后使用Bezier将每个节点进行连接即可。当然,在用户拖动相应的View的时候,需要对当前图进行重绘。

下方这个方法就是往父视图上添加相应的节点视图,在节点视图初始化后,要设置一个闭包回调,该回调用来移动后图的重绘。在该闭包回调中,我们会调用drawLine()方法。当然在创建节点View时,我们也创建了相应的BezierPath的对象。每个节点对应一个BezierPath对象,用来绘制该节点所连节点的线。具体代码如下所示:

  

我们整个图的关系是存储在邻接矩阵中的,所以我们要对邻接矩阵进行创建,在重绘时要对该邻接矩阵进行初始化。下方就是该邻接矩阵创建和初始化的代码,关于邻接矩阵的内容在此就不做过多赘述了,具体内容请参考之前的博客。

  

节点View和邻接矩阵的准备工作完成后,接下来就是画线的工作了。下方就是画线的核心代码,在画线之前我们要先将相应的BezierPath对象上的点移除掉,然后再添加上新的点,最后就是进行重绘了。在往BezierPath对象上添加点时,我们要将节点的关系在邻接矩阵中进行记录。如果两个点之间已经画完线了,那么邻接矩阵上的内容我们设置为true,未画线的节点之间则是false。具体代码如下所示。

  

在上述方法调用setNeedsDisplay()方法后,就会执行View的draw()方法,我们就在此方法中进行线条的绘制。当然下方的代码比较简单,在此就不做过多赘述了。

  

上述这些代码就是本部分所展示的效果图核心代码,完整示例请移步本篇博客末尾的github分享链接。

二、图的自动变换

上一部分是我们手动的拖动让创建的图进行变换的,接下来我们对上述代码进行改造一下,使其自动的进行变换。在点自动移动时,如果碰到屏幕的边界,我们让其反弹接着进行移动。下方就是我们本部分要实现的效果。

  

当然有了第一部分作为基础,我们实现本部分的效果并不复杂。我们需要做的事情是随机生成每个节点所移动的方向。然后判断移动时是不是超出屏幕范围,如果超出屏幕范围我们就要对运动方向进行修正,让其往反方向进行移动。本部分我们只需要修改节点View,而节点View的父视图不做修改。

下方这段代码片就是为了让其自动变换所实现的方法。下方的这两个方法会替换掉第一部分的TouchesMoved方法。下方的randomIncrement()方法用来生成当前View的x坐标和y坐标的偏移量。x的偏移量为1则表示往右运动,-1表示往左运动。y的偏移量为1则往下运动,-1则是往上运行。

下方的changePoint()就是根据x和y的偏移量不断修改当前节点View的坐标的方法。为了简单,此处使用了UIView自带的Animate来实现的。在修改x和y坐标的值时要判断是否超出屏幕边距,如果超出屏幕边界就往反方向移动。为了让点一直运动下去,我们需要不断的调用changePoint()方法,如下所示。当然每调用一次changePoint()方法,我们就需要调用一下重绘的回调。具体代码如下所示。

  

三、特定区域内画图

接下来我们要做的就是继续在上述内容中做一些东西。在节点自动运动的过程中,我们不把所有的点都连接起来,本部分要做的事情是当点运动时,我们以改点为中心划定个区域,如果有其他点在该区域内,我们就将该区域内的点进行连接。如果点在运动的过程中超出了划定的范围,那么我们就去除之前画的线。效果如下所示:

  

本部分主要修改的内容是节点View的父视图,核心就是要计算当前点与周围点的距离,如果该距离小于我们规定的距离的话,那么我们就画线,否则就不画线。下方代码片段就是本部分的核心代码。主要就是往贝塞尔上添加点时进行距离的判断。下方的countDistance()函数就是用来计算两点之间直线距离的函数,在areaPoints()中调用了该函数来确定当前区域中的点。核心代码如下所示:

  

四、点击新增节点

本部分也将在上述部分的代码上进行更新。该部分要做的事情是点击屏幕,往屏幕上添加新的节点。这一点在上述基础上实现是比较简单的。只需给节点的父View添加上新的节点即可。下方就是第四部分要实现的效果,每点击一次屏幕,就会在屏幕点击的地方生成一个节点,该节点就会运动。具体效果如下所示。

  

要想实现上述效果,下方是我们修改的代码片段。就是给父视图添加了一个TouchesEnded事件,在点击的地方生成一个节点View即可。具体如下所示:

  

本篇博客Demo的github分享地址为:https://github.com/lizelu/FlyOver

p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px Menlo; color: #00b1ff }
span.s1 { }
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 14.0px Menlo; color: #ffffff }
span.s1 { }

iOS可视化动态绘制连通图的更多相关文章

  1. iOS可视化动态绘制连通图(Swift版)

    上篇博客<iOS可视化动态绘制八种排序过程>可视化了一下一些排序的过程,本篇博客就来聊聊图的东西.在之前的博客中详细的讲过图的相关内容,比如<图的物理存储结构与深搜.广搜>.当 ...

  2. iOS可视化动态绘制八种排序过程

    前面几篇博客都是关于排序的,在之前陆陆续续发布的博客中,我们先后介绍了冒泡排序.选择排序.插入排序.希尔排序.堆排序.归并排序以及快速排序.俗话说的好,做事儿要善始善终,本篇博客就算是对之前那几篇博客 ...

  3. iOS可视化动态绘制八种排序过程(Swift版)

    前面几篇博客都是关于排序的,在之前陆陆续续发布的博客中,我们先后介绍了冒泡排序.选择排序.插入排序.希尔排序.堆排序.归并排序以及快速排序.俗话说的好,做事儿要善始善终,本篇博客就算是对之前那几篇博客 ...

  4. iOS本地动态验证码生成-b

    用于ios本地动态生成验证码,效果如下: demo.gif 导入CoreGraphics.framework用于绘制图形 封装UIView,便捷使用,代码如下: AuthcodeView.h #imp ...

  5. iOS应用动态部署方案

    iOS的动态部署能极大的节约成本.苹果的审核周期很长,有的时候,你可能不得不等待将近2个星期去上架你的新功能或者bug.所以动态部署是有价值的. 我这里讨论的情况不把纯web应用考虑在内,因为用户体验 ...

  6. WWDC2014之iOS使用动态库 framework【转】

    from:http://www.cocoachina.com/industry/20140613/8810.html JUN 12TH, 2014 苹果的开放态度 WWDC2014上发布的Xcode6 ...

  7. iOS界面的绘制和渲染

    界面的绘制和渲染 UIView是如何到显示的屏幕上的. 这件事要从RunLoop开始,RunLoop是一个60fps的回调,也就是说每16.7ms绘制一次屏幕,也就是我们需要在这个时间内完成view的 ...

  8. iOS 使用drawRect: 绘制虚线椭圆

    iOS 使用drawRect: 绘制虚线椭圆 1:首先如果要使用 drawRect 绘图 要导入 CoreGraphics.framework 框架 然后 创建 自定义view, 即是 myView继 ...

  9. C# 动态绘制任务栏图标的实现

    通常我们在做一个应用时会遇到这样的需求:将收到的消息条数显示到任务栏,比如如下的效果 怎么实现呢? 答案是采用WindowsAPICodePack实现,具体参见:Windows 7 任务栏开发 之 覆 ...

随机推荐

  1. Spring @Autowired注解在utils静态工具类

    [转] 使用场景:在一个静态方法中,如何使用以下注入: @Autowired private ItemMapper itemMapper; @Component public class TestUt ...

  2. 有关利用python获取网页, 以及KDD近几年论文标题与摘要链接

    最近一直在看KDD的论文,不过,由于老师并没有什么合理的方向性,所以考虑把kdd中的大部分内容都利用python将所有标题.摘要获取下来. 还有一个原因在于,看acm上的摘要,都只显示了两行,再看多点 ...

  3. BZOJ1047: [HAOI2007]理想的正方形

    传送门 蛤省省选果然水啊,我这种蒟蒻都能一遍A. 横向纵向维护两个单调队列,做两次求最大和最小的,总复杂度$O(NM)$ 码农题,考察代码实现能力 //BZOJ 1047 //by Cydiater ...

  4. asp.net 后台实现删除,划掉效果

    效果: name = "<S>" + fircon + "</br>" + "</S>"; 增加“< ...

  5. sqlserver 查看锁表,解锁

    查看被锁表: 代码如下 复制代码 select request_session_id spid,OBJECT_NAME(resource_associated_entity_id) tableName ...

  6. Objective-C中的@property

    1:@property 是什么? 在Objective-C中,@property 是声明属性的语法,它可以快速方便的为实例变量创建getter/setter方法. 2:@property 的本质? @ ...

  7. CSS3/SVG clip-path路径剪裁遮罩属性简介

    一.SVG属性和CSS3属性千丝万缕的关系 CSS3新增属性除了我们现在用的比较多的border-radius, box-shadow, gradient, ...之类,还有很重要的一个分支:SVG属 ...

  8. Java集合的区别和选择

              Collection |--List       有序,可重复 |--ArrayList 底层数据结构是数组,查询快,增删慢. 线程不安全,效率高 |--Vector 底层数据结构 ...

  9. 从Thread.start()方法看Thread源码,多次start一个线程会怎么样

    这篇文章作为Thread类源码剖析的补充,从一个侧面来看Thread源码.也解答了面试高频问题:"多次start一个线程会怎么样?" 答案是:java.lang.IllegalTh ...

  10. windows钩子函数

    一 什么时候用到钩子?(when)Windows操作系统是建立在事件驱动的消息处理机制之上,系统各部分之间的沟通也都是通过消息的相互传递而实现的.通常情况下,应用程序只能处理当前进程的消息,如果需要对 ...