首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >自定义RecyclerView打造Android TV桌面

自定义RecyclerView打造Android TV桌面

作者头像
开发者技术前线
发布2020-11-23 11:22:13
发布2020-11-23 11:22:13
2.7K10
代码可运行
举报
运行总次数:0
代码可运行

最近家里网络出问题了,不能按约定时间将很多不错的文章第一时间推到你的手中,对您说声道歉!本公众号以后也会由其他人打理来发文章!以方便你第一时间阅读供稿者的文章!

前言

Android TV Launcher页在RecyclerView出来之前大家用GridView去实现。TV开发有五向键的监听,遥控器hover监听,点击事件等。用GridView去处理焦点是有一定挑战性的,往往会出现不可预料焦点错乱问题。这里封装了一个针对TV的RecyclerView,很方便的处理了这些事件。

首先上效果图:


这里封装了RecyclerView实现了下面的一些功能:

  • 1.响应五向键,按下五向键的上下左右会跟着移动,并获得焦点,在获得焦点时会抬高。
  • 2.在鼠标hover在条目上时会获得焦点。
  • 3.添加了条目的点击和长按事件。
  • 4.添加了是否第一个可见条目和是否是最后一个可见条目的方法。
  • 5.在item获得焦点时和失去焦点时,这里有相应的回调方法。

实现

下面分析一些关键的点: 1.鼠标滑动时避免跟着滑动,只响应五向键和左右箭头

2.使用StaggeredGridLayoutManager实现管理,如果使用GridLayoutManager会出现焦点的错乱,当使用五向键左右移动时,会从上面转移到下面。原因是GridLayoutManager会存在分组。

3.设置RecyclerView的item有焦点。按五向键,焦点会跟着一起移动

代码语言:javascript
代码运行次数:0
运行
复制
holder.itemView.setFocusable(true);

4,左右键,让RecyclerView跟着一起滚动,并获得焦点:

代码语言:javascript
代码运行次数:0
运行
复制

这里请求获取焦点的方法是:

代码语言:javascript
代码运行次数:0
运行
复制
rightView.requestFocusFromTouch();

TV的焦点的处理的逻辑比较复杂:

可以参考这篇文章:http://www.cnblogs.com/myzh/p/3664544.html

5.在holder里监听到焦点变化时做一些处理:

这里抽象了两个方法,当item获得焦点和失去焦点时调用。获得焦点时条目会抬高,这里是抬高了Z轴。

6.获取在第一个和最后一个可见的条目,根据这些状态去显示和隐藏左右箭头。

代码语言:javascript
代码运行次数:0
运行
复制
   /**
    * 第一个条目是否可见
    *
    * @return 可见返回true,不可见返回false
    */
   public boolean isFirstItemVisible() {
       LayoutManager layoutManager = getLayoutManager();        if (layoutManager instanceof StaggeredGridLayoutManager) {            int[] firstVisibleItems = null;
           firstVisibleItems = ((StaggeredGridLayoutManager) layoutManager).
                   findFirstCompletelyVisibleItemPositions(firstVisibleItems);            int position = firstVisibleItems[0];            return position == 0;
       } else if (layoutManager instanceof LinearLayoutManager) {            int position = ((LinearLayoutManager) layoutManager).findFirstCompletelyVisibleItemPosition();            return position == 0;
       }        return false;
   }    /**
    * 最后一个条目是否可见
    *
    * @param lineNum    行数
    * @param allItemNum item总数
    * @return 可见返回true,不可见返回false
    */
   public boolean isLastItemVisible(int lineNum, int allItemNum) {
       LayoutManager layoutManager = getLayoutManager();        if (layoutManager instanceof StaggeredGridLayoutManager) {            int[] lastVisibleItems = null;
           lastVisibleItems = ((StaggeredGridLayoutManager) layoutManager).findLastCompletelyVisibleItemPositions(lastVisibleItems);            int position = lastVisibleItems[0];
           LogUtil.i(this, "lastVisiblePosition:" + position);            boolean isVisible = position >= (allItemNum - lineNum);            if (isVisible) {
               scrollBy(1, 0);
           }            return isVisible;
       } else if (layoutManager instanceof LinearLayoutManager) {            int position = ((LinearLayoutManager) layoutManager).findLastCompletelyVisibleItemPosition();            return position == allItemNum - 1;
       }        return false;
   }

下面说一个坑,在处理最后一个条目时可见时,我发现拿到的数据并不是一种情况,当一共有三行时。

用下面的代码来打出位置:

有三种情况:

1.最后一列只有有一个时,打出的log是

01-06 02:40:51.868 4135-4135/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:0----->last position:1201-06 02:40:51.869 4135-4135/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:1----->last position:1001-06 02:40:51.869 4135-4135/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:2----->last position:11

2.当最后一列有两个时:

01-06 02:41:54.285 6109-6109/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:0----->last position:1201-06 02:41:54.286 6109-6109/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:1----->last position:1301-06 02:41:54.286 6109-6109/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:2----->last position:11

3.当最后一行有三个时:

01-06 02:43:21.336 8818-8818/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:0----->last position:1201-06 02:43:21.337 8818-8818/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:1----->last position:1301-06 02:43:21.337 8818-8818/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:2----->last position:14

所以这里的处理是传入行数:

代码语言:javascript
代码运行次数:0
运行
复制
boolean isVisible = position >= (allItemNum - lineNum);来判断是否可见。

7.在Recycler滚动时候去处理箭头的显示状态:

结束

注意在使用该控件时,要设置RecyclerView的宽度是Item的整数倍,左右箭头点击滑动的距离也要设置为RecyclerView宽度。 项目的地址:https://github.com/songwenju/CustomTvRecyclerView 如果对你有帮助,欢迎star和fork。

供稿/wenju_song(简书作者) 原文链接:http://www.jianshu.com/p/566bd6188f4d

Tamic开发社区

专业高水准的移动社区

Android & iOS

长按二维码关注

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2016-12-10,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 开发者技术前线 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 首先上效果图:
  • 实现
  • 结束
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档