详解iOS App中图片的线段涂鸦功能的添加方法

 更新时间:2016年03月30日 09:30:31   作者:lwjok2007  
这篇文章主要介绍了如何设计iOS App中图片的线段涂鸦功能,也就是很多应用中图片上传时带有的编辑功能的基础,需要的朋友可以参考下

接下来我们要讲图片的涂鸦,我们分开一点一点拓展,先给图片上划线
创建项目 起名testAddLine

201633092150487.png (730×518)

201633092222470.png (730×518)

接下来我们在默认生成的ViewController中添加一张图片 待用
同时添加一个按钮

复制代码 代码如下:

- (void)viewDidLoad { 
    [super viewDidLoad]; 
    // Do any additional setup after loading the view, typically from a nib. 
     
    UIImageView *imageV = [[UIImageView alloc]initWithFrame:CGRectMake(10, 120, screen_Width-20, screen_Height-150)]; 
    imageV.image = [UIImage imageNamed:@"640-960-1.jpg"]; 
    [self.view addSubview:imageV]; 
     
    UIButton *testBtn = [[UIButton alloc]initWithFrame:CGRectMake(screen_Width/2.0-60, 60, 120, 36)]; 
    [testBtn setTitleColor:[UIColor blackColor] forState:UIControlStateNormal]; 
    [testBtn setTitle:@"添加直线" forState:UIControlStateNormal]; 
    [testBtn addTarget:self action:@selector(addLineAct:) forControlEvents:UIControlEventTouchUpInside]; 
    [self.view addSubview:testBtn]; 

 
- (void)addLineAct:(id)sender{ 
    NSLog(@"测试按钮"); 


接下来我们创建一个UIView 用来添加直线 起名:DrawLine

201633092246372.png (730×518)

201633092304425.png (730×518)

创建几个变量

复制代码 代码如下:

@property(nonatomic,strong) NSMutableArray * completeLines; //已经画好的线条 存入数组 
@property(nonatomic,strong) NSMutableDictionary* LinesInProscess; //正在画的线条 存入字典 
@property(nonatomic,strong) UIColor *lineColor;//线条颜色 
@property (nonatomic)float lineWidth;//线条的粗细 

初始化DrawLine
复制代码 代码如下:

//初始化 
- (id)initWithFrame:(CGRect)frame{ 
    if (self = [super initWithFrame:frame]) { 
        //初始化变量 
        _completeLines = [[NSMutableArray alloc]init]; 
        _LinesInProscess = [[NSMutableDictionary alloc]init]; 
        //设置透明背景 
        self.backgroundColor = [UIColor clearColor]; 
         
    } 
     
    return  self; 


我们把线条单独抽象出来 创建一个类 创建对象 起名 Line

201633092342893.png (730×518)

201633092404783.png (730×518)

线条 两个属性 起始点 结束点(这就是数学中的两点确定一条直线)
给Line 类创建两个属性

复制代码 代码如下:

#import <Foundation/Foundation.h> 
#import <UIKit/UIKit.h> 
 
@interface Line : NSObject 
 
@property(nonatomic)CGPoint begin; //线条开始点 
 
@property(nonatomic)CGPoint end; //线条结束点 
 
@end 

接下来 我们重写DrawLine 的  drawRect 方法  绘制线条
复制代码 代码如下:

// Only override drawRect: if you perform custom drawing. 
// An empty implementation adversely affects performance during animation. 
- (void)drawRect:(CGRect)rect { 
    // Drawing code 
     
    //获取上下文 
    CGContextRef cgt=UIGraphicsGetCurrentContext(); 
    //设置线条宽度 
    CGContextSetLineWidth(cgt, self.lineWidth); 
    //设置线条两端形状为圆角 
    CGContextSetLineCap(cgt, kCGLineCapRound); 
     
    //设置颜色 
    [self.lineColor set]; 
    //绘制已经完成的线段 
    for (Line *line in _completeLines){ 
        CGContextMoveToPoint(cgt, [line begin].x, [line begin].y); 
        CGContextAddLineToPoint(cgt, [line end].x, [line end].y ); 
        CGContextStrokePath(cgt); 
    } 
     
     
    //绘制正在画的线段 
    for (NSArray *v in _LinesInProscess) { 
        Line *line =[_LinesInProscess objectForKey:v]; 
        CGContextMoveToPoint(cgt, [line begin].x, [line begin].y); 
        CGContextAddLineToPoint(cgt, [line end].x, [line end].y ); 
        CGContextStrokePath(cgt); 
    } 
     


实现几个手指滑动方法 用来接受手指的位置画线
复制代码 代码如下:

//清空画板 
-(void)clearAll 

    [_completeLines removeLastObject]; 
    [_LinesInProscess removeAllObjects]; 
    [self setNeedsDisplay]; 

 
 
-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event 

    //判断是否连按 
    for (UITouch *t in touches) { 
        if ([t tapCount]>1) { 
            //第二次画线时第一条线还未完成时结束画线 
            [self clearAll]; 
            return; 
        } 
         
        //NSValue 作为键使用 
        NSValue *key=[NSValue valueWithNonretainedObject:t]; 
         
        // 根据触摸位置创建Line对象 
        CGPoint loc=[t locationInView:self]; 
        Line *newLine=[[Line alloc]init ]; 
        newLine.begin=loc; 
        newLine.end=loc; 
        //将当前正在画的线存入字典 
        [_LinesInProscess setObject:newLine forKey:key]; 
         
    } 

 
-(void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event 

    //手指移动过程中按照当前手指的位置动态更新线条 
    for (UITouch * t in touches) { 
        NSValue *key=[NSValue valueWithNonretainedObject:t]; 
        // 找对象当前UITouch对象的Line对象 
        Line *line =[_LinesInProscess objectForKey:key]; 
         
        CGPoint loc=[t locationInView:self]; 
        line.end=loc; 
    } 
    [self setNeedsDisplay]; 

 
-(void)endTouches:(NSSet *) touches 

    //画线完成之后将当前线条加入_completeLines 数组中 同时删除字典_LinesInProscess里的线条 
    for (UITouch *t in touches) { 
        NSValue *key=[NSValue valueWithNonretainedObject:t]; 
        Line *line =[_LinesInProscess objectForKey:key]; 
        if (line) { 
            [_completeLines addObject:line]; 
            [_LinesInProscess removeObjectForKey:key]; 
        } 
    } 
    [self setNeedsDisplay]; 

 
-(void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event 

    [self endTouches:touches]; 

 
-(void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event 

    [self endTouches:touches]; 

回到 ViewController中 给按钮点击事件中 添加DrawLine到ImageView上

复制代码 代码如下:

- (void)addLineAct:(id)sender{ 
    NSLog(@"测试按钮"); 
     
    DrawLine *touchdrawView = [[DrawLine alloc]initWithFrame:imageV.frame]; 
 
    touchdrawView.lineColor = [UIColor yellowColor]; 
    touchdrawView.lineWidth = 5.0; 
    touchdrawView.tag = 902; 
    [self.view addSubview:touchdrawView]; 
     
     


好了 运行程序试试
点击 添加直线 按钮之后 试试在图片上画线

201633092437200.jpg (640×1136)

带剪头的线条
在上面例子的基础上稍微拓展一下,给线段末尾加上一个箭头
给DrawLine 类中添的方法 drawRect 中添加一段代码

201633092553144.jpg (675×244)

复制代码 代码如下:

//添加剪头 
double r = sqrt((line.end.x-line.begin.x)*(line.end.x-line.begin.x)+(line.begin.y-line.end.y)*(line.begin.y-line.end.y));//线条长度 
CGContextMoveToPoint(cgt,line.end.x,line.end.y); 
//P1 
CGContextAddLineToPoint(cgt,line.end.x-(10*(line.begin.y-line.end.y)/r),line.end.y-(10*(line.end.x-line.begin.x)/r)); 
//P3 
CGContextAddLineToPoint(cgt,line.end.x+(20*(line.end.x-line.begin.x)/r), line.end.y-(20*(line.begin.y-line.end.y)/r)); 
//P2 
CGContextAddLineToPoint(cgt,line.end.x+(10*(line.begin.y-line.end.y)/r),line.end.y+(10*(line.end.x-line.begin.x)/r)); 
 
CGContextAddLineToPoint(cgt, line.end.x,line.end.y); 
CGContextDrawPath(cgt,kCGPathFillStroke); 
CGContextStrokePath(cgt); 

以上方法的思路 就是在线段画完之后 确定三个点 画一个三角形作为箭头形状

201633092502978.jpg (556×506)

相关文章

  • IOS倒计时设置UIButton标题title的抖动问题

    IOS倒计时设置UIButton标题title的抖动问题

    这篇文章主要介绍了IOS倒计时设置UIButton标题title的抖动问题,在发送验证码后,button状态需要变为disable,每隔一秒显示倒计时时间,下面通过本文给大家介绍设置方法,一起看看吧
    2016-12-12
  • IOS App 无代码入侵的方法hook详细介绍

    IOS App 无代码入侵的方法hook详细介绍

    这篇文章主要介绍了IOS App 无代码入侵的方法hook详细介绍的相关资料,需要的朋友可以参考下
    2016-12-12
  • iOS UIAlertController中UITextField添加晃动效果与边框颜色详解

    iOS UIAlertController中UITextField添加晃动效果与边框颜色详解

    这篇文章主要给大家介绍了关于iOS UIAlertController中UITextField添加晃动效果与边框颜色的相关资料,实现后的效果非常适合在开发中使用,文中给出了详细的示例代码,需要的朋友可以参考借鉴,下面随着小编来一起看看吧。
    2017-10-10
  • iOS 对plist文件进行读写,增删改查的实例

    iOS 对plist文件进行读写,增删改查的实例

    下面小编就为大家带来一篇iOS 对plist文件进行读写,增删改查的实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-02-02
  • iOS中表单列表样式键盘遮挡的解决方案

    iOS中表单列表样式键盘遮挡的解决方案

    这篇文章主要给大家介绍了关于iOS中表单列表样式键盘遮挡的解决方案,文中通过示例代码将解决的方法一步步介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面来一起看看吧
    2019-01-01
  • iOS 用Swipe手势和动画实现循环播放图片示例

    iOS 用Swipe手势和动画实现循环播放图片示例

    本篇文章主要介绍了iOS 用Swipe手势和动画实现循环播放图片示例,非常具有实用价值,需要的朋友可以参考下。
    2017-03-03
  • iOS中读写锁的简单实现方法实例

    iOS中读写锁的简单实现方法实例

    读写锁是计算机程序的并发控制的一种同步机制,也称“共享-互斥锁”、多读者-单写者锁,读操作可并发重入,写操作是互斥的,这篇文章主要给大家介绍了关于iOS中读写锁的简单实现方法,需要的朋友可以参考下
    2021-11-11
  • iOS开发————详解适配iOS10问题

    iOS开发————详解适配iOS10问题

    ios10已经推出一段时间了,这篇文章主要介绍了iOS开发————详解适配iOS10,有兴趣的可以了解一下。
    2016-12-12
  • 分享一些iOS开发实用的小技巧

    分享一些iOS开发实用的小技巧

    这篇文章主要给大家分享了一些iOS开发实用的小技巧,这些小技巧在大家开发iOS的时候还是相当实用,有需要的朋友们下面来一起看看吧。
    2016-09-09
  • iOS下一键调试Push的方法详解

    iOS下一键调试Push的方法详解

    这篇文章主要给大家介绍了关于iOS下如何一键调试Push的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-03-03

最新评论