转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52835829

前言:上篇中,《Android TV开发总结(二)构建一个TV Metro界面(仿泰捷视频TV版)》对应的源码解析见《TV Metro界面(仿泰捷视频TV版)源码解析》一文,链接:http://blog.csdn.net/hejjunlin/article/details/52822499,github对应地址:https://github.com/hejunlin2013/TVSample,截至到当前发稿,已突破200star,如果喜欢的话,可以star,也表示下支持,今天主要总结下TV开发中有焦点问题。

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

  • 采用Android自带的直接控制焦点上下左右
  • 采用setOnFocusChangeListener控制焦点
  • adb按键派发控制焦点
  • 遇到的坑-遥控器按键失灵案列

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

android提供了一些焦点相关的属性,在现有的框架层下通过设置View的属性来获得焦点





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

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

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

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

看如下一段布局:





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,焦点跳过去。

二、采用setOnFocusChangeListener控制焦点

看如下一段代码:





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

三、adb按键派发控制焦点

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





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

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

在adb中,可以通过注入的方式模拟按键,如 adb shell input keyevent 3 给示模拟Home键

以下是的KEYCODE供参考:

  1. KEYCODE_UNKNOWN=0;
  2. KEYCODE_SOFT_LEFT=1;
  3. KEYCODE_SOFT_RIGHT=2;
  4. KEYCODE_HOME=3;
  5. KEYCODE_BACK=4;
  6. KEYCODE_CALL=5;
  7. KEYCODE_ENDCALL=6;
  8. KEYCODE_0=7;
  9. KEYCODE_1=8;
  10. KEYCODE_2=9;
  11. KEYCODE_3=10;
  12. KEYCODE_4=11;
  13. KEYCODE_5=12;
  14. KEYCODE_6=13;
  15. KEYCODE_7=14;
  16. KEYCODE_8=15;
  17. KEYCODE_9=16;
  18. KEYCODE_STAR=17;
  19. KEYCODE_POUND=18;
  20. KEYCODE_DPAD_UP=19;
  21. KEYCODE_DPAD_DOWN=20;
  22. KEYCODE_DPAD_LEFT=21;
  23. KEYCODE_DPAD_RIGHT=22;
  24. KEYCODE_DPAD_CENTER=23;
  25. KEYCODE_VOLUME_UP=24;
  26. KEYCODE_VOLUME_DOWN=25;
  27. KEYCODE_POWER=26;
  28. KEYCODE_CAMERA=27;
  29. KEYCODE_CLEAR=28;
  30. KEYCODE_A=29;
  31. KEYCODE_B=30;
  32. KEYCODE_C=31;
  33. KEYCODE_D=32;
  34. KEYCODE_E=33;
  35. KEYCODE_F=34;
  36. KEYCODE_G=35;
  37. KEYCODE_H=36;
  38. KEYCODE_I=37;
  39. KEYCODE_J=38;
  40. KEYCODE_K=39;
  41. KEYCODE_L=40;
  42. KEYCODE_M=41;
  43. KEYCODE_N=42;
  44. KEYCODE_O=43;
  45. KEYCODE_P=44;
  46. KEYCODE_Q=45;
  47. KEYCODE_R=46;
  48. KEYCODE_S=47;
  49. KEYCODE_T=48;
  50. KEYCODE_U=49;
  51. KEYCODE_V=50;
  52. KEYCODE_W=51;
  53. KEYCODE_X=52;
  54. KEYCODE_Y=53;
  55. KEYCODE_Z=54;
  56. KEYCODE_COMMA=55;
  57. KEYCODE_PERIOD=56;
  58. KEYCODE_ALT_LEFT=57;
  59. KEYCODE_ALT_RIGHT=58;
  60. KEYCODE_SHIFT_LEFT=59;
  61. KEYCODE_SHIFT_RIGHT=60;
  62. KEYCODE_TAB=61;
  63. KEYCODE_SPACE=62;
  64. KEYCODE_SYM=63;
  65. KEYCODE_EXPLORER=64;
  66. KEYCODE_ENVELOPE=65;
  67. KEYCODE_ENTER=66;
  68. KEYCODE_DEL=67;
  69. KEYCODE_GRAVE=68;
  70. KEYCODE_MINUS=69;
  71. KEYCODE_EQUALS=70;
  72. KEYCODE_LEFT_BRACKET=71;
  73. KEYCODE_RIGHT_BRACKET=72;
  74. KEYCODE_BACKSLASH=73;
  75. KEYCODE_SEMICOLON=74;
  76. KEYCODE_APOSTROPHE=75;
  77. KEYCODE_SLASH=76;
  78. KEYCODE_AT=77;
  79. KEYCODE_NUM=78;
  80. KEYCODE_HEADSETHOOK=79;
  81. KEYCODE_FOCUS=80;//*Camera*focus
  82. KEYCODE_PLUS=81;
  83. KEYCODE_MENU=82;
  84. KEYCODE_NOTIFICATION=83;
  85. KEYCODE_SEARCH=84;
  86. KEYCODE_MEDIA_PLAY_PAUSE=85;
  87. KEYCODE_MEDIA_STOP=86;
  88. KEYCODE_MEDIA_NEXT=87;
  89. KEYCODE_MEDIA_PREVIOUS=88;
  90. KEYCODE_MEDIA_REWIND=89;
  91. KEYCODE_MEDIA_FAST_FORWARD=90;
  92. KEYCODE_MUTE=91;

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

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

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

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

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

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

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



  • 问题修复:

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



