60

Ho un cassetto di navigazione come questa immagine. Voglio aggiungere un separatore di sezione (come la linea che separa Nettuno). Sembra semplice ma non riesco a trovare nulla sul web che è stato utile per il mio caso.Come aggiungere un separatore di sezione per Navigation Drawer in Android?

Ecco il mio 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); 
     mDrawerList = (ListView) findViewById(R.id.left_drawer); 

     // set a custom shadow that overlays the main content when the drawer opens 
     mDrawerLayout.setDrawerShadow(R.drawable.drawer_shadow, GravityCompat.START); 
     // set up the drawer's list view with items and click listener 
     mDrawerList.setAdapter(new ArrayAdapter<String>(this, 
       R.layout.drawer_list_item, mPlanetTitles)); 
     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); 
     } 
    } 

    @Override 
    public boolean onCreateOptionsMenu(Menu menu) { 
     MenuInflater inflater = getMenuInflater(); 
     inflater.inflate(R.menu.main, menu); 
     return super.onCreateOptionsMenu(menu); 
    } 

    /* Called whenever we call invalidateOptionsMenu() */ 
    @Override 
    public boolean onPrepareOptionsMenu(Menu menu) { 
     // If the nav drawer is open, hide action items related to the content view 
     boolean drawerOpen = mDrawerLayout.isDrawerOpen(mDrawerList); 
     menu.findItem(R.id.action_websearch).setVisible(!drawerOpen); 
     return super.onPrepareOptionsMenu(menu); 
    } 

    @Override 
    public boolean onOptionsItemSelected(MenuItem item) { 
     // The action bar home/up action should open or close the drawer. 
     // ActionBarDrawerToggle will take care of this. 
     if (mDrawerToggle.onOptionsItemSelected(item)) { 
      return true; 
     } 
     // Handle action buttons 
     switch(item.getItemId()) { 
     case R.id.action_websearch: 
      // create intent to perform web search for this planet 
      Intent intent = new Intent(Intent.ACTION_WEB_SEARCH); 
      intent.putExtra(SearchManager.QUERY, getActionBar().getTitle()); 
      // catch event that there's no activity to handle intent 
      if (intent.resolveActivity(getPackageManager()) != null) { 
       startActivity(intent); 
      } else { 
       Toast.makeText(this, R.string.app_not_available, Toast.LENGTH_LONG).show(); 
      } 
      return true; 
     default: 
      return super.onOptionsItemSelected(item); 
     } 
    } 

    /* The click listner for ListView in the navigation drawer */ 
    private class DrawerItemClickListener implements ListView.OnItemClickListener { 
     @Override 
     public void onItemClick(AdapterView<?> parent, View view, int position, long id) { 
      selectItem(position); 
     } 
    } 

    private void selectItem(int position) { 
     // update the main content by replacing fragments 
     Fragment fragment = new PlanetFragment(); 
     Bundle args = new Bundle(); 
     args.putInt(PlanetFragment.ARG_PLANET_NUMBER, position); 
     fragment.setArguments(args); 

     FragmentManager fragmentManager = getFragmentManager(); 
     fragmentManager.beginTransaction().replace(R.id.content_frame, fragment).commit(); 

     // update selected item and title, then close the drawer 
     mDrawerList.setItemChecked(position, true); 
     setTitle(mPlanetTitles[position]); 
     mDrawerLayout.closeDrawer(mDrawerList); 
    } 

    @Override 
    public void setTitle(CharSequence title) { 
     mTitle = title; 
     getActionBar().setTitle(mTitle); 
    } 

    /** 
    * When using the ActionBarDrawerToggle, you must call it during 
    * onPostCreate() and onConfigurationChanged()... 
    */ 

    @Override 
    protected void onPostCreate(Bundle savedInstanceState) { 
     super.onPostCreate(savedInstanceState); 
     // Sync the toggle state after onRestoreInstanceState has occurred. 
     mDrawerToggle.syncState(); 
    } 

    @Override 
    public void onConfigurationChanged(Configuration newConfig) { 
     super.onConfigurationChanged(newConfig); 
     // Pass any configuration change to the drawer toggls 
     mDrawerToggle.onConfigurationChanged(newConfig); 
    } 

    /** 
    * Fragment that appears in the "content_frame", shows a planet 
    */ 
    public static class PlanetFragment extends Fragment { 
     public static final String ARG_PLANET_NUMBER = "planet_number"; 

     public PlanetFragment() { 
      // Empty constructor required for fragment subclasses 
     } 

     @Override 
     public View onCreateView(LayoutInflater inflater, ViewGroup container, 
       Bundle savedInstanceState) { 
      View rootView = inflater.inflate(R.layout.fragment_planet, container, false); 
      int i = getArguments().getInt(ARG_PLANET_NUMBER); 
      String planet = getResources().getStringArray(R.array.planets_array)[i]; 

      int imageId = getResources().getIdentifier(planet.toLowerCase(Locale.getDefault()), 
          "drawable", getActivity().getPackageName()); 
      ((ImageView) rootView.findViewById(R.id.image)).setImageResource(imageId); 
      getActivity().setTitle(planet); 
      return rootView; 
     } 
    } 
} 

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/left_drawer" 
     android:layout_width="240dp" 
     android:layout_height="match_parent" 
     android:layout_gravity="start" 
     android:choiceMode="singleChoice" 
     android:divider="@android:color/transparent" 
     android:dividerHeight="0dp" 
     android:background="#111"/> 
