1

Я относительно новичок в разработке Android, поэтому я думаю, что, возможно, это такая основная вещь, что это причина, по которой я не могу найти и ответить нигде.Навигация на ящике - изменение списка элементов после создания

Что у меня есть:

  • A Главной активность, которая начинается, когда приложение запускается, что я называю DrawerMainActivity. Эта активность является шаблоном, созданным Android Studio под названием «Активность навигационного ящика».

Что я хочу:

  • Запустить активность и создать пустой ящик без вариантов, пока я получаю информацию о пользователях и, после этого, изменить ящик для отображения списка опций, что нужно или
  • Запустите действие и задержите создание ящика до тех пор, пока я не получу необходимую информацию учетной записи пользователя.

Как я могу изменить список параметров внутри ящика во время выполнения после его создания?

DrawerMainActivity.java Layout (шаблон навигации Выдвижные активность из AndroidStudion):

<android.support.v4.widget.DrawerLayout xmlns:android="http://schemas.android.com/apk/res/android" 
xmlns:tools="http://schemas.android.com/tools" android:id="@+id/drawer_layout" 
android:layout_width="match_parent" android:layout_height="match_parent" 
tools:context="something_bla_bla"> 

<FrameLayout android:id="@+id/container" android:layout_width="match_parent" 
    android:layout_height="match_parent" /> 

<fragment android:id="@+id/navigation_drawer" 
    android:layout_width="@dimen/navigation_drawer_width" 
    android:layout_height="match_parent" 
    android:layout_gravity="start" 
    android:name="bla.bla.bla.NavigationDrawerFragment" 
    tools:layout="@layout/fragment_navigation_drawer" /> 

DrawerMainActivity (созданный AndroidStudio):

public class DrawerMainActivity extends ActionBarActivity 
    implements NavigationDrawerFragment.NavigationDrawerCallbacks { 

/** 
* Fragment managing the behaviors, interactions and presentation of the navigation drawer. 
*/ 
private NavigationDrawerFragment mNavigationDrawerFragment; 

/** 
* Used to store the last screen title. For use in {@link #restoreActionBar()}. 
*/ 
private CharSequence mTitle; 

@Override 
protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    setContentView(R.layout.activity_main); 

    mNavigationDrawerFragment = (NavigationDrawerFragment) 
      getSupportFragmentManager().findFragmentById(R.id.navigation_drawer); 
    mTitle = getTitle(); 

    // Set up the drawer. 
    mNavigationDrawerFragment.setUp(
      R.id.navigation_drawer, 
      (DrawerLayout) findViewById(R.id.drawer_layout)); 
} 

@Override 
public void onNavigationDrawerItemSelected(int position) { 
    // update the main content by replacing fragments 
    FragmentManager fragmentManager = getSupportFragmentManager(); 
    fragmentManager.beginTransaction() 
      .replace(R.id.container, PlaceholderFragment.newInstance(position + 1)) 
      .commit(); 
} 

public void onSectionAttached(int number) { 
    switch (number) { 
     case 1: 
      mTitle = getString(R.string.title_section1); 
      break; 
     case 2: 
      mTitle = getString(R.string.title_section2); 
      break; 
     case 3: 
      mTitle = getString(R.string.title_section3); 
      break; 
    } 
} 

public void restoreActionBar() { 
    ActionBar actionBar = getSupportActionBar(); 
    actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_STANDARD); 
    actionBar.setDisplayShowTitleEnabled(true); 
    actionBar.setTitle(mTitle); 
} 


@Override 
public boolean onCreateOptionsMenu(Menu menu) { 
    if (!mNavigationDrawerFragment.isDrawerOpen()) { 
     // Only show items in the action bar relevant to this screen 
     // if the drawer is not showing. Otherwise, let the drawer 
     // decide what to show in the action bar. 
     getMenuInflater().inflate(R.menu.main, menu); 
     restoreActionBar(); 
     return true; 
    } 
    return super.onCreateOptionsMenu(menu); 
} 

