编辑:我的滑动标签碎片ATabs似乎有问题。没有使用ATabs片段,只需使用导航抽屉在片段A1和B之间切换,一切都很好!可以添加列表项。编辑的主要活动:
@Override
public void onNavigationDrawerItemSelected(int position) {
mFragment = null;
switch (position) {
case 0:
mFragment = A1.newInstance(position + 1, this);
break;
case 1:
mFragment = B.newInstance(position + 1);
break;
}
onSectionAttached(position);
FragmentManager fragmentManager = getSupportFragmentManager();
fragmentManager.beginTransaction()
.replace(R.id.container, mFragment)
.commit();
mFragment = null;
}
但当我使用ATabs片段(见下面的MainActivity)并切换到它并返回时,就不可能再添加ListItems了。尽管Logcat显示日志。v(LOG_标签,“添加项”) 每次按add()。
我不知道是什么导致了这个问题。如果ATabs需要是一个活动而不是一个片段,或者我的PagerAdapter导致了这个问题。在我的ATabs课程中,我尝试使用两种不同的PageRadaper:
public static class MyAdapter extends FragmentStatePagerAdapter {
...
}
和
public static class MyAdapter extends FragmentPagerAdapter {
...
}
后者甚至导致整个片段无法正确显示,也无法在选项卡之间切换。
我有一个导航抽屉,显示两个不同的片段ATabs和B。片段ATabs包含两个滑动选项卡。每个选项卡都包含另一个片段A1和A2。到目前为止,A2和B只是带有简单文本视图的空片段,因此基本上相同的A2=B。当应用程序启动时,片段A1有一个带有一些预定义编辑文本项的列表视图。用户可以通过在该特定片段(A1)的“自定义”操作栏内按add来添加其他EditText项。
一般来说,片段A1应该有不同于片段A2和B的动作条项目。这似乎已经起作用了。为此,我添加了setsethaoptions菜单(true)
并从资源中设置所需的菜单。
这已经是我的第一个问题:是否有必要在所有使用默认主活动操作栏的片段中执行此操作?
我的主要问题是在列表视图中添加其他项:
当我在应用程序启动时显示的初始片段A1和片段A2之间切换,并在ATabs片段内返回A1时,一切正常。我可以添加其他项目,以前添加的项目也仍然可见。但是,一旦我使用导航抽屉在片段B和A1之间切换,我就不能再添加任何项目,也看不到以前添加的项目。稍后我将使用数据库来存储我的项目,但仍然可以像以前一样添加其他项目。
这里有一些屏幕可以更好地展示我的问题:
应用程序以A1片段开始,在A1和A2之间切换,可以使用项目添加
在我使用导航抽屉在atab和B之间切换后,显示add item(),但对片段A1没有影响。
下面是我保存导航抽屉的主要活动:我不确定在片段之间的内部切换是否正确,或者每次切换时是否创建新片段。Android是在删除之前的片段,还是在多次切换后导致内存溢出?我的“第一个”A1片段仍然“隐藏在”新创建的片段之后?我对Android非常陌生,我真的不明白为什么在切换导航抽屉片段后不能添加任何项目。
public class MainActivity extends ActionBarActivity
implements NavigationDrawerFragment.NavigationDrawerCallbacks, A1.OnFragmentInteractionListener, A2.OnFragmentInteractionListener, B.OnFragmentInteractionListener {
private NavigationDrawerFragment mNavigationDrawerFragment;
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
Fragment fragment = null;
switch (position) {
case 0:
fragment = ATabs.newInstance(position + 1, this);
break;
case 1:
fragment = B.newInstance(position + 1);
break;
}
onSectionAttached(position);
FragmentManager fragmentManager = getSupportFragmentManager();
fragmentManager.beginTransaction()
.replace(R.id.container, fragment)
.commit();
}
public void onSectionAttached(int number) {
switch (number+1) {
case 1:
mTitle = getString(R.string.title_ATabs);
break;
case 2:
mTitle = getString(R.string.title_B);
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);
}
@Override
public void onFragmentInteraction(Uri uri) {
}
}
带有滑动选项卡的片段ATab:
public class ATabs extends Fragment {
static final String LOG_TAG = "ATabs";
private static final String ARG_PARAM1 = "param1";
private static Context mContext;
private SlidingTabLayout mSlidingTabLayout;
private ViewPager mViewPager;
public static ATabs newInstance(int sectionNumber, Context context) {
ATabs fragment = new ATabs();
mContext = context;
Bundle args = new Bundle();
args.putInt(ARG_PARAM1, sectionNumber);
fragment.setArguments(args);
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setHasOptionsMenu(true);
}
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
menu.clear();
super.onCreateOptionsMenu(menu, inflater);
inflater.inflate(R.menu.main, menu);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
return inflater.inflate(R.layout.fragment_atabs, container, false);
}
@Override
public void onViewCreated(View view, Bundle savedInstanceState) {
mViewPager = (ViewPager) view.findViewById(R.id.viewpager);
mViewPager.setAdapter(new MyAdapter(getFragmentManager()));
mSlidingTabLayout = (SlidingTabLayout) view.findViewById(R.id.sliding_tabs);
mSlidingTabLayout.setViewPager(mViewPager);
}
public static class MyAdapter extends FragmentStatePagerAdapter {
public MyAdapter(FragmentManager fm) {
super(fm);
}
@Override
public int getCount() {
return 2;
}
@Override
public Fragment getItem(int position) {
Fragment fragment = null;
switch (position) {
case 0:
fragment = A1.newInstance(position, mContext);
break;
case 1:
fragment = A2.newInstance(position);
break;
}
return fragment;
}
@Override
public CharSequence getPageTitle(int position) {
CharSequence pageTitle = "NOT SET";
switch (position) {
case 0:
pageTitle = "A1 Tab";
break;
case 1:
pageTitle = "A2 Tab";
break;
}
return pageTitle;
}
}
}
如前所述:是否有必要在使用默认主活动操作栏的片段中调用
setHasOptionsMenu(true)
并覆盖onCreateOptionsMenu()
?
下面是使用默认主活动操作栏的片段A2(与类B相同):
public class A2 extends Fragment {
private static final String ARG_PARAM1 = "param1";
private String mParam1;
private OnFragmentInteractionListener mListener;
public static A2 newInstance(int sectionNumber) {
A2 fragment = new A2();
Bundle args = new Bundle();
args.putInt(ARG_PARAM1, sectionNumber);
fragment.setArguments(args);
return fragment;
}
public A2() {
// Required empty public constructor
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getArguments() != null) {
mParam1 = getArguments().getString(ARG_PARAM1);
}
setHasOptionsMenu(true);
}
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
menu.clear();
super.onCreateOptionsMenu(menu, inflater);
inflater.inflate(R.menu.main, menu);
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
// Inflate the layout for this fragment
return inflater.inflate(R.layout.fragment_a2, container, false);
}
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
try {
mListener = (OnFragmentInteractionListener) activity;
} catch (ClassCastException e) {
throw new ClassCastException(activity.toString()
+ " must implement OnFragmentInteractionListener");
}
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
public interface OnFragmentInteractionListener {
// TODO: Update argument type and name
public void onFragmentInteraction(Uri uri);
}
}
片段A1使用自己的a1操作栏菜单:附加添加项。
public class A1 extends Fragment {
private static final String ARG_PARAM1 = "param1";
private final String LOG_TAG = A1.class.getSimpleName();
private ArrayAdapter<String> mForecastAdapter;
private static Context mContext = null;
private ListView myList;
private MyAdapter myAdapter;
private String mParam1;
private OnFragmentInteractionListener mListener;
public static A1 newInstance(int sectionNumber, Context context) {
A1 fragment = new A1();
mContext = context;
Bundle args = new Bundle();
args.putInt(ARG_PARAM1, sectionNumber);
fragment.setArguments(args);
return fragment;
}
public A1() {
// Required empty public constructor
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (getArguments() != null) {
mParam1 = getArguments().getString(ARG_PARAM1);
}
setHasOptionsMenu(true);
}
@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
super.onCreateOptionsMenu(menu, inflater);
menu.clear();
inflater.inflate(R.menu.a1, menu);
}
public ArrayList myItems = new ArrayList();
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// handle item selection
switch (item.getItemId()) {
case R.id.action_add_item:
ListItem listItem = new ListItem();
listItem.caption = "Added Item";
myItems.add(listItem);
Log.v(LOG_TAG, "Added Item");
myAdapter.notifyDataSetChanged();
return true;
default:
return super.onOptionsItemSelected(item);
}
}
public class MyAdapter extends BaseAdapter {
private LayoutInflater mInflater;
public MyAdapter() {
mInflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
for (int i = 0; i < 3; i++) {
ListItem listItem = new ListItem();
listItem.caption = "Caption" + i;
myItems.add(listItem);
}
notifyDataSetChanged();
}
public int getCount() {
return myItems.size();
}
public Object getItem(int position) {
return position;
}
public long getItemId(int position) {
return position;
}
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
if (convertView == null) {
holder = new ViewHolder();
convertView = mInflater.inflate(R.layout.list_item_a1, null);
holder.caption = (EditText) convertView
.findViewById(R.id.list_item_a1_edit_text);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
//Fill EditText with the value you have in data source
ListItem item = (ListItem) myItems.get(position);
holder.caption.setText(item.caption);
holder.caption.setId(position);
//we need to update adapter once we finish with editing
holder.caption.setOnFocusChangeListener(new View.OnFocusChangeListener() {
public void onFocusChange(View v, boolean hasFocus) {
if (!hasFocus){
final int position = v.getId();
final EditText Caption = (EditText) v;
ListItem item = (ListItem) myItems.get(position);
item.caption = Caption.getText().toString();
}
}
});
return convertView;
}
}
class ViewHolder {
EditText caption;
}
class ListItem {
String caption;
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View rootView = inflater.inflate(R.layout.fragment_a1, container, false);
// Get a reference to the ListView, and attach this adapter to it.
myList = (ListView) rootView.findViewById(R.id.listview_a1);
myList.setItemsCanFocus(true);
myAdapter = new MyAdapter();
myList.setAdapter(myAdapter);
return rootView;
}
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
try {
mListener = (OnFragmentInteractionListener) activity;
} catch (ClassCastException e) {
throw new ClassCastException(activity.toString()
+ " must implement OnFragmentInteractionListener");
}
}
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
public interface OnFragmentInteractionListener {
public void onFragmentInteraction(Uri uri);
}
}
最后是操作栏菜单:main:
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools" tools:context=".MainActivity">
<item
android:id="@+id/action_example"
android:title="@string/action_example"
app:showAsAction="never" />
<item
android:id="@+id/action_settings"
android:title="@string/action_settings"
android:orderInCategory="100"
app:showAsAction="never" />
<item
android:id="@+id/action_help"
android:title="@string/action_help"
android:orderInCategory="100"
app:showAsAction="never" />
</menu>
a1:
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools" tools:context=".MainActivity">
<item
android:id="@+id/action_add_item"
android:title="@string/action_add_item"
android:icon="@drawable/ic_action_new"
app:showAsAction="always" />
<item
android:id="@+id/action_example"
android:title="@string/action_example"
app:showAsAction="never" />
<item
android:id="@+id/action_settings"
android:title="@string/action_settings"
android:orderInCategory="100"
app:showAsAction="never" />
<item
android:id="@+id/action_help"
android:title="@string/action_help"
android:orderInCategory="100"
app:showAsAction="never" />
</menu>
如果需要任何其他代码来解决我的问题,请告诉我。
感谢@Luksprog,主要问题通过在ATabs中使用getChildFragmentManager()
而不是FragmentManager()
来解决。
我有一个带有导航抽屉的应用程序,它为每个菜单项使用片段。 每次单击一个项目时,我都会替换当前片段。 问题是,在用户单击后显示新片段需要很长时间,并且它会被卡住。加载时间最长的片段是一个片段,其中还包含子片段的选项卡(一个片段包含一个RecyclerView)<有什么方法可以加快碎片的装载速度吗?(如果可能的话,可能会提前初始化它们?) 这是我的代码: HalachaFragment。cs(包含选项
我希望能够使用导航抽屉,并根据导航中的选择在不同片段之间进行更改。 我正在使用Android Studio,基本上我所做的是这样的: 使用内置模板“导航抽屉活动”创建了一个新项目 创建了一个空白片段 然后我更改了onNavigationDrawerItemSelected方法中的一些代码。 我的程序崩溃了,给了我很多我不理解的错误。我做错了什么?
我在我的应用程序中使用抽屉菜单,并实现了在NavigationDrawer片段类中切换片段的逻辑。我最近读到片段切换只能从托管活动中发生。 有一个从NavigationDrawerFragment调用的接口,用于通知导航抽屉列表视图中所选位置的MainActivity。 我感到困惑的是,在Mainactive中有一个静态片段,它是使用NavigationDrawer片段中接口提供的位置调用的。 P
我不是Android方面的专家,但对片段非常感兴趣。非常感谢您的帮助! 我在我的应用程序中实现了一个导航抽屉。有一个基本抽屉活动和一些片段,用户可以从抽屉菜单中切换。如果我想转移到另一个片段,没有问题,但当我使用意图开始一个新的活动时,它是有效的。如果你看一下我下面的代码,你会发现我使用了两个独立的菜单:一个用于操作栏图标(购物车和搜索),意图在其中完美工作,另一个用于导航抽屉在片段之间跳转。有没
我有一个主活动启动导航抽屉。 也有一些片段。 我需要从片段内部更改导航抽屉列表的标题。 我尝试了这个内部片段: 但是到目前为止还没有运气。这是我的主要活动: 在主活动类中使用相同的方法有效,但在片段中无效。我不知道如何更新字符串并通知抽屉此更改!
1) 我遵循了Android开发者文档中的导航抽屉示例。Androidcom/training/implementing navigation/nav drawer。并创建了我的整个应用程序。在给定的示例中,他们为抽屉中选择的每个项目使用了片段,称为片段,代码如下 2)现在我想要一个片段内部的标签行为,即,当我在导航抽屉中选择一个特定的项目时,加载的片段应该在顶部显示一个标签栏,就像这样。http