前言:关于《TV Metro界面(仿泰捷视频TV版)源码解析》由于都是相关代码,就不发公众号了,有兴趣的可以看链接:http://blog.csdn.net/hejjunlin/article/details/52822499,今天介绍下TV开发中有焦点问题。

在TV开发中没有以前我们phone端的dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent 事件来分发,而需要使用dispatchKeyEvent、onKeyDown、onKeyLisenter 等事件来分发处理焦点事件传递,而且TV端焦点没有什么好办法可以全局控制,需要我们自己来想办法规定焦点走向,可以参考我的View焦点总结[《Android View框架总结(二)View焦点》](http://blog.csdn.net/hejjunlin/article/details/52263256),本篇做应用场景补充说明

Android TV 开发与一般Android开发最大的区别在于焦点控制 , 用户在使用Android TV设备主要是通过遥控器操作app。焦点就是让用户知道的直接交互行为。 然而一些app,依据系统对focus的判断,会出现的状况: 上下导航时,不是想要的结果. 边缘移动时,会出现焦点丢失的状况. 有时想直接定位到某个位置上. 
android提供了一些焦点相关的属性,在现有的框架层下通过设置View的属性来获得焦点

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java

下面列出三种方法处理焦点问题



1采用Android自带的直接控制焦点上下左右的方法



因此在进行布局时有必须要通过view.setId(…)指定view的特定ID,然后通过view.setNextLeftView(…)等四个方法控制该view的上下左右移动后所到达的view。然而这种方法只适用于前提就设置好ID的场景,不适合动态布局的场景。 

看如下一段布局:

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_02

rg_a1,rg_a2,rg_a3,rg_a4实际业务中会写明其含义,而不是a1,a2之类,这里只是为介绍,这个自定义的MyCustomButton,按遥控器左键时,将找id为rg_a1的view,焦点跳过去,按遥控器右键时,将找id为rg_a2的view,焦点跳过去,按遥控器下键时,将找id为rg_a3的view,焦点跳过去,按遥控器上键时,将找id为rg_a5的view,焦点跳过去。



2setOnFocusChangeListener



看如下一段代码:

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_03

当OnFocusChangeListener时,就是从一个焦点跳到另一个view上的变化过程。 



3按键派发

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_04


按键的派发须了解一此KeyCode,下面是平时用到的主要的一些方向键:


在按键过程中 按下和松开的Action主要是ACTION_DOWN、ACTION_UP事件分发和处理是在ACTION_DOWN中处理

当设置View.setFocusable(true); 改变控件是否可以获得焦点,然而同时会触发 setOnFocusChangeListener事件

在adb中,可以通过注入的方式模拟按键进行焦点移动,如 adb shell input keyevent 3 给示模拟Home键 
以下是的KEYCODE供参考:

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_05

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_06


遇到的坑-遥控器按键失灵

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_07



问题描述:播放中出现屏保,从屏保回到某页面后,遥控器失灵。从某页进入全屏播放,暂停,等小米的屏保出来后将屏保消失,返回到某页面继续播放。此时遥控器方向键、确定键均无响应。Home键、电源键有响应。菜单键有响应。

分析:没有响应,按键被拦截,查了下当时改动的代码时,对按键并未做特殊拦截…. 对比之前的版本,没有这个问题。确认问题出现在浮层…从log中看,onWindowFocusChanged,在从h5界面/屏保界面回到某页面,没有被调用。

说明从window到activity这层,按键就被吃掉了。



接着分析:整个BaseActivity,没有接收到Action_Down事件

Log中打印的“Dropping event due to no window focus”,思路又断了。

继续做对比,发现show出浮层时,没有任何异常,但是只要show时,焦点移动,就能复现按键不响应。最后就定位到一个自定义控件上

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_08


再接着分析:这个控件1600多行代码,最初一直在找之前版本改动的地方区别,之前版本主要是做一些对这个控件的定制化,其他的先不考虑,排除法,找和event相关的方法。dispatchKeyEvent没有任何异常,又没有思路了,既然是系统级别的传递过程中就被吃了,会不会和view相关,因为只要焦点移动,就失灵,焦点移动伴随着,有一个popupwindow弹出,最终定位在onAttachWindow,好像也没有做什么特殊的事,用了一个getHandler,起初我以为是个自己写的方法,追点进去getHandler一看,是View的


Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_09

问题修复:

总结:用View的Handler以前是为处理popupwindow时,popupwindow通过post的方式去show,但是如果此时activity ondestory则会导致出错,所以加了onDetachWindow和onAttachWindow,原因主要是在onDetachWindow时,mHandler.removeCallbacksAndMessages(null);这句话导致,它相当于是把window发给View这层message给移除了。最后修改只移除它对应的的runnable,问题修复。

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑_Java_10