前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >图片中多个二维码选择的实现

图片中多个二维码选择的实现

作者头像
莫空9081
发布2023-10-16 09:55:45
4050
发布2023-10-16 09:55:45
举报
文章被收录于专栏:iOS 备忘录

背景

买早餐的时候会遇到,支付宝和微信的二维码贴在一起,然后扫码的时候两个二维码一起被识别出来的情况。之前的处理可能是:APP内部判断 是自己的 Scheme 的时,自动跳转;后来发现变成了识别到多个二维码时,弹出二维码选择页,用户选择具体二维码后,再跳转。

公司的项目一直没有做这个功能,最近有时间,就来整理添加到项目中,这里分享记录一下实现的过程。

过程

整个的过程是:

  • 识别二维码
    • 只有一个,则直接跳转;
    • 有多个二维码信息,则跳转二维码选择页面;
      • 二维码选择页面标记出每个二维码的位置;
      • 点击对应位置的二维码,跳转对应的链接。

二维码识别

二维码识别的逻辑,代码如下:

代码语言:javascript
复制

// UIImage + Category

//识别二维码图片
- (NSArray <CIFeature*> *)imageQRFeatures {
    CIImage *ciImage = [[CIImage alloc] initWithCGImage:self.CGImage options:nil];
    
    CIContext *content = [CIContext contextWithOptions:@{kCIContextUseSoftwareRenderer : @(YES)}];
    CIDetector *detector = [CIDetector detectorOfType:CIDetectorTypeQRCode context:content options:@{CIDetectorAccuracy : CIDetectorAccuracyLow}];
    NSArray *features = [detector featuresInImage:ciImage];
    return features;
}

上面方法获取到的 features数组元素有几个,就有几个二维码。features数组中的元素是CIQRCodeFeature对象,这个对象中包含有对应二维码的位置和信息。

判断features,如果count > 1,则遍历features,把对应二维码的位置标记出来,生成新的图片,这里需要注意的是,CIQRCodeFeature中返回的坐标位置不能直接使用,由于坐标系不同的原因,所以需要转换。

代码如下:

代码语言:javascript
复制

// 使用的类
UIImage *targetImage = [UIImage imageNamed:@"Your Image"];
NSArray *features = [targetImage imageQRFeatures];
if ((features) && (features.count > 1))  {
    // 说明有不止一个二维码
    for (CIQRCodeFeature *feature in features) {
        firstImage = [firstImage drawQRBorder:firstImage features:feature];
    }
}


// UIImage + Category

- (UIImage *)drawQRBorder:(UIImage *)targetImage features:(CIQRCodeFeature *)feature {
    CGSize size = targetImage.size;
    UIGraphicsBeginImageContext(size);
    [targetImage drawInRect:CGRectMake(0.0, 0.0, size.width, size.height)];
    
    // 绘制边框,识别出的 bounds 和 image 的坐标系不同,所以需要翻转
    CGContextRef context = UIGraphicsGetCurrentContext();
    // 翻转坐标系
    CGContextScaleCTM(context, 1, -1);
    CGContextTranslateCTM(context, 0, -size.height);
    
    UIBezierPath *path = [UIBezierPath bezierPathWithRect:feature.bounds];

    // 标记框的颜色
    [[UIColor colorWithRed:255.0/255.0 green:59.0/255.0 blue:48.0/255.0 alpha:1.0] setStroke];

    path.lineWidth = 3.0;
    [path stroke];
    
    UIImage *resultImage = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    return resultImage;
}

生成对应标记好二维码位置的图片后,用新界面显示出来,接下来的问题是,如何判断点击的具体是哪个二维码,这里有两种实现方案:

  • 方案一:根据二维码的位置,添加透明的 button 到指定位置,大小等于或大于二维码大小,然后响应按钮事件;
    • 方案二:根据 touch事件,判断 touch 的点在哪个二维码的 frame 范围内,则响应哪个事件。

实现过程: 不管是方案一还是方案二,实现过程除了需要注意坐标系的转换外,还要注意缩放比例、偏移的问题,即图片的实际大小和图片要显示的大小计算出缩放比例,按照比例计算出要显示的位置的偏移,然后在对坐标系转换后,进行缩放和偏移处理得到最终的位置。

