问题: 判断parent view中的手势是否出现在旋转之后的subview的范围中的解决方法(iOS&Android)
解决思路: 取出点击位置的point, 然后translate到subview的坐标系中, 判断是否在subview的矩形中即可.
iOS:
现象:
data:image/s3,"s3://crabby-images/24e1d/24e1df6cf85a8d74fa1bebb3d2db5a52707160b8" alt="7DBEE162-CB37-44E7-8560-4E50556532DA.png"
代码:
data:image/s3,"s3://crabby-images/c6656/c66565bcf46260e2c8b4bddae124159b2438acee" alt="0FDD8A27-5C7F-48EB-A0DD-853FDA4B846C.png"
结果:
data:image/s3,"s3://crabby-images/c8722/c8722f152f06e4b2e0db0790ae84dd0b125b4f96" alt="78067F60-E64B-4FFB-AA67-68783D824B32.png"
部分解释: iOS中的view有frame/bounds/center/transform等坐标属性,
四者的区别(具体可查看文档, 非常详细)
frame
: 相对于parent view的坐标系
bounds
: 相对于自己的坐标系
center
: view的中心点(跟layer的anchorPoint有关, anchorPoint默认值是(0.5, 0.5), 所以是中心点)
transform
: 相对于center的缩放/旋转等二维操作. layer中有个transform是三维的.
三者之间的关系是-bounds/center/transform会影响frame的值.
大部分情况, 我们使用的frame
(x, y, width, height)只影响到了bounds
和center
, 此时transform
是初始值CGAffineTransformIdentity
; 而当我们要设置transform
的时候, frame的值就会变得很奇怪. 其实frame
还是那个矩形, 不过它代表的含义是能包含这个view的最小矩形而已. 因为transform
涉及到scale/rotation等操作, 所以frame
看起来和我们真正想要的值不太一样, 而这时候就是需要分开使用center/bounds/transform的时候, 不能单独设置frame.
layer
的相关属性不介绍, 有兴趣可以直接查看API文档.
Android:
类似的, 我开始在Android中寻找相应的解决方案. 首先需要明白的几个点
- Android的每一个view也有自己的坐标系, 左上角是原点(0, 0)
- Android的view的基础坐标属性是left/top/right/bottom. width和height是由前四个值推算得来.
- Android的view的matrix作用对应于iOS的view的transform.
解决思路还是上面所述, 但是取出点击的point容易, translate到subview的坐标系中遇到了一些问题, iOS中此api存在于UIView中, 而Android的View却没有. 搜索一圈, 还是在Android的事件分发的源码中获取答案.
现象:
data:image/s3,"s3://crabby-images/6c683/6c6837aae3be52b4c08a742bf45215d53a018249" alt="A5809894-B554-433E-B603-172DC01AEB93.png"
代码:
data:image/s3,"s3://crabby-images/a0234/a0234d9760720e00a303d75a4ff76af5c48f5a17" alt="B7B8A792-4D22-4EDF-B672-A96C1A5FCCA4.png"
布局代码:
data:image/s3,"s3://crabby-images/d1c45/d1c45500b1548db60ed1a317d850fd6b5ef98512" alt="007A1AED-36E2-4A51-B5CA-7970D56DC40C.png"
结果:
data:image/s3,"s3://crabby-images/529ac/529ac5bc0e7f509db2be8b9f2f36886a4a2b4fed" alt="2C8C9B9F-FD20-4986-BE8B-BAAA346943FD.png"
部分解释:
代码部分借鉴了Android源码.
data:image/s3,"s3://crabby-images/332ec/332ec59f5e7c5d46d047f4802c6d937654611c53" alt="069605CE-BF37-4BDA-B054-0514E4090EA4.png"
- 先deep copy出一份新的event.
- 设置event的offsetLocation.
- 给event应用subview的inverse matrix.
- 还有很多类似对应的属性(后边继续学习, 本次没有用到)
然后event就被成功的转换到旋转后的view的坐标系中(斜着的), 原点是左上角.
遇到的问题:
- ev.offsetLocation: 因为我是在activity中override onTouchEvent, 所以在计算deltaX和deltaY的时候先减去了relativeLayout距离window的x和y, 再减去了demoTextView的left和top. 1.为什么不直接减去demoTextView距离window的x和y? 因为我发现, demoTextView被旋转之后, 它的left/top/right/bottom并没有改变, 从现象的截图里我们也可以看出. 而demoTextView.getLocationInWindow获取的point却是旋转之后的值, 从log中可以看出. 2.为什么不减去relativeLayout的left和top, 而是减去距离window的x和y? 因为relativeLayout只是activity的contentView, 外层还有多少ViewGroup是未知, 所以直接减去距离window的x和y.
- ev.transform: 源码里面的写法是
transformedEvent.transform(child.getInverseMatrix());
, 而我自己去调用demoTextView.getInverseMatrix()
的时候, 一直报错, 而且API文档中并没有这个方法, 所以就直接去Matrix的类里面去搜关键词inverse
, 发现有这样的方法可以使用.