</android.support.v4.widget.DrawerLayout> 

drawer_list_item.xml:

<TextView xmlns:android="http://schemas.android.com/apk/res/android" 
    android:id="@android:id/text1" 
    android:layout_width="match_parent" 
    android:layout_height="wrap_content" 
    android:textAppearance="?android:attr/textAppearanceListItemSmall" 
    android:gravity="center_vertical" 
    android:paddingLeft="16dp" 
    android:paddingRight="16dp" 
    android:textColor="#fff" 
    android:background="?android:attr/activatedBackgroundIndicator" 
    android:minHeight="?android:attr/listPreferredItemHeightSmall"/> 

io voglio che questo sia semplice, ma non riesco a trovare niente di buono sul web. C'è un modo per isolare Nettuno e creare una sezione per questo? Qualcuno ha qualche suggerimento? Grazie.

enter image description here

+0

non sarete in grado di farlo in quel modo con un adattatore standard. sarà necessario creare un adattatore personalizzato e avere qualche accesso semplice su getView per nascondere o mostrare il separatore –

+0

@TomerShemesh Hai un esempio di come posso farlo? – Alias

+0

ci sono librerie su github che ti aiuteranno a raggiungere lo stesso risultato. MaterialDrawer di mikepenz è buono e viene mantenuto bene. https://github.com/mikepenz/MaterialDrawer. Puoi anche seguire l'app di programmazione di google io su github. Il separatore è solo una vista con altezza 1px e larghezza match_parent – Raghunandan

risposta

225

Assicurarsi di definire diversi grups con ID diversi.

Ad esempio, questo è il mio drawer_menu.xml:

<?xml version="1.0" encoding="utf-8"?> 
<menu xmlns:android="http://schemas.android.com/apk/res/android"> 
    <group 
     android:id="@+id/menu_top" 
     android:checkableBehavior="single"> 
     <item 
      android:checked="true" 
      android:id="@+id/drawer_item_timeline" 
      android:icon="@drawable/ic_timer_grey600_24dp" 
      android:title="@string/drawer_timeline"/> 
     <item 
      android:id="@+id/drawer_item_reports" 
      android:icon="@drawable/ic_timetable_grey600_24dp" 
      android:title="@string/drawer_reports"/> 
    </group> 

    <group 
     android:id="@+id/menu_bottom" 
     android:checkableBehavior="none"> 

     <item 
      android:id="@+id/drawer_item_settings" 
      android:icon="@drawable/ic_settings_black_24dp" 
      android:title="@string/drawer_settings" > 
     </item> 
    </group> 
</menu> 

Sample drawer

Gabriel aggiunge sotto nei commenti che se il gruppo non ha un ID, non apparirà il separatore.

+2

Grazie. Finalmente una buona soluzione! +1 e accettato – Alias

+0

ama la nuova vista di navigazione - è così facile cambiare l'interfaccia utente – Ash

+77

Importante per dire che se il gruppo non ha un ID, il separatore non apparirà! –

0

si hanno due scelte

  1. I suoi elementi possono essere separati (un elenco nella parte superiore, e vista classici in basso). Quindi anziché il listview nel tuo layout principale (android: id = "@ + id/left_drawer") puoi avere un LinearLayout piuttosto complesso che include quei 3 elementi (elenco, separatore e viste in basso)
  2. I tuoi articoli devono essere esattamente come nel tuo esempio, allora hai bisogno del separatore nella lista, puoi usare un po 'di logica nell'adattatore per disegnare una vista sopra l'elemento della lista dove hai bisogno del separatore. (significa che la tua voce di elenco non sarà più una singola finestra di testo, ma a volte visibile secondo la logica della tua scheda)

Per aiutarti con qualche codice di esempio, puoi per favore ? inviare tutti gli elementi necessari nel vostro menu Dobbiamo sapere esattamente quale sarà statica e quale sarà scorrevole

Edit:. Se si vuole che il lavoro con l'exemple, sbarazzarsi della linea

mDrawerList.setAdapter(new ArrayAdapter<String>(this, ...) 

È necessario fornire una casa pazza e l'adattatore in questo modo: https://github.com/codepath/android_guides/wiki/Using-an-ArrayAdapter-with-ListView

Come ho detto in 2, l'adattatore, si avrà la logica, e quindi si può dire nel metodo GetView()

if(myPlanet.isNeptune()) 
    holder.mSepatator.setVisibility(View.VISIBLE); 
else 
    holder.mSepatator.setVisibility(View.GONE); 
+0

Questa è la parte principale del codice che sto usando. Il resto sono tutti file xml come string.xml ... – Alias

+0

Sì, ma questo è il campione Android per il layout del cassetto, cosa volete avere? – Mostrapotski

0

Il mio metodo hacky è simile a Mostrapotski.

Nel mio layout per il mio adattatore personalizzato, sto aggiungendo un separatore orizzontale all'inizio di ogni elemento e impostando la sua visibilità è sparita.

Per gli elementi che segnano l'inizio di un nuovo gruppo, ho impostato la visibilità su visibile in modo che il separatore venga visualizzato sopra.