@Override 
public boolean onOptionsItemSelected(MenuItem item) { 
    // Handle action bar item clicks here. The action bar will 
    // automatically handle clicks on the Home/Up button, so long 
    // as you specify a parent activity in AndroidManifest.xml. 
    int id = item.getItemId(); 

    //noinspection SimplifiableIfStatement 
    if (id == R.id.action_settings) { 
     return true; 
    } 

    return super.onOptionsItemSelected(item); 
} 

/** 
* A placeholder fragment containing a simple view. 
*/ 
public static class PlaceholderFragment extends Fragment { 
    /** 
    * The fragment argument representing the section number for this 
    * fragment. 
    */ 
    private static final String ARG_SECTION_NUMBER = "section_number"; 

    /** 
    * Returns a new instance of this fragment for the given section 
    * number. 
    */ 
    public static PlaceholderFragment newInstance(int sectionNumber) { 
     PlaceholderFragment fragment = new PlaceholderFragment(); 
     Bundle args = new Bundle(); 
     args.putInt(ARG_SECTION_NUMBER, sectionNumber); 
     fragment.setArguments(args); 
     return fragment; 
    } 

    public PlaceholderFragment() { 
    } 

    @Override 
    public View onCreateView(LayoutInflater inflater, ViewGroup container, 
          Bundle savedInstanceState) { 
     View rootView = inflater.inflate(R.layout.fragment_main, container, false); 
     return rootView; 
    } 

    @Override 
    public void onAttach(Activity activity) { 
     super.onAttach(activity); 
     ((MainActivity) activity).onSectionAttached(
       getArguments().getInt(ARG_SECTION_NUMBER)); 
    } 
} 

}

FragmentLayout.xml (созданный AndroidStudio):

<ListView xmlns:android="http://schemas.android.com/apk/res/android" 
xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" 
android:layout_height="match_parent" android:choiceMode="singleChoice" 
android:divider="@android:color/transparent" android:dividerHeight="0dp" 
android:background="#cccc" 
tools:context="bla.bla.bla.Fragments.NavigationDrawerFragment" /> 

NavigationDrawerFragment.java класс (созданный AndroidStudio):

public class NavigationDrawerFragment extends Fragment { 

/** 
* Remember the position of the selected item. 
*/ 
private static final String STATE_SELECTED_POSITION = "selected_navigation_drawer_position"; 

/** 
* Per the design guidelines, you should show the drawer on launch until the user manually 
* expands it. This shared preference tracks this. 
*/ 
private static final String PREF_USER_LEARNED_DRAWER = "navigation_drawer_learned"; 

/** 
* A pointer to the current callbacks instance (the Activity). 
*/ 
private NavigationDrawerCallbacks mCallbacks; 

/** 
* Helper component that ties the action bar to the navigation drawer. 
*/ 
private ActionBarDrawerToggle mDrawerToggle; 

private DrawerLayout mDrawerLayout; 
private ListView mDrawerListView; 
private View mFragmentContainerView; 

private int mCurrentSelectedPosition = 0; 
private boolean mFromSavedInstanceState; 
private boolean mUserLearnedDrawer; 


public NavigationDrawerFragment() { 
} 

@Override 
public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 

    // Read in the flag indicating whether or not the user has demonstrated awareness of the 
    // drawer. See PREF_USER_LEARNED_DRAWER for details. 
    SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(getActivity()); 
    mUserLearnedDrawer = sp.getBoolean(PREF_USER_LEARNED_DRAWER, false); 

    if (savedInstanceState != null) { 
     mCurrentSelectedPosition = savedInstanceState.getInt(STATE_SELECTED_POSITION); 
     mFromSavedInstanceState = true; 
    } 

    // Select either the default item (0) or the last selected item. 
    selectItem(mCurrentSelectedPosition); 
} 

