使用Android GridView进行键盘导航不会滚动网格

时间:2015-04-25 20:13:54

标签: android gridview focus

我正在尝试使用键盘浏览项目的GridView。

在简短的演示中,它只包含一个GridView(包含带有android:focusable="true"的项目视图),你可以看到没有任何项目获得焦点 - 网格是滚动的东西(项目不转动)橙色)。

scrolling short demo

添加android:descendantFocusability="afterDescendants"允许首先聚焦每个项目,但仍然不会继续滚动GridView以便您可以向下导航:

with descendent focusability set

单击某个项目(使用鼠标或按下返回项,如果项目已聚焦,则在后代可聚焦性修复后)将项目变为蓝色 - 表示它处于按下状态。

clicking grid items

@布局/ activity_my.xml:

<?xml version="1.0" encoding="utf-8"?>
<GridView xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/listview"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:numColumns="2" />

@布局/ dummy_item.xml:

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
  android:id="@+id/dummy_item_parent"
  android:layout_width="match_parent"
  android:layout_height="150dp"
  android:gravity="center"
  android:focusable="true"
  android:clickable="true"
  android:background="@drawable/item_background" />

@绘制/ item_background.xml:

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
  <item android:state_pressed="true" android:drawable="@android:color/holo_blue_light" />
  <item android:state_focused="true" android:drawable="@android:color/holo_orange_light" />
  <item android:drawable="@android:color/holo_red_light" />
</selector>

MyActivity.java(使用ListAdapter):

public class MyActivity extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my);

        AbsListView listview = (AbsListView) findViewById(R.id.listview);
        listview.setAdapter(new DummyAdapter(getLayoutInflater()));
    }

    private static class DummyAdapter extends BaseAdapter {

        private static final int COUNT = 25;

        private final LayoutInflater layoutInflater;

        DummyAdapter(LayoutInflater layoutInflater) {
            this.layoutInflater = layoutInflater;
        }

        @Override
        public int getCount() {
            return COUNT;
        }

        @Override
        public String getItem(int position) {
            return "Item " + position;
        }

        @Override
        public long getItemId(int position) {
            return position;
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            View view = convertView;
            if (view == null) {
                view = layoutInflater.inflate(R.layout.dummy_item, parent, false);
            }
            ((TextView) view).setText(getItem(position));
            return view;
        }

    }

}

交换GridView for ListView,而不更改适配器或项目视图/项目视图属性,将按预期运行 - 项目可以聚焦,ListView将滚动到底部,以便所有项目可以使用只是键盘输入。此外,使用带有LinearLayoutManager / GridLayoutManager的RecyclerView也可以正常工作。

我在API 16和22上尝试了相同的结果。我已尝试将GridView上的android:descendantFocusability属性设置为afterDescendents,结果相同。

1 个答案:

答案 0 :(得分:1)

我尝试过你的代码并且它不适用于GridView,正如你所说的那样,但我对ListView也有一些问题。 ListView正确滚动,但只突出显示每个页面的第一项。

通过这些小修改,您可以在GridView和ListView上实现所需的效果。

删除activity_my.xml中的android:descendantFocusability

<?xml version="1.0" encoding="utf-8"?>
<GridView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/listview"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:numColumns="2" />

删除dummy_item.xml中的android:focusable

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/dummy_item_parent"
    android:layout_width="match_parent"
    android:layout_height="150dp"
    android:gravity="center"
    android:clickable="true"
    android:background="@drawable/item_background" />

在item_background.xml中将android:state_focused更改为android:state_selected

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="true" android:drawable="@android:color/holo_blue_light" />
    <item android:state_selected="true" android:drawable="@android:color/holo_orange_light" />
    <item android:drawable="@android:color/holo_red_light" />
</selector>