第一时间获得博客更新提醒,以及更多android干货,源码分析,欢迎关注我的微信公众号,扫一扫下方二维码或者长按识别二维码,即可关注。





如果你觉得好,随手点赞,也是对笔者的肯定,也可以分享此公众号给你更多的人,原创不易

Android TV开发总结(三)构建一个TV app的焦点控制及遇到的坑的更多相关文章

  1. Android TV开发总结(六)构建一个TV app的直播节目实例

    请尊重分享成果,转载请注明出处:http://blog.csdn.net/hejjunlin/article/details/52966319 近年来,Android TV的迅速发展,传统的有线电视受 ...

  2. Android TV开发总结(七)构建一个TV app中的剧集列表控件

    原文:Android TV开发总结(七)构建一个TV app中的剧集列表控件 版权声明:我已委托"维权骑士"(rightknights.com)为我的文章进行维权行动.转载务必转载 ...

  3. Android ROM开发(三)——精简官方ROM并且内置ROOT权限,开启Romer之路

    Android ROM开发(三)--精简官方ROM并且内置ROOT权限,开启Romer之路 相信ROM的相关信息大家通过前几篇的学习都是有所了解了,这里就不在一一提示了,这里我们下载一个官方包,我们还 ...

  4. Android UI开发第三十九篇——Tab界面实现汇总及比较

    Tab布局是iOS的经典布局,Android应用中也有大量应用,前面也写过Android中TAb的实现,<Android UI开发第十八篇——ActivityGroup实现tab功能>.这 ...

  5. 《Android Studio开发实战 从零基础到App上线》资源下载和内容勘误

    转载于:https://blog.csdn.net/aqi00/article/details/73065392 资源下载 下面是<Android Studio开发实战 从零基础到App上线&g ...

  6. Android TV开发总结(四)通过RecycleView构建一个TV app列表页(仿腾讯视频TV版)

    转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52854131 前言:昨晚看锤子手 ...

  7. Android TV开发总结(一)构建一个TV app前要知道的事儿

    转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52792562 前言:近年来,智能 ...

  8. Android TV开发总结(五)TV上屏幕适配总结

    前言:前面几篇总结一些TV上的小Sample,开源到GitHub:https://github.com/hejunlin2013/TVSample, 点击链接,可以持续关注.今天总结下TV上屏幕适配. ...

  9. 安卓TV开发(三) 移动智能设备之实现主流TV电视盒子焦点可控UI

    前言:移动智能设备的发展,推动了安卓另一个领域,包括智能电视和智能家居,以及可穿戴设备的大量使用,但是这些设备上的开发并不是和传统手机开发一样,特别是焦点控制和用户操作体验上有很大的区别,本系列博文主 ...

随机推荐

  1. Python中的上下文管理器和with语句

    Python2.5之后引入了上下文管理器(context manager),算是Python的黑魔法之一,它用于规定某个对象的使用范围.本文是针对于该功能的思考总结. 为什么需要上下文管理器? 首先, ...

  2. Hive优化案例

    1.Hadoop计算框架的特点 数据量大不是问题,数据倾斜是个问题. jobs数比较多的作业效率相对比较低,比如即使有几百万的表,如果多次关联多次汇总,产生十几个jobs,耗时很长.原因是map re ...

  3. [LeetCode] Relative Ranks 相对排名

    Given scores of N athletes, find their relative ranks and the people with the top three highest scor ...

  4. css 宽高自适应的div 元素 如何居中 垂直居中

    在我们 编写css 样式的时候经常会遇见一个问题 那就是一个 宽高未知的元素 要让他 垂直居中如何实现这个呢 下面是我常用的两种方法 上代码 下面的是 结构代码 <div class=" ...

  5. bzoj 2435: [Noi2011]道路修建

    Description 在 W 星球上有 n 个国家.为了各自国家的经济发展,他们决定在各个国家 之间建设双向道路使得国家之间连通.但是每个国家的国王都很吝啬,他们只愿 意修建恰好 n – 1条双向道 ...

  6. bzoj2811[Apio2012]Guard 贪心

    2811: [Apio2012]Guard Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 905  Solved: 387[Submit][Statu ...

  7. HWM、PCTFREE、PCTUSED

    什么是水线(High Water Mark)? HWM通常增长的幅度为一次5个数据块,原则上HWM只会增大,不会缩小,即使将表中的数据全部删除,HWM还是为原值,由于这个特点,使HWM很象一个水库的历 ...

  8. I/O控制的主要功能

    主要功能: 1.  解释用户的I/O系统调用.将用户I/O系统调用转换为I/O控制模块认识的命令模式. 2.  设备驱动.根据得到的I/O命令,启动物理设备完成指定的I/O操作. 3.  中断处理.对 ...

  9. Linux学习之CentOS(十九)------linux 下压缩与解压之 tar、gzip、bzip2、zip、rar

    将文件存储到归档文件中或者从归档文件中获取原始文件,以及为文件创建归档文件 tar [option] [modifiers] [file-list] 参数 file-list是tar进行归档和提取的文 ...

  10. H3C S3100交换机配置VLAN和远程管理

    一.基本设置 1. console线连接成功 2. 进入系统模式 <H3C>system-view //提示符由<H3C> 变为 [H3C] 3. 更改设备名称 [H3C]sy ...