故而得到实际位置的实现过程如下:

  1. 得到坐标系转换的 tansform。
  2. 根据显示宽度和图片实际宽度,计算缩放比例,得到要缩放的 transform。
  3. 根据缩放比例,和图片显示位置,得到偏移的大小;eg: 图片居中显示,所以(屏幕高度 - 图片高度 * 缩放比例) / 2.0,即是要偏移的大小。
  4. 遍历识别图片二维码后得到的features数组,对数组中每一个元素CIQRCodeFeature,依次进行坐标系转换、缩放、偏移处理,添加按钮到最终计算后的位置
方案一的实现:

方案一得到最终位置后,在对应位置添加button,设置 tag,最后根据按钮的响应事件判断点击的是哪个二维码。

代码如下:

代码语言:javascript
复制

static NSInteger kTagBeginValue = 1000;

- (void)addAlphaButtons {
    self.messageList = [NSMutableArray array];
    
    // 坐标系转换的 transform
    CGAffineTransform transform = CGAffineTransformIdentity;
    transform = CGAffineTransformScale(transform, 1, -1);
    transform = CGAffineTransformTranslate(transform, 0, -self.displayImage.size.height);

    // 计算缩放比例,展示宽度(屏幕宽度) / 图片实际宽度
    CGFloat scaleX = self.view.bounds.size.width / self.displayImage.size.width;
    CGFloat scaleY = scaleX;

    // 得到要缩放的 transform
    CGAffineTransform scaleTransform = CGAffineTransformMakeScale(scaleX, scaleY);

    // 图片居中显示,所以(屏幕高度 - 图片高度 * 缩放比例) / 2.0,即是要偏移的大小
    CGFloat offsetY = (zScreenHeight - self.displayImage.size.height * scaleY) / 2.0;
    
    for (CIQRCodeFeature *feature in self.features) {
        NSInteger index = [self.features indexOfObject:feature];
        if (!IsNilString(feature.messageString) &&
            (index != NSNotFound)) {
            
            // 坐标系转换
            CGRect frame = CGRectApplyAffineTransform(feature.bounds, transform);
            // 缩放转换
            frame = CGRectApplyAffineTransform(frame, scaleTransform);
            // 偏移量处理
            frame.origin.y += offsetY;
            
            UIButton *tempButton = [UIButton buttonWithType:UIButtonTypeCustom];
            tempButton.backgroundColor = [UIColor clearColor];
            tempButton.frame = frame;
            [self.view addSubview:tempButton];
            tempButton.tag = kTagBeginValue + index;
            [tempButton addTarget:self action:@selector(handleBtnAction:) forControlEvents:UIControlEventTouchUpInside];
            
            [self.messageList addObject:feature.messageString];
        }
    }
}

- (void)handleBtnAction:(UIButton *)sender {
    NSInteger index = sender.tag - kTagBeginValue;
    if (index < self.messageList.count) {
        NSString *scanQRStr = self.messageList[index];
        if (self.selectScanStrBlock) {
            self.selectScanStrBlock(scanQRStr);
            [self dismissViewControllerAnimated:NO completion:nil];
        }
    }
}
方案二的实现:

方案二得到最终位置之后,用对象把位置和二维码信息存储起来,在 touchesBegin:withEvent: 事件中,获取到点击的点,然后判断点击的点在不在二维码范围内,在哪个二维码范围内。

代码如下:

首先定义一个对象,存储二维码信息和二维码位置;并且定义一个方法,根据点判断是否在二维码范围内,可设置误差大小(超出二维码多大范围也算有效)。

代码语言:javascript
复制

// WPSSelectScanImageItem.h

@interface WPSSelectScanImageItem : NSObject

@property (nonatomic, strong) NSString *qrcodeStr;
@property (nonatomic, assign) CGRect qrcodeFrame;

// 判断point 是否在二维码范围内
- (BOOL)isPointInQrcodeFrame:(CGPoint)targetPoint;

@end


// WPSSelectScanImageItem.m

#import "WPSSelectScanImageItem.h"

@implementation WPSSelectScanImageItem

