ios开发-OC使用CIFilter生成二维码图片

二维码又称QR Code,QR全称Quick Response,是一个近几年来移动设备上超流行的一种编码方式,它比传统的Bar Code条形码能存更多的信息,也能表示更多的数据类型:比如:字符,数字,日文,中文等等。

在iOS7之后,苹果自身集成了二维码的生成和读取功能。生成二维码包括以下步骤

  • 使用CIFilter滤镜类生成二维码
  • 对生成的二维码进行加工,使其更清晰
    经过上面两步得到会得到原始的二维码图片,在这基础上可以进行以下个性化定制
  • 自定义二维码背景色、填充色
  • 自定义定位角标
  • 在二维码中心插入小图片

二维码生成

本文通过类的方式来生成个性化二维码定制。

初始化二维码信息尺寸和默认值

- (instancetype)init
{
    if (self = [super init]) {
        //默认值
        _info = @"http://mkiltech.com";
        _backgroundColor = [UIColor whiteColor];
        _fillColor = [UIColor blackColor];
    }
    return self;
}

- (void)setInfo:(NSString *)info withSize:(CGFloat)size
{
    _info = info;
    
    CGFloat scale = [UIScreen mainScreen].scale;
    CGRect rect = CGRectMake(0, 0, size, size);
    _size = CGRectGetWidth(rect) * scale;

}

使用CIFilter滤镜类生成二维码

- (void)generateQRCodeFilter
{
    CIFilter *filter = [CIFilter filterWithName:@"CIQRCodeGenerator"];
    [filter setDefaults];
    NSData *data = [_info dataUsingEncoding:NSUTF8StringEncoding];
    [filter setValue:data forKey:@"inputMessage"];              //通过kvo方式给一个字符串,生成二维码
    [filter setValue:@"H" forKey:@"inputCorrectionLevel"];      //设置二维码的纠错水平,越高纠错水平越高,可以污损的范围越大
    
    //设置背景颜色和填充颜色 默认白色背景黑色填充
    
    CIColor *color1 = [CIColor colorWithCGColor:_fillColor.CGColor];
    CIColor *color2 = [CIColor colorWithCGColor:_backgroundColor.CGColor];
    NSDictionary *parameters = [NSDictionary dictionaryWithObjectsAndKeys: filter.outputImage ,@"inputImage",
                                color1,@"inputColor0",
                                color2,@"inputColor1",nil];
    CIFilter *newFilter = [CIFilter filterWithName:@"CIFalseColor" withInputParameters:parameters];
    
    _outPutImage = [newFilter outputImage];                     //拿到二维码图片
}

对生成的二维码进行加工,使其更清晰

- (UIImage *)createNonInterpolatedUIImageFormCIImage:(CIImage *)image withSize:(CGFloat) size {
    CGRect extent = CGRectIntegral(image.extent);
    CGFloat scale = MIN(size/CGRectGetWidth(extent), size/CGRectGetHeight(extent));
    
    // 1.创建bitmap;
    size_t width = CGRectGetWidth(extent) * scale;
    size_t height = CGRectGetHeight(extent) * scale;
    //创建一个DeviceRGB颜色空间
    CGColorSpaceRef cs = CGColorSpaceCreateDeviceRGB();
    //CGBitmapContextCreate(void * _Nullable data, size_t width, size_t height, size_t bitsPerComponent, size_t bytesPerRow, CGColorSpaceRef  _Nullable space, uint32_t bitmapInfo)
    //width:图片宽度像素
    //height:图片高度像素
    //bitsPerComponent:每个颜色的比特值,例如在rgba-32模式下为8
    //bitmapInfo:指定的位图应该包含一个alpha通道。

    CGContextRef bitmapRef = CGBitmapContextCreate(nil, width, height, 8, 0, cs, (CGBitmapInfo)kCGImageAlphaPremultipliedLast);
    CIContext *context = [CIContext contextWithOptions:nil];
    //创建CoreGraphics image
    CGImageRef bitmapImage = [context createCGImage:image fromRect:extent];
    
    CGContextSetInterpolationQuality(bitmapRef, kCGInterpolationNone);
    CGContextScaleCTM(bitmapRef, scale, scale);
    CGContextDrawImage(bitmapRef, extent, bitmapImage);
    
    // 2.保存bitmap到图片
    CGImageRef scaledImage = CGBitmapContextCreateImage(bitmapRef);
    CGContextRelease(bitmapRef); CGImageRelease(bitmapImage);
    
    //原图
    UIImage *outputImage = [UIImage imageWithCGImage:scaledImage];
    return outputImage;
}

获取二维码定位图案位置

二维码一共有40个尺寸。官方叫版本Version。Version 1是21 x 21的矩阵,Version 2是 25 x 25的矩阵,Version 3是29的尺寸,每增加一个version,就会增加4的尺寸,公式是:(V-1)4 + 21(V是版本号) 最高Version 40,(40-1)4+21 = 177,所以最高是177 x 177 的正方形。

