在做Android?TV應(yīng)用時(shí),經(jīng)常會(huì)有二級(jí)菜單。
左側(cè)是個(gè)listview,右側(cè)是兩個(gè)fragment,選中item時(shí)會(huì)切換银择。
問(wèn)題1:
選中我的會(huì)員進(jìn)入我的會(huì)員fragment,返回listview菜單時(shí)谨履,選中的卻是會(huì)員權(quán)益item欢摄。按照要求應(yīng)是我的會(huì)員處于選中狀態(tài),也就是焦點(diǎn)在我的會(huì)員上笋粟。
原因:
ListView的onFocusChanged方法會(huì)尋找最近的item怀挠,然后選中它。
解決辦法:
重寫(xiě)listview的onFocusChanged方法
public class MemListView extends ListView {
public?MemListView(Context?context)?{
super(context);
}
public?MemListView(Context?context,?AttributeSet?attrs)?{
super(context,?attrs);
}
public?MemListView(Context?context,?AttributeSet?attrs,?int?defStyle)?{
super(context,?attrs,?defStyle);
}
//主要是這個(gè)方法
@Override
protected?void?onFocusChanged(boolean?gainFocus,?int?direction,?Rect?previouslyFocusedRect)?{
int?lastSelectItem?=?getSelectedItemPosition();
super.onFocusChanged(gainFocus,?direction,?previouslyFocusedRect);
if?(gainFocus)?{
setSelection(lastSelectItem);
}
}
}
上面解決返回菜單時(shí)不是原先選中的item害捕,但是又出現(xiàn)了個(gè)小bug绿淋。
問(wèn)題2:listview在初始化的時(shí)候,是沒(méi)有默認(rèn)焦點(diǎn)的尝盼。所以導(dǎo)致初始化時(shí)是焦點(diǎn)的position是-1
設(shè)置listView.requestFocus();listView.setSelection(0);也沒(méi)有用
從源碼中發(fā)現(xiàn)原因:
//AdapterView.class
public int getSelectedItemPosition() {
return mNextSelectedPosition;
}
發(fā)現(xiàn)它返回的是一個(gè)叫mNextSelectedPosition的變量吞滞。說(shuō)明這個(gè)變量是決定當(dāng)前選中的是哪一個(gè)位置的值。
//ListView.class
@Override
public void setSelection(int position) {
setSelectionFromTop(position, 0);
再去到函數(shù)setSelectionFromTop(position, 0)盾沫,重點(diǎn)關(guān)注我注釋了的方法
//ListView.class
public void setSelectionFromTop(int position, int y) {
if (mAdapter == null) {
return;
}
if (!isInTouchMode()) {
position = lookForSelectablePosition(position, true); //查找position是否能被選中裁赠,能被選中則返回原position,這里返回的是原position
if (position >= 0) {
setNextSelectedPositionInt(position); //發(fā)現(xiàn)這里有設(shè)置mNextSelectedPosition的方法
}
} else {
mResurrectToPosition = position;
}
if (position >= 0) {
mLayoutMode = LAYOUT_SPECIFIC;
mSpecificTop = mListPadding.top + y;
if (mNeedSync) {
mSyncPosition = position;
mSyncRowId = mAdapter.getItemId(position);
}
if (mPositionScroller != null) {
mPositionScroller.stop();
}
requestLayout();
}
}
我們可以看到一個(gè)熟悉的名字setNextSelectedPositionInt(position);赴精,剛好在getSelectedItemPosition()里面返回的就是它佩捞,那我們進(jìn)去該方法看看是怎么設(shè)置的。
//AdapterView.class
/**
* Utility to keep mNextSelectedPosition and mNextSelectedRowId in sync
* @param position Intended value for mSelectedPosition the next time we go
* through layout
*/
void setNextSelectedPositionInt(int position) {
mNextSelectedPosition = position; //賦值
mNextSelectedRowId = getItemIdAtPosition(position);
// If we are trying to sync to the selection, update that too
if (mNeedSync && mSyncMode == SYNC_SELECTED_POSITION && position >= 0) {
mSyncPosition = position;
mSyncRowId = mNextSelectedRowId;
}
}
第一句就看到喜聞樂(lè)見(jiàn)的賦值蕾哟,剩下的看起來(lái)好像對(duì)我們也沒(méi)有什么用一忱。
源碼了解地差不多了,可是由結(jié)果(getSelectedItemPosition() = -1)我們知道最后mNextSelectedPosition并沒(méi)有賦值成為我們?cè)O(shè)置的那個(gè) 0谭确,原因嘛帘营,我還不知道,但是我有個(gè)想法逐哈,就是通過(guò)反射把mNextSelectedPosition強(qiáng)制改成 0 芬迄。
mListView.post(new Runnable() {
@Override
public void run() {
//反射
try {
Field field = AdapterView.class.getDeclaredField("mNextSelectedPosition"); //得到該屬性字段,注意該屬性是AdapterView的昂秃,不是ListView的
field.setAccessible(true); //禁用安全檢查薯鼠,如果屬性不是public的,必須要加上此句才能訪問(wèn)
field.set(mListView, 0); //設(shè)置 menuListView 實(shí)例中的 mNextSelectedPosition 為0械蹋,可以想象成 mListView.set(field, 0)
Log.d(TAG, "反射后curPos = " + mListView.getSelectedItemPosition());
} catch (Exception e) {
Log.e(TAG, "反射失敗后curPos = " + mListView.getSelectedItemPosition());
}
}
});
第一句是得到mNextSelectedPosition屬性的字段出皇,用 Field 這個(gè)類(lèi)來(lái)保存,可以這么想:每個(gè)類(lèi)中的屬性都是一個(gè) Field哗戈。
第一句中的AdapterView.class還可以換種寫(xiě)法:Class.forName(android.widget.AdapterView)郊艘,用包名來(lái)得到該類(lèi),效果是一樣的。
第二句是取消Java語(yǔ)言安全性檢查纱注,如果屬性不是public的畏浆,必須要加上此句才能訪問(wèn),不然會(huì)報(bào)錯(cuò)狞贱,并且加上此句好像還能提高性能(因?yàn)楸苊饬藱z查)刻获。
第三句也是我覺(jué)得反射最好玩的地方了!~
field.set(mListView, 0);field是mNextSelectedPosition的字段瞎嬉,mListView是我們的ListView的實(shí)例蝎毡,0是我們要付給mNextSelectedPosition的值。
本來(lái)應(yīng)該是 mListView 里面包含 mNextSelectedPosition字段氧枣,用mListView.mNextSelectedPosition = 0的方法來(lái)設(shè)置值的沐兵,但是因?yàn)槲覀儾荒苤苯拥玫絤NextSelectedPosition屬性,所以通過(guò)反射方法field.set(mListView, 0);來(lái)實(shí)現(xiàn)便监。它們效果是一樣的扎谎,只是寫(xiě)法不同,你可以這么簡(jiǎn)單理解烧董。還可以想象成 mListView.set(field, 0)毁靶。
mListView.setFocusable(true); //設(shè)置可獲得焦點(diǎn)
mListView.setFocusableInTouchMode(true); //設(shè)置可在touch模式獲得焦點(diǎn)
mListView.requestFocus(); //請(qǐng)求獲得焦點(diǎn)