- (BOOL)isPointInQrcodeFrame:(CGPoint)targetPoint {
    BOOL result = NO;
    
    // 误差大小
    CGFloat offsetValue = 10.0;
    
    // 二维码有效范围
    CGFloat minX = self.qrcodeFrame.origin.x - 10;
    CGFloat minY = self.qrcodeFrame.origin.y - 10;
    CGFloat maxX = self.qrcodeFrame.origin.x + self.qrcodeFrame.size.width + offsetValue;
    CGFloat maxY = self.qrcodeFrame.origin.y + self.qrcodeFrame.size.height + offsetValue;
    
    // 要判断的点
    CGFloat targetX = targetPoint.x;
    CGFloat targetY = targetPoint.y;
    
    // 判断点是否在二维码的范围内
    if ((targetX >= minX) &&
        (targetX <= maxX) &&
        (targetY >= minY) &&
        (targetY <= maxY)) {
        result = YES;
    }
    return result;
}

@end

然后计算二维码的实际显示的位置,并存储,代码如下:

代码语言:javascript
复制

- (void)initData {
    self.qrcodeItemList = [NSMutableArray array];
    
    // 坐标系转换的 transform
    CGAffineTransform transform = CGAffineTransformIdentity;
    transform = CGAffineTransformScale(transform, 1, -1);
    transform = CGAffineTransformTranslate(transform, 0, -self.displayImage.size.height);

    // 计算缩放比例,展示宽度(屏幕宽度) / 图片实际宽度
    CGFloat scaleX = self.view.bounds.size.width / self.displayImage.size.width;
    CGFloat scaleY = scaleX;

    // 得到要缩放的 transform
    CGAffineTransform scaleTransform = CGAffineTransformMakeScale(scaleX, scaleY);

    // 图片居中显示,所以(屏幕高度 - 图片高度 * 缩放比例) / 2.0,即是要偏移的大小
    CGFloat offsetY = (zScreenHeight - self.displayImage.size.height * scaleY) / 2.0;
    
    for (CIQRCodeFeature *feature in self.features) {
        NSInteger index = [self.features indexOfObject:feature];
        if (!IsNilString(feature.messageString) &&
            (index != NSNotFound)) {
            
            // 坐标系转换
            CGRect frame = CGRectApplyAffineTransform(feature.bounds, transform);
            // 缩放转换
            frame = CGRectApplyAffineTransform(frame, scaleTransform);
            // 偏移量处理
            frame.origin.y += offsetY;
            
            WPSSelectScanImageItem *item = [WPSSelectScanImageItem new];
            item.qrcodeFrame = frame;
            item.qrcodeStr = feature.messageString;
            [self.qrcodeItemList addObject:item];
        }
    }
}

然后在touchesBegin:withEvent:方法中,得到点击点,判断点击点是否在二维码范围内,在哪个二维码范围内,代码如下:

代码语言:javascript
复制

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    [super touchesBegan:touches withEvent:event];
    
    // 获取touch 对象
    UITouch *touch = touches.anyObject;
    // 获取 touch 点
    CGPoint touchPoint = [touch locationInView:self.view];
    
    // 判断 touch 点在不在二维码范围内
    for (WPSSelectScanImageItem *item in self.qrcodeItemList) {
        BOOL isPointInFrame = [item isPointInQrcodeFrame:touchPoint];
        if (isPointInFrame) {
            if (self.selectScanStrBlock) {
                self.selectScanStrBlock(item.qrcodeStr);
                [self dismissViewControllerAnimated:NO completion:nil];
            }
            break;
        }
    }
}

完整代码已放在 Github,地址:https://github.com/mokong/MultipleQRHandle.git

参考

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022-03-29,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 背景
  • 过程
    • 二维码识别
      • 方案一的实现:
      • 方案二的实现:
  • 参考
相关产品与服务
人脸识别
腾讯云神图·人脸识别(Face Recognition)基于腾讯优图强大的面部分析技术,提供包括人脸检测与分析、比对、搜索、验证、五官定位、活体检测等多种功能,为开发者和企业提供高性能高可用的人脸识别服务。 可应用于在线娱乐、在线身份认证等多种应用场景,充分满足各行业客户的人脸属性识别及用户身份确认等需求。
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档