Ящик навигации: добавьте заголовки в listview
Я создал проект с новым ящиком навигации.
Я хотел бы настроить макет меню, добавить еще один объект, такой как TextView, ImageView... И для начала я хотел бы изменить макет по умолчанию, который состоит только с одним списком, добавив 2 или 3 заголовки в списке.
Сегодня я попытался использовать "addHeaderView", но считаю возможным использовать его только для добавления одного заголовка.
Как я могу добавить заголовок и настроить меню макета? Поскольку, по API-интерфейсу разработчика, кажется, что разрешено только двум дочерним группам под "android.support.v4.widget.DrawerLayout".
Вот сегодня мой захват:
![Заголовки навигационных ящиков в списке]()
И вот захват, который я хочу создать:
![Navigation Drawer Headers in listview]()
Вот фрагмент кода моей MainActivity:
public class MainActivity extends Activity {
private DrawerLayout mDrawerLayout;
private ListView mDrawerList;
private ActionBarDrawerToggle mDrawerToggle;
private CharSequence mDrawerTitle;
private CharSequence mTitle;
private String[] mPlanetTitles;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mTitle = mDrawerTitle = getTitle();
mPlanetTitles = getResources().getStringArray(R.array.planets_array);
mDrawerLayout = (DrawerLayout) findViewById(R.id.drawer_layout);
// Declaration of the 2 listview
mDrawerList = (ListView) findViewById(R.id.dernieres_news);
LayoutInflater inflater = getLayoutInflater();
// Add header news title
ViewGroup header_news = (ViewGroup)inflater.inflate(R.layout.header_dernieres_news, mDrawerList, false);
mDrawerList.addHeaderView(header_news, null, false);
// set a custom shadow that overlays the main content when the drawer opens
mDrawerLayout.setDrawerShadow(R.drawable.drawer_shadow, GravityCompat.START);
String[] names=new String[]{"Title 1", "Title 2", "Title 3", "Title 4", "Title 5"};
/*Array of Images*/
int[] image = new int[] {R.drawable.ic_action_feed, R.drawable.ic_action_feed, R.drawable.ic_action_feed, R.drawable.ic_action_feed, R.drawable.ic_action_feed};
List<HashMap<String, String>> listinfo = new ArrayList<HashMap<String, String>>();
listinfo.clear();
for(int i=0;i<5;i++){
HashMap<String, String> hm = new HashMap<String, String>();
hm.put("name", names[i]);
hm.put("image", Integer.toString(image[i]));
listinfo.add(hm);
}
// Keys used in Hashmap
String[] from = { "image", "name" };
int[] to = { R.id.img, R.id.txt };
SimpleAdapter adapter = new SimpleAdapter(getBaseContext(), listinfo, R.layout.drawer_list_item, from, to);
mDrawerList.setAdapter(adapter);
mDrawerList.setOnItemClickListener(new DrawerItemClickListener());
// enable ActionBar app icon to behave as action to toggle nav drawer
getActionBar().setDisplayHomeAsUpEnabled(true);
getActionBar().setHomeButtonEnabled(true);
// ActionBarDrawerToggle ties together the the proper interactions
// between the sliding drawer and the action bar app icon
mDrawerToggle = new ActionBarDrawerToggle(
this, /* host Activity */
mDrawerLayout, /* DrawerLayout object */
R.drawable.ic_drawer, /* nav drawer image to replace 'Up' caret */
R.string.drawer_open, /* "open drawer" description for accessibility */
R.string.drawer_close /* "close drawer" description for accessibility */
) {
public void onDrawerClosed(View view) {
getActionBar().setTitle(mTitle);
invalidateOptionsMenu(); // creates call to onPrepareOptionsMenu()
}
public void onDrawerOpened(View drawerView) {
getActionBar().setTitle(mDrawerTitle);
invalidateOptionsMenu(); // creates call to onPrepareOptionsMenu()
}
};
mDrawerLayout.setDrawerListener(mDrawerToggle);
if (savedInstanceState == null) {
selectItem(0);
}
}
И код activity_main.xml:
<android.support.v4.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/drawer_layout"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<FrameLayout
android:id="@+id/content_frame"
android:layout_width="match_parent"
android:layout_height="match_parent" />
<ListView
android:id="@+id/dernieres_news"
android:layout_width="240dp"
android:layout_height="match_parent"
android:layout_gravity="start"
android:background="#F3F3F4"
android:choiceMode="singleChoice"
android:divider="#E3E9E3"
android:dividerHeight="1dp" />
Ответы
Ответ 1
Вы сделали бы это так же, как и добавить заголовки в любом другом ListView
, обучив ваш ListAdapter
возвращать строки заголовков, а также строковые строки. На низком уровне это включает в себя следующие методы, такие как getViewTypeCount()
и getItemViewType()
в вашем ListAdapter
, плюс getView()
знать разницу между типами строк. Или используйте существующую высокоуровневую реализацию, например https://github.com/emilsjolander/StickyListHeaders или http://code.google.com/p/android-amazing-listview/, или любую другую найденный при поиске android listview headers
.
Ответ 2
Другие ответы верны.
Я нашел действительно хороший пример настройки представления, чтобы содержать два типа элементов: menu section
и menu item
. Конечно, вы можете изменить его так, как хотите.
Пример также содержит реализацию абстрактного класса активности, который наследует каждое действие с ящиком nav.
http://www.michenux.net/android-navigation-drawer-748.html
Ответ 3
Вам придется подклассировать BaseAdapter и использовать это вместо SimpleAdapter в ListView.
Вы поставляете адаптер с заполненными данными, где дополнительными данными являются заголовки. Названия и сами списки будут членами одного и того же общего класса. Затем в адаптере вы решаете на основе элемента данных, если фактическое представление является заголовком или элементом и соответствующим образом раздувает его.
UPDATE:
Вот хороший пример:
http://w2davids.wordpress.com/android-sectioned-headers-in-listviews/
Это фактически отделяет заголовки от элементов данных и правильно использует convertView, в отличие от решения, которое я использовал до сих пор в своих предыдущих приложениях.