天天看點

android tv 自定義頻道,針對Android Tv的自定義RecyclerView

前言:Android TV Launcher頁在RecyclerView出來之前大家用GridView去實作。TV開發有五向鍵的監聽,遙控器hover監聽,點選事件等。用GridView去處理焦點是有一定挑戰性的,往往會出現不可預料焦點錯亂問題。這裡封裝了一個針對TV的RecyclerView,很友善的處理了這些事件。

首先上效果圖:

android tv 自定義頻道,針對Android Tv的自定義RecyclerView

tvRecycler.gif

這裡封裝了RecyclerView實作了下面的一些功能:

1.響應五向鍵,按下五向鍵的上下左右會跟着移動,并獲得焦點,在獲得焦點時會擡高。

2.在滑鼠hover在條目上時會獲得焦點。

3.添加了條目的點選和長按事件。

4.添加了是否第一個可見條目和是否是最後一個可見條目的方法。

5.在item獲得焦點時和失去焦點時,這裡有相應的回調方法。

實作

下面分析一些關鍵的點:

1.滑鼠滑動時避免跟着滑動,隻響應五向鍵和左右箭頭

@Override

public boolean dispatchTouchEvent(MotionEvent ev) {

//在recyclerView的move事件情況下,攔截調,隻讓它響應五向鍵和左右箭頭移動

LogUtil.i(this, "CustomRecycleView.dispatchTouchEvent.");

return ev.getAction() == MotionEvent.ACTION_MOVE || super.dispatchTouchEvent(ev);

}

2.使用StaggeredGridLayoutManager實作管理,如果使用GridLayoutManager會出現焦點的錯亂,當使用五向鍵左右移動時,會從上面轉移到下面。原因是GridLayoutManager會存在分組。

//設定布局管理器

StaggeredGridLayoutManager layoutManager = new StaggeredGridLayoutManager(3,StaggeredGridLayoutManager.HORIZONTAL);

mRecyclerView.setLayoutManager(layoutManager);

3.設定RecyclerView的item有焦點。按五向鍵,焦點會跟着一起移動

//item可以獲得焦點,需要設定這個屬性。

holder.itemView.setFocusable(true);

4,左右鍵,讓RecyclerView跟着一起滾動,并獲得焦點:

@Override

public boolean dispatchKeyEvent(KeyEvent event) {

boolean result = super.dispatchKeyEvent(event);

int dx = this.getChildAt(0).getWidth();

View focusView = this.getFocusedChild();

if (focusView != null) {

switch (event.getKeyCode()) {

case KeyEvent.KEYCODE_DPAD_RIGHT:

if (event.getAction() == KeyEvent.ACTION_UP) {

return true;

} else {

View rightView = FocusFinder.getInstance().findNextFocus(this, focusView, View.FOCUS_RIGHT);

LogUtil.i(this, "rightView is null:" + (rightView == null));

if (rightView != null) {

rightView.requestFocusFromTouch();

return true;

} else {

this.smoothScrollBy(dx, 0);

return true;

}

}

case KeyEvent.KEYCODE_DPAD_LEFT:

View leftView = FocusFinder.getInstance().findNextFocus(this, focusView, View.FOCUS_LEFT);

LogUtil.i(this, "left is null:" + (leftView == null));

if (event.getAction() == KeyEvent.ACTION_UP) {

return true;

} else {

if (leftView != null) {

leftView.requestFocusFromTouch();

return true;

} else {

this.smoothScrollBy(-dx, 0);

return true;

}

}

}

}

return result;

}

這裡請求擷取焦點的方法是:

rightView.requestFocusFromTouch();

TV的焦點的處理的邏輯比較複雜:

5.在holder裡監聽到焦點變化時做一些處理:

holder.itemView.setOnFocusChangeListener(new View.OnFocusChangeListener() {

@Override

public void onFocusChange(View v, boolean hasFocus) {

if (hasFocus) {

focusStatus(v);

} else {

normalStatus(v);

}

}

});

private void focusStatus(View itemView) {

if (itemView == null) {

return;

}

if (Build.VERSION.SDK_INT >= 21) {

//擡高Z軸

ViewCompat.animate(itemView).scaleX(1.10f).scaleY(1.10f).translationZ(1).start();

} else {

ViewCompat.animate(itemView).scaleX(1.10f).scaleY(1.10f).start();

ViewGroup parent = (ViewGroup) itemView.getParent();

parent.requestLayout();

parent.invalidate();

}

onItemFocus(itemView);

}

protected abstract void onItemFocus(View itemView);

private void normalStatus(View itemView) {

if (itemView == null) {

return;

}

if (Build.VERSION.SDK_INT >= 21) {

ViewCompat.animate(itemView).scaleX(1.0f).scaleY(1.0f).translationZ(0).start();

} else {

ViewCompat.animate(itemView).scaleX(1.0f).scaleY(1.0f).start();

ViewGroup parent = (ViewGroup) itemView.getParent();

parent.requestLayout();

parent.invalidate();

}

onItemGetNormal(itemView);

}

protected abstract void onItemGetNormal(View itemView);

這裡抽象了兩個方法,當item獲得焦點和失去焦點時調用。獲得焦點時條目會擡高,這裡是擡高了Z軸。

6.擷取在第一個和最後一個可見的條目,根據這些狀态去顯示和隐藏左右箭頭。

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;

}

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;

}

下面說一個坑,在處理最後一個條目時可見時,我發現拿到的資料并不是一種情況,當一共有三行時。

用下面的代碼來打出位置:

for (int i = 0; i < lastVisibleItems.length; i++) {

LogUtil.i(this, "order:"+i +"----->last position:" + lastVisibleItems[i]);

}

有三種情況:

1.最後一列隻有有一個時,打出的log是

01-06 02:40:51.868 4135-4135/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:0----->last position:12

01-06 02:40:51.869 4135-4135/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:1----->last position:10

01-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:12

01-06 02:41:54.286 6109-6109/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:1----->last position:13

01-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:12

01-06 02:43:21.337 8818-8818/com.songwenju.customtvrecyclerview I/swjCustomRecyclerView: order:1----->last position:13

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

是以這裡的處理是傳入行數:

boolean isVisible = position >= (allItemNum - lineNum);來判斷是否可見。

7.在Recycler滾動時候去處理箭頭的顯示狀态:

private class MyOnScrollListener extends RecyclerView.OnScrollListener {

@Override

public void onScrollStateChanged(RecyclerView recyclerView, int newState) {

super.onScrollStateChanged(recyclerView, newState);

}

@Override

public void onScrolled(RecyclerView recyclerView, int dx, int dy) {

super.onScrolled(recyclerView, dx, dy);

//在滾動的時候處理箭頭的狀态

setLeftArrStatus();

setRightArrStatus();

}

}

結束:

注意在使用該控件時,要設定RecyclerView的寬度是Item的整數倍,左右箭頭點選滑動的距離也要設定為RecyclerView寬度。

項目的位址:https://github.com/songwenju/CustomTvRecyclerView

如果對你有幫助,歡迎star和fork。

推薦:

歡迎關注我建立的Android TV 簡書專題,會定期給大家分享一些AndroidTv相關的内容:

http://www.jianshu.com/c/37efc6e9799b

版權聲明:本文為部落客原創文章,并已授權微信公衆号Tamic開發社群獨家釋出。