@Override 
public void onActivityCreated(Bundle savedInstanceState) { 
    super.onActivityCreated(savedInstanceState); 
    // Indicate that this fragment would like to influence the set of actions in the action bar. 
    setHasOptionsMenu(true); 
} 

@Override 
public View onCreateView(LayoutInflater inflater, ViewGroup container, 
         Bundle savedInstanceState) { 


     mDrawerListView = (ListView) inflater.inflate(
      R.layout.fragment_navigation_drawer, container, false); 
     mDrawerListView.setOnItemClickListener(new AdapterView.OnItemClickListener() { 
      @Override 
      public void onItemClick(AdapterView<?> parent, View view, int position, long id) { 
       selectItem(position); 
      } 
     }); 
     mDrawerListView.setAdapter(new ArrayAdapter<String>(
       getActionBar().getThemedContext(), 
       android.R.layout.simple_list_item_1, 
       android.R.id.text1, 
       new String[]{ 
         getString(R.string.title_section1), 
         getString(R.string.title_section2), 
         getString(R.string.title_section3), 
       })); 
     mDrawerListView.setItemChecked(mCurrentSelectedPosition, true); 
     return mDrawerListView; 

} 

public boolean isDrawerOpen() { 
    return mDrawerLayout != null && mDrawerLayout.isDrawerOpen(mFragmentContainerView); 
} 

/** 
* Users of this fragment must call this method to set up the navigation drawer interactions. 
* 
* @param fragmentId The android:id of this fragment in its activity's layout. 
* @param drawerLayout The DrawerLayout containing this fragment's UI. 
*/ 
public void setUp(int fragmentId, DrawerLayout drawerLayout) 
{ 
    mFragmentContainerView = getActivity().findViewById(fragmentId); 
    mDrawerLayout = drawerLayout; 

    // 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 

    ActionBar actionBar = getActionBar(); 
    actionBar.setDisplayHomeAsUpEnabled(true); 
    actionBar.setHomeButtonEnabled(true); 

    // ActionBarDrawerToggle ties together the the proper interactions 
    // between the navigation drawer and the action bar app icon. 
    mDrawerToggle = new ActionBarDrawerToggle(
      getActivity(),     /* host Activity */ 
      mDrawerLayout,     /* DrawerLayout object */ 
      R.drawable.ic_drawer,    /* nav drawer image to replace 'Up' caret */ 
      R.string.navigation_drawer_open, /* "open drawer" description for accessibility */ 
      R.string.navigation_drawer_close /* "close drawer" description for accessibility */ 
    ) { 
     @Override 
     public void onDrawerClosed(View drawerView) { 
      super.onDrawerClosed(drawerView); 
      if (!isAdded()) { 
       return; 
      } 

      getActivity().supportInvalidateOptionsMenu(); // calls onPrepareOptionsMenu() 
     } 

     @Override 
     public void onDrawerOpened(View drawerView) { 
      super.onDrawerOpened(drawerView); 
      if (!isAdded()) { 
       return; 
      } 

      if (!mUserLearnedDrawer) { 
       // The user manually opened the drawer; store this flag to prevent auto-showing 
       // the navigation drawer automatically in the future. 
       mUserLearnedDrawer = true; 
       SharedPreferences sp = PreferenceManager 
         .getDefaultSharedPreferences(getActivity()); 
       sp.edit().putBoolean(PREF_USER_LEARNED_DRAWER, true).commit(); 
      } 

      getActivity().supportInvalidateOptionsMenu(); // calls onPrepareOptionsMenu() 
     } 
    }; 



    // If the user hasn't 'learned' about the drawer, open it to introduce them to the drawer, 
    // per the navigation drawer design guidelines. 
    if (!mUserLearnedDrawer && !mFromSavedInstanceState) { 
     mDrawerLayout.openDrawer(mFragmentContainerView); 
    } 

    // Defer code dependent on restoration of previous instance state. 
    mDrawerLayout.post(new Runnable() { 
     @Override 
     public void run() { 
      mDrawerToggle.syncState(); 
     } 
    }); 

    mDrawerLayout.setDrawerListener(mDrawerToggle); 
} 