下面我们看看一个二维码的样例:


QR-Code-Overview.jpeg

定位图案

  • Position Detection Pattern是定位图案,用于标记二维码的矩形大小。这三个定位图案有白边叫Separators for Postion Detection Patterns。之所以三个而不是四个意思就是三个就可以标识一个矩形了。
  • Timing Patterns也是用于定位的。原因是二维码有40种尺寸,尺寸过大了后需要有根标准线,不然扫描的时候可能会扫歪了。
  • Alignment Patterns 只有Version 2以上(包括Version2)的二维码需要这个东东,同样是为了定位用的。

功能性数据

  • Format Information 存在于所有的尺寸中,用于存放一些格式化数据的。
  • Version Information 在 >= Version 7以上,需要预留两块3 x 6的区域存放一些版本信息。

数据码和纠错码

  • 除了上述的那些地方,剩下的地方存放 Data Code 数据码 和 Error Correction Code 纠错码。

获取Version

- (CGFloat)fetchVersion {
    
    return ((_outPutImage.extent.size.width - 21)/4.0 + 1);
}

换算成绘制坐标

定位图案外层坐标

- (UIBezierPath *) outerPositionPathWidth:(CGFloat)width withVersion:(CGFloat )version wihtPosition:(MKQRPosition) position
{
    CGFloat zonePathWidth = width/((version - 1) * 4 + 21);
    CGFloat positionFrameWidth = zonePathWidth * outerPositionPathOriginLength;
    CGPoint topLeftPoint = CGPointMake(zonePathWidth * 1.5, zonePathWidth * 1.5);
    CGRect rect = CGRectMake(topLeftPoint.x - 0.2, topLeftPoint.y - 0.2, positionFrameWidth, positionFrameWidth);
    
    rect = CGRectIntegral(rect);
    rect = CGRectInset(rect, 1, 1);
    UIBezierPath *path;
    CGFloat offset;
    switch (position) {
        case TopLeft:
            
            path = [UIBezierPath bezierPathWithRect:rect];
            path.lineWidth = zonePathWidth + 1.5;
            path.lineCapStyle = kCGLineCapSquare;
            break;
        case TopRight:
            
            offset = width - positionFrameWidth - topLeftPoint.x * 2;
            rect = CGRectOffset(rect, offset, 0);
            path = [UIBezierPath bezierPathWithRect:rect];
            path.lineWidth = zonePathWidth + 1.5;
            path.lineCapStyle = kCGLineCapSquare;
            break;
        case BottomLeft:
            
            offset = width - positionFrameWidth - topLeftPoint.x * 2;
            rect = CGRectOffset(rect, 0, offset);
            path = [UIBezierPath bezierPathWithRect:rect];
            path.lineWidth = zonePathWidth + 1.5;
            path.lineCapStyle = kCGLineCapSquare;
            break;
        case QuietZone:
            rect = CGRectMake(zonePathWidth * 0.5, zonePathWidth * 0.5, width - zonePathWidth, width - zonePathWidth);
            path = [UIBezierPath bezierPathWithRect:rect];
            path.lineWidth = zonePathWidth + [UIScreen mainScreen].scale;
            path.lineCapStyle = kCGLineCapSquare;
            break;
        default:
            
            path = [UIBezierPath bezierPath];
            break;
    }
    return path;
    
}

定位图案内层坐标包括中心图片位置

- (CGRect)innerPositionRectWidth:(CGFloat )width withVersion:(CGFloat )version wihtPosition:(MKQRPosition) position
{
    CGFloat leftMargin = width * 3 / ((version - 1) * 4 + 21);
    CGFloat tileWidth = leftMargin;
    CGFloat centerImageWith = width * 7 / ((version - 1) * 4 + 21);
    
    CGRect rect = CGRectMake(leftMargin + 1.5, leftMargin + 1.5, leftMargin - 3, leftMargin - 3);
    rect = CGRectIntegral(rect);
    rect = CGRectInset(rect, -1, -1);
    
    CGFloat offset;
    switch (position) {
        case TopLeft:
            
            break;
        case TopRight:
            
            offset = width - tileWidth - leftMargin*2;
            rect = CGRectOffset(rect, offset, 0);
            break;
        case BottomLeft:
            
            offset = width - tileWidth - leftMargin * 2;
            rect = CGRectOffset(rect, 0, offset);
            break;
        case Center:
            
            rect = CGRectMake(CGPointZero.x, CGPointZero.y, centerImageWith, centerImageWith);
            offset = width/2 - centerImageWith/2;
            rect = CGRectOffset(rect, offset, offset);
            break;
        default:
            rect = CGRectZero;
            break;
    }
    
    return rect;
}

效果图

QRCodeRendering.png

完整代码请前往 https://github.com/ymkil/MKQRCode 如果满意的话,给个星哦!!
转载请注明出处。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容