从0开始自定义一个支付密码输入框(1)

对于涉及到资金的app,类似支付宝支付密码(一般为6位数字)的输入框似乎已成为标配?
所以可以封装一个自定义控件来用(主要还是想练练手),效果如下:


从0开始自定义一个支付密码输入框(1)_第1张图片
效果图

1.思路分析

要实现这个效果,有几种思路:
1.先定义单个单元的自定义控件,然后根据个数创建排列。
2.使用CGContextRef直接绘制出边框与黑点。

2.需要用到的姿势

经过考虑,决定选用第二种方案,所需技能:
1.绘制:CGContextRef和UIBezierPath
2.实时预览(可选,加上这个可以在xib中实时预览):IB_DESIGNABLE

3.分析可变属性

1.用一个String保存输入的字符串
2.边框颜色、宽度、圆角弧度以及间距
3.圆点大小、颜色
综上,.h文件中公开的属性如下

@property (nonatomic,copy) IBInspectable NSString * text;  //内容

@property (nonatomic,assign) IBInspectable CGFloat dotRadius;  //圆点的半径大小
@property (nonatomic,copy) IBInspectable UIColor * dotColor;  //圆点的颜色

@property (nonatomic,assign) IBInspectable NSInteger boxCount;  //可以输入几位数
@property (nonatomic,assign) IBInspectable CGFloat boxMargin;  //输入框的间距
@property (nonatomic,assign) IBInspectable CGFloat boxRadius;  //输入框的圆角半径

@property (nonatomic,assign) IBInspectable CGFloat borderWidth;  //输入框边框宽度
@property (nonatomic,copy) IBInspectable UIColor * borderColor;  //输入框边框颜色

4.实现

1、计算每个方框的位置

/**
 * 计算方框的坐标
 * i     第几个方框
 * rect
 **/
-(CGRect)getBoxRect:(int)i
               size:(CGSize)size{
    //考虑到最后一个box需要顶到右侧,所以需要加上一个_boxMargin
    float boxWidth = (size.width + _boxMargin)/self.boxCount - _boxMargin;
    float boxHeight = size.height;
    float left = (self.boxMargin + boxWidth) * i;
    float top = 0;
    return CGRectMake(left, top , boxWidth , boxHeight);
}

2、使用CGContextRef绘制方框

/**
 * 绘制方框
 * context   上下文
 * rect
 **/
-(void)drawBorder:(CGContextRef)context
             size:(CGSize)size{
    //设置线的颜色
    CGContextSetStrokeColorWithColor(context, self.borderColor.CGColor);
    //设置线的宽度
    CGContextSetLineWidth(context, self.borderWidth);
    //调整位置,因为边框如果超出rect会被截取
    CGRect bounds = CGRectInset(rect, _borderWidth * 0.5, _borderWidth * 0.5);
    for (int i = 0 ; i < self.boxCount; i ++) {
            CGRect rect = [self getBoxRect:i size:bounds.size];
            CGRect boxRect = CGRectInset(rect, _borderWidth * 0.5, _borderWidth * 0.5);
            UIBezierPath *bezierPath = [UIBezierPath bezierPathWithRoundedRect:boxRect cornerRadius:_boxRadius];
            CGContextAddPath(context, bezierPath.CGPath);
    }
    //绘制
    CGContextDrawPath(context, kCGPathStroke);
}

3、绘制圆点

/**
 * 绘制圆点
 * context   上下文
 * size  总大小(self的size)
 **/
-(void)drawDot:(CGContextRef)context
          rect:(CGRect)rect{
    //调整位置,因为边框如果超出rect会被截取
    CGRect bounds = CGRectInset(rect, _borderWidth * 0.5, _borderWidth * 0.5);
    for (int i = 0; i < self.boxCount; i ++) {
        //设置填充颜色
        CGContextSetFillColorWithColor(context, [self.dotColor colorWithAlphaComponent:[self.dotAlphas[i] floatValue]].CGColor);
        CGRect rect = [self getBoxRect:i size:bounds.size];
        //圆心的y坐标
        float cy = CGRectGetMidY(rect);
        //圆心的x坐标
        float cx = CGRectGetMidX(rect);
        //半径
        float half = self.dotRadius * [self.dotScans[i] floatValue];
        //添加一个圆
        CGContextAddArc(context, cx, cy,half, 0, 2 * M_PI, 0);
        //绘制填充
        CGContextDrawPath(context, kCGPathFill);
    }
}

4、然后在drawRect:中绘制

-(void)drawRect:(CGRect)rect{
    CGContextRef context = UIGraphicsGetCurrentContext();
    CGSize size = self.frame.size;
    [self drawBorder:context size:size];
    [self drawDot:context size:size];
}

5、试试效果
到这里,基本上的效果已经可以看到了。
由于之前加了IB_DESIGNABLE,所以我们可以在xib中直接预览和修改参数:


从0开始自定义一个支付密码输入框(1)_第2张图片
预览

6、改变文字
经过前面几个步骤,视图效果出来了,但是圆点始终是充满所有方框的,现在我们需要根据text的改变,改变圆点的显示个数(只需要改变一下drawDot:size:中的判断条件就可以了):

-(void)drawDot:(CGContextRef)context
          size:(CGSize)size{
    //圆心的y坐标
    float cy = size.height / 2;
    //半径
    float half = size.width / self.boxCount / 2;
    for (int i = 0; i < MIN(self.text.length, self.boxCount); i ++) {
        //设置填充颜色
        CGContextSetFillColorWithColor(context, self.dotColor.CGColor);
        //圆心的x坐标
        float cx = size.width * i / self.boxCount + half;
        //添加一个圆
        CGContextAddArc(context, cx, cy, self.dotRadius, 0, 2 * M_PI, 0);
        //绘制填充
        CGContextDrawPath(context, kCGPathFill);
    }
}

看看效果:


从0开始自定义一个支付密码输入框(1)_第3张图片
预览1

以上,密码框的基础功能已经实现,这篇文章也告一段落。
下一篇文章,要做的是给圆点加上出现和消失的动画。

你可能感兴趣的:(从0开始自定义一个支付密码输入框(1))