背景
买早餐的时候会遇到,支付宝和微信的二维码贴在一起,然后扫码的时候两个二维码一起被识别出来的情况。之前的处理可能是:APP内部判断 是自己的 Scheme 的时,自动跳转;后来发现变成了识别到多个二维码时,弹出二维码选择页,用户选择具体二维码后,再跳转。
公司的项目一直没有做这个功能,最近有时间,就来整理添加到项目中,这里分享记录一下实现的过程。
过程
整个的过程是:
- 识别二维码
- 只有一个,则直接跳转;
- 有多个二维码信息,则跳转二维码选择页面;
- 二维码选择页面标记出每个二维码的位置;
- 点击对应位置的二维码,跳转对应的链接。
二维码识别
二维码识别的逻辑,代码如下:
1 |
|
上面方法获取到的 features
数组元素有几个,就有几个二维码。features
数组中的元素是CIQRCodeFeature
对象,这个对象中包含有对应二维码的位置和信息。
判断features
,如果count > 1
,则遍历features
,把对应二维码的位置标记出来,生成新的图片,这里需要注意的是,CIQRCodeFeature
中返回的坐标位置不能直接使用,由于坐标系不同的原因,所以需要转换。
代码如下:
1 |
|
生成对应标记好二维码位置的图片后,用新界面显示出来,接下来的问题是,如何判断点击的具体是哪个二维码,这里有两种实现方案:
- 方案一:根据二维码的位置,添加透明的 button 到指定位置,大小等于或大于二维码大小,然后响应按钮事件;
- 方案二:根据 touch事件,判断 touch 的点在哪个二维码的 frame 范围内,则响应哪个事件。
实现过程:
不管是方案一还是方案二,实现过程除了需要注意坐标系的转换外,还要注意缩放比例、偏移的问题,即图片的实际大小和图片要显示的大小计算出缩放比例,按照比例计算出要显示的位置的偏移,然后在对坐标系转换后,进行缩放和偏移处理得到最终的位置。
故而得到实际位置的实现过程如下:
- 得到坐标系转换的 tansform。
- 根据显示宽度和图片实际宽度,计算缩放比例,得到要缩放的 transform。
- 根据缩放比例,和图片显示位置,得到偏移的大小;eg: 图片居中显示,所以(屏幕高度 - 图片高度 * 缩放比例) / 2.0,即是要偏移的大小。
- 遍历识别图片二维码后得到的
features
数组,对数组中每一个元素CIQRCodeFeature
,依次进行坐标系转换、缩放、偏移处理,添加按钮到最终计算后的位置
方案一的实现:
方案一得到最终位置后,在对应位置添加button
,设置 tag,最后根据按钮的响应事件判断点击的是哪个二维码。
代码如下:
1 |
|
方案二的实现:
方案二得到最终位置之后,用对象把位置和二维码信息存储起来,在 touchesBegin:withEvent:
事件中,获取到点击的点,然后判断点击的点在不在二维码范围内,在哪个二维码范围内。
代码如下:
首先定义一个对象,存储二维码信息和二维码位置;并且定义一个方法,根据点判断是否在二维码范围内,可设置误差大小(超出二维码多大范围也算有效)。
1 |
|
然后计算二维码的实际显示的位置,并存储,代码如下:
1 |
|
然后在touchesBegin:withEvent:
方法中,得到点击点,判断点击点是否在二维码范围内,在哪个二维码范围内,代码如下:
1 |
|
整体效果演示如下:
完整代码已放在 Github,地址:https://github.com/mokong/MultipleQRHandle.git