private void selectItem(int position) { 
    mCurrentSelectedPosition = position; 
    if (mDrawerListView != null) { 
     mDrawerListView.setItemChecked(position, true); 
    } 
    if (mDrawerLayout != null) { 
     mDrawerLayout.closeDrawer(mFragmentContainerView); 
    } 
    if (mCallbacks != null) { 
     mCallbacks.onNavigationDrawerItemSelected(position); 
    } 
} 

@Override 
public void onAttach(Activity activity) { 
    super.onAttach(activity); 
    try { 
     mCallbacks = (NavigationDrawerCallbacks) activity; 
    } catch (ClassCastException e) { 
     throw new ClassCastException("Activity must implement NavigationDrawerCallbacks."); 
    } 
} 

@Override 
public void onDetach() { 
    super.onDetach(); 
    mCallbacks = null; 
} 

@Override 
public void onSaveInstanceState(Bundle outState) { 
    super.onSaveInstanceState(outState); 
    outState.putInt(STATE_SELECTED_POSITION, mCurrentSelectedPosition); 
} 

@Override 
public void onConfigurationChanged(Configuration newConfig) { 
    super.onConfigurationChanged(newConfig); 
    // Forward the new configuration the drawer toggle component. 
    mDrawerToggle.onConfigurationChanged(newConfig); 
} 

@Override 
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { 
    // If the drawer is open, show the global app actions in the action bar. See also 
    // showGlobalContextActionBar, which controls the top-left area of the action bar. 
    if (mDrawerLayout != null && isDrawerOpen()) { 
     inflater.inflate(R.menu.global, menu); 
     showGlobalContextActionBar(); 
    } 
    super.onCreateOptionsMenu(menu, inflater); 
} 

@Override 
public boolean onOptionsItemSelected(MenuItem item) { 
    if (mDrawerToggle.onOptionsItemSelected(item)) { 
     return true; 
    } 

    /* 
    if (item.getItemId() == R.id.action_example) { 
     Toast.makeText(getActivity(), "Example action.", Toast.LENGTH_SHORT).show(); 
     return true; 
    }*/ 

    return super.onOptionsItemSelected(item); 
} 

/** 
* Per the navigation drawer design guidelines, updates the action bar to show the global app 
* 'context', rather than just what's in the current screen. 
*/ 
private void showGlobalContextActionBar() { 
    ActionBar actionBar = getActionBar(); 
    actionBar.setDisplayShowTitleEnabled(true); 
    actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_STANDARD); 
    actionBar.setTitle(R.string.app_name); 
} 

private ActionBar getActionBar() { 
    return ((ActionBarActivity) getActivity()).getSupportActionBar(); 
} 

/** 
* Callbacks interface that all activities using this fragment must implement. 
*/ 
public static interface NavigationDrawerCallbacks { 
    /** 
    * Called when an item in the navigation drawer is selected. 
    */ 
    void onNavigationDrawerItemSelected(int position); 
} 

}

Первоначальный список создается в onCreatedView(...) в NavigationDrawerFragment. Я хочу изменить этот список после его первого создания из основного действия. Я был вокруг этой проблемы в течение нескольких дней уже, головная боль реальна ...

+0

Внесите вызов drawerlistner и isOpenDrawer при открытии ящика навигации. Попробуй. –

+0

Не могу понять, как это поможет мне изменить список уже созданных параметров. Упорядочиваться, пожалуйста, пожалуйста? – Meh

+0

@Meh вы нашли решение? У меня такой же сценарий, и я не могу понять. пожалуйста помогите. –

ответ

0

Хорошо. Я выйду наружу и ударился головой о стену. Похоже, что просто изменив объект ListView, указанный в файле NavigationDrawerFragment.java - mDrawerListView, после его создания он также изменит текущий список.

Пример решения

DrawerMainActivity.java:

private void gotUserData(User user) 
{ 
    mNavigationDrawerFragment.changeDrawerOptions(user); 
} 

NavigationDrawerFragment.Java:

public void changeDrawerOptions(User user) 
{ 
    mDrawerListView.setAdapter(new ArrayAdapter<String>(
      getActionBar().getThemedContext(), 
      android.R.layout.simple_list_item_1, 
      android.R.id.text1, 
      new String[]{ 
        "New", 
        "List", 
        "Options", 
      })); 
} 
0

следовать этому коду,

public class activity extends Activity implements DrawerListener { 

private DrawerLayout drawer; 

@Override 
public void onCreate(Bundle savedInstanceState) { 
    // TODO Auto-generated method stub 
    super.onCreate(savedInstanceState); 
    final boolean headrTitle=requestWindowFeature(Window.FEATURE_CUSTOM_TITLE); 
    setContentView(R.layout.activity_home); 
    drawer = (DrawerLayout)findViewById(R.id.drawer_layout); 
    drawer.setDrawerListener(this); 
} //oncreate over 

@Override 
public void onDrawerClosed(View arg0) { 
    // TODO Auto-generated method stub 

} 

@Override 
public void onDrawerOpened(View arg0) { 
    // TODO Auto-generated method stub 
    //in this method you can change view, text, what you want. 
} 

@Override 
public void onDrawerSlide(View arg0, float arg1) { 
    // TODO Auto-generated method stub 

} 

@Override 
public void onDrawerStateChanged(int arg0) { 
    // TODO Auto-generated method stub 

} 
+0

Из того, что я понимаю из этого, даже если я могу изменить Fragment ListView оттуда (я не думаю, что так возможно), это происходит только тогда, когда ящик открыт. Поведение, которое я хочу, это следующее: DrawerMainActivity lanched -> Фрагмент ящика, созданный с помощью списка empy -> Индикатор выполнения показывает, когда данные учетной записи пользователя извлекаются -> Список списков ящиков списков изменен для повторного выбора пользовательских параметров -> ProgressBar закрывается -> Пользователь может перемещаться. – Meh

0

Это может быть из-за мой проект, для более низкой версии API, но по умолчанию ящик Android студии является NavigationView и не является фрагментом.
Он имеет идентификатор «nav_view» по умолчанию, который можно использовать для получения ссылки на него.
Из моего опыта Фрагменты являются источником многих головных уборов, и я бы рекомендовал переключиться на виды, где это возможно.

После того, как у вас есть ссылка на NavigationView заголовок и меню могут быть изменены следующим образом:

//this function is in a class all my Activities inherit from 
private void resetDrawer(){ 
    //myDrawer is a NavigationView initialized in activity onCreate 
    if(myDrawer == null) 
     return; 
    //make a header view. 
    View header = getDrawerHeaderView();//abstract function that all my activities implement 
    if(header!=null){ 
     for(int i = myDrawer.getHeaderCount() - 1; i >= 0 ;i--){ 
      myDrawer.removeHeaderView(myDrawer.getHeaderView(i)); 
     } 
     myDrawer.addHeaderView(header); 
    } 

    //init the menu 
    Menu drawerItems = myDrawer.getMenu(); 
    MenuMaker[] drawerMenus = getDrawerMenu();//another abstract function 
    if(drawerMenus!=null) { 

     drawerItems.clear(); 

     for (MenuMaker menuMaker : drawerMenus) { 
      menuMaker.init(drawerItems); 
     } 
    } 
} 

Это, как я сделать динамический ящик, который обновляется при входе пользователя в/из.
Я загружаю все данные за один раз, вы можете изменить это или просто получить ссылку на меню NavigationView с помощью getMenu() и добавить к нему элементы при их загрузке.

Смежные вопросы