在AndroidTV中一般会把当前拥有焦点的view放大显示,而如果view之间的间隔较小放大后可能会造成压边情况,这时候常见的解决办法就是让拥有焦点的view掉用bringToFront()方法,但这种方法有一个问题就是如果该view的父布局是LinearLayout的话,获取焦点的view会跑到LinearLayout的最后一个位置,也就是说LinearLayout中哪个子view获取焦点哪个在LinearLayout的最后面
解决办法:
使用下面Github中的BringToFrontLinearLayout,直接替换掉布局中的LinearLayout即可,无需其他操作
Github地址:
原因分析:
我们先来看bringToFront()方法到的做了什么,为什么能够解决压边问题,
查看view源码可以看到view中的bringToFront()其实是掉用父布局的bringChildToFront()方法
而在viewGroup中bringChildToFront方法中的操作是
1,将这个子view从保存所有子view的数组中删掉
2,将view加入到子view数组到最后面
3,重新布局和重绘
所以我们知道了bringToFront()方法其实就是把该view移到了父布局维护到子view数组的最后面,而viewGroup默认绘制子view流程是按其维护的数组依次绘制,最后位置的最后绘制自然不会被其他子view遮挡到
而linearLayout中掉用bringToFront()方法的子view会跑到linearLayout最后一个位置的原因,就是因为linearLayout的onLayout中会根据维护的子view数组顺序进行布局,这一点在
bringToFront()方法的注释中其实已经说到了
if the parent container uses an order-dependent layout scheme (e.g., LinearLayout)
解决办法:
前面已经说了viewGroup的默认绘制子view流程是按子view数组顺序进行的,那可不可以更改其绘制顺序呢?答案是可以!
查看viewGroup源码可以发现在dispatchDraw()方法中有上图这样一段代码,我们可以看到在代码中viwGroup绘制子view时会先掉用isChildrenDrawingOrderEnabled()方法得到一个boolean值,然后把这个值传入getAndVerifyPreorderedIndex()方法获得下一个要绘制的子view的index
查看getAndVerifyPreorderedIndex()方法发现如果传入的boolean值是true则会掉用getChildDrawingOrder()方法获取index,否则返回默认值
根据上面的分析,我们可以找到修改viewGroup绘制子view顺序的方法:
1,让isChildrenDrawingOrderEnabled()方法返回true(掉用setChildrenDrawingOrderEnabled(true)即可)
2,重写getChildDrawingOrder()方法实现自定义绘制顺序
这样就可以做到既不会被其他元素压边又不会因为掉用bringToFront()方法而影响到到LinearLayout等线性布局的子view顺序
Github地址: