- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我对 Android 相当陌生,我参加了 coursera 网站上的类(class)。我有一个应用程序需要在平板电脑上以两 Pane 模式运行,在普通设备上以单 Pane 模式运行。作为奖励练习之一,我需要在设备方向更改时保持两 Pane 模式的状态(记住在左侧 ListFragment
中选择了哪个项目以及在右侧 Fragment
中显示了哪些文本)。
我最终做了这个练习,但是我注意到应用程序没有正确销毁 ListFragment - 查看日志。应用程序运行后日志输出被清除,这只是由于方向改变。
正如你所看到的,fragment 被破坏了,但随后 android 又不知从哪里附加了它。
我只包含了单 Pane 代码和布局的代码 - 问题就在那里。两 Pane 布局工作正常。
02-10 07:19:20.845: I/DEBUG(1297): ListFragment onPause()
02-10 07:19:20.845: I/DEBUG(1297): Activity onPause()
02-10 07:19:20.855: I/DEBUG(1297): ListFragment onStop()
02-10 07:19:20.855: I/DEBUG(1297): Activity onStop()
02-10 07:19:20.855: I/DEBUG(1297): ListFragment onDestroyView()
02-10 07:19:20.875: I/DEBUG(1297): ListFragment onDestroy()
02-10 07:19:20.875: I/DEBUG(1297): ListFragment onDetach()
02-10 07:19:20.905: I/DEBUG(1297): Activity onDestroy()
02-10 07:19:21.115: I/DEBUG(1297): ListFragment onAttach()
02-10 07:19:21.115: I/DEBUG(1297): ListFragment onCreate()
02-10 07:19:21.115: I/DEBUG(1297): Activity onCreate()
02-10 07:19:21.315: I/DEBUG(1297): ListFragment onCreateView()
02-10 07:19:21.455: I/DEBUG(1297): ListFragment onActivityCreated()
02-10 07:19:21.466: I/DEBUG(1297): ListFragment onAttach()
02-10 07:19:21.466: I/DEBUG(1297): ListFragment onCreate()
02-10 07:19:21.466: I/DEBUG(1297): ListFragment onCreateView()
02-10 07:19:21.535: I/DEBUG(1297): ListFragment onActivityCreated()
02-10 07:19:21.535: I/DEBUG(1297): Activity onStart()
02-10 07:19:21.535: I/DEBUG(1297): ListFragment onStart()
02-10 07:19:21.535: I/DEBUG(1297): ListFragment onStart()
02-10 07:19:21.555: I/DEBUG(1297): Activity onResum()
02-10 07:19:21.555: I/DEBUG(1297): ListFragment onResume()
02-10 07:19:21.555: I/DEBUG(1297): ListFragment onResume()
MainActivity.java
public class MainActivity extends Activity implements
FriendsFragment.SelectionListener {
private static final String TAG = "Lab-Fragments";
private static final String DEBUG = "DEBUG";
private static final String STRING_DEBUG = "Activity ";
private FriendsFragment mFriendsFragment;
private FeedFragment mFeedFragment;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.i(DEBUG, STRING_DEBUG + "onCreate()");
setContentView(R.layout.main_activity);
// If the layout is single-pane, create the FriendsFragment
// and add it to the Activity
if (!isInTwoPaneMode()) {
if (mFriendsFragment == null) {//TO REMOVE
mFriendsFragment = new FriendsFragment();
}
//TODO 1 - add the FriendsFragment to the fragment_container
FragmentTransaction transaction = getFragmentManager().beginTransaction();
transaction.add(R.id.fragment_container, mFriendsFragment);
transaction.commit();
} else {
// Otherwise, save a reference to the FeedFragment for later use
mFeedFragment = (FeedFragment) getFragmentManager().findFragmentById(R.id.feed_frag);
}
}
// If there is no fragment_container ID, then the application is in
// two-pane mode
private boolean isInTwoPaneMode() {
return findViewById(R.id.fragment_container) == null;
}
// Display selected Twitter feed
public void onItemSelected(int position) {
Log.i(TAG, "Entered onItemSelected(" + position + ")");
// If there is no FeedFragment instance, then create one
if (mFeedFragment == null)
mFeedFragment = new FeedFragment();
// If in single-pane mode, replace single visible Fragment
if (!isInTwoPaneMode()) {
//TODO 2 - replace the fragment_container with the FeedFragment
FragmentTransaction transaction = getFragmentManager().beginTransaction();
transaction.replace(R.id.fragment_container, mFeedFragment);
transaction.addToBackStack(null);
transaction.commit();
// execute transaction now
getFragmentManager().executePendingTransactions();
}
// Update Twitter feed display on FriendFragment
mFeedFragment.updateFeedDisplay(position);
}
@Override
public void onStart() {
super.onStart();
Log.i(DEBUG, STRING_DEBUG + "onStart()");
}
@Override
public void onResume() {
super.onResume();
Log.i(DEBUG, STRING_DEBUG + "onResum()");
}
@Override
public void onPause() {
super.onPause();
Log.i(DEBUG, STRING_DEBUG + "onPause()");
}
@Override
public void onStop() {
super.onStop();
Log.i(DEBUG, STRING_DEBUG + "onStop()");
}
public void onRestart() {
super.onRestart();
Log.i(DEBUG, STRING_DEBUG + "onRestart()");
}
@Override
public void onDestroy() {
super.onDestroy();
Log.i(DEBUG, STRING_DEBUG + "onDestroy()");
}
}
ListFragment.java // FriendsFragment
public class FriendsFragment extends ListFragment {
private static final String[] FRIENDS = { "ladygaga", "msrebeccablack", "taylorswift13" };
private static final String TAG = "Lab-Fragments";
private static final String DEBUG = "DEBUG";
private static final String STRING_DEBUG = "ListFragment ";
private int currentIndex = -1;
public interface SelectionListener {
public void onItemSelected(int position);
}
private SelectionListener mCallback;
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
Log.i(DEBUG, STRING_DEBUG + "onAttach()");
// Make sure that the hosting Activity has implemented
// the SelectionListener callback interface. We need this
// because when an item in this ListFragment is selected,
// the hosting Activity's onItemSelected() method will be called.
try {
mCallback = (SelectionListener) activity;
} catch (ClassCastException e) {
throw new ClassCastException(activity.toString()
+ " must implement SelectionListener");
}
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.i(DEBUG, STRING_DEBUG + "onCreate()");
// use different layout definition, depending on whether device is pre-
// or post-honeycomb
int layout = Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB
? android.R.layout.simple_list_item_activated_1
: android.R.layout.simple_list_item_1;
// Set the list adapter for this ListFragment
setListAdapter(new ArrayAdapter<String>(getActivity(), layout, FRIENDS));
if (isInTwoPaneMode()) {
setRetainInstance(true);
}
}
// Note: ListFragments come with a default onCreateView() method.
// For other Fragments you'll normally implement this method.
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
Log.i(DEBUG, STRING_DEBUG + "onCreateView()");
return super.onCreateView(inflater, container, savedInstanceState);
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
Log.i(DEBUG, STRING_DEBUG + "onActivityCreated()");
Log.i(TAG, "Entered onActivityCreated()");
// When using two-pane layout, configure the ListView to highlight the
// selected list item
if (isInTwoPaneMode()) {
getListView().setChoiceMode(ListView.CHOICE_MODE_SINGLE);
}
if (isInTwoPaneMode() && -1 != currentIndex) {
getListView().setItemChecked(currentIndex, true);
}
}
@Override
public void onStart() {
super.onStart();
Log.i(DEBUG, STRING_DEBUG + "onStart()");
}
@Override
public void onResume() {
super.onResume();
Log.i(DEBUG, STRING_DEBUG + "onResume()");
}
@Override
public void onPause() {
super.onPause();
Log.i(DEBUG, STRING_DEBUG + "onPause()");
}
public void onStop() {
super.onStop();
Log.i(DEBUG, STRING_DEBUG + "onStop()");
}
public void onDestroyView() {
super.onDestroyView();
Log.i(DEBUG, STRING_DEBUG + "onDestroyView()");
}
public void onDestroy() {
super.onDestroy();
Log.i(DEBUG, STRING_DEBUG + "onDestroy()");
}
public void onDetach() {
super.onDetach();
Log.i(DEBUG, STRING_DEBUG + "onDetach()");
}
@Override
public void onListItemClick(ListView l, View view, int position, long id) {
// Notify the hosting Activity that a selection has been made.
currentIndex = position;
mCallback.onItemSelected(position);
}
// If there is a FeedFragment, then the layout is two-pane
private boolean isInTwoPaneMode() {
return getFragmentManager().findFragmentById(R.id.feed_frag) != null;
}
}
Fragment // QuoteFragment //DisplayFragment
package course.labs.fragmentslab;
import android.app.Activity;
import android.app.Fragment;
import android.os.Bundle;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
public class FeedFragment extends Fragment {
private static final String TAG = "Lab-Fragments";
private TextView mTextView;
private static FeedFragmentData feedFragmentData;
private static final String DEBUG = "DEBUG";
private static final String STRING_DEBUG = "Fragment ";
//Index of current selected item to display in FeedFragment
private int currentIndex = -1;
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
Log.i(DEBUG, STRING_DEBUG + "onAttach()");
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.i(DEBUG, STRING_DEBUG + "onCreate()");
if (isInTwoPaneMode()) {
setRetainInstance(true);
}
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
Log.i(DEBUG, STRING_DEBUG + "onCreateView()");
return inflater.inflate(R.layout.feed, container, false);
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
Log.i(DEBUG, STRING_DEBUG + "onActivityCreated()");
mTextView = (TextView) getView().findViewById(R.id.feed_view);
// Read in all Twitter feeds
if (null == feedFragmentData) {
feedFragmentData = new FeedFragmentData(getActivity());
}
Log.d("DEBUG", "Index = " + currentIndex);
if (-1 != currentIndex) {
updateFeedDisplay(currentIndex);
}
}
// Display Twitter feed for selected feed
void updateFeedDisplay(int position) {
Log.i(TAG, "Entered updateFeedDisplay()");
currentIndex = position;
mTextView.setText(feedFragmentData.getFeed(position));
}
// If there is a FeedFragment, then the layout is two-pane
private boolean isInTwoPaneMode() {
return getFragmentManager().findFragmentById(R.id.feed_frag) != null;
}
@Override
public void onStart() {
super.onStart();
Log.i(DEBUG, STRING_DEBUG + "onStart()");
}
@Override
public void onResume() {
super.onResume();
Log.i(DEBUG, STRING_DEBUG + "onResume()");
}
@Override
public void onPause() {
super.onPause();
Log.i(DEBUG, STRING_DEBUG + "onPause()");
}
public void onStop() {
super.onStop();
Log.i(DEBUG, STRING_DEBUG + "onStop()");
}
public void onDestroyView() {
super.onDestroyView();
Log.i(DEBUG, STRING_DEBUG + "onDestroyView()");
}
public void onDestroy() {
super.onDestroy();
Log.i(DEBUG, STRING_DEBUG + "onDestroy()");
}
public void onDetach() {
super.onDetach();
Log.i(DEBUG, STRING_DEBUG + "onDetach()");
}
}
main_activity.xml
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/fragment_container"
android:layout_width="match_parent"
android:layout_height="match_parent" />
feed.xml
<TextView
android:id="@+id/feed_view"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="@string/greeting" />
最佳答案
在 fragment Activity 的 onCreate
方法中,您可以检查是否 savedInstanceState == null
并仅在该条件为true。否则 Android 将自动重新附加现有 fragment 。我建议使用以下代码:
...
if (!isInTwoPaneMode()) {
if (savedInstanceState == null) {
mFriendsFragment = new FriendsFragment();
FragmentTransaction transaction = getFragmentManager().beginTransaction();
transaction.add(R.id.fragment_container, mFriendsFragment);
transaction.commit();
}
}
...
关于java - Android 在每次方向改变时都会创建额外的 ListFragment,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21677605/
我有一个包含项目类别的 ListView 。当我按下列表中的类别标题时,它将显示另一个 ListView ,其中包含所选类别中的项目。 我正在使用 ListFragment 执行此操作。我是否必须开始
首先我应该提到我使用 ActionBarSherlock 库来实现向后兼容性。 我有一个添加 ListFragment 的 Activity 当它第一次启动时。我有一个自定义 Loader我实现并遵循
谁能告诉我一个使用 CursorLoader 查询 SQLite 数据库并填充 ListFragment 的简单示例?下面的代码可以编译,但是当我运行它时,LogCat 告诉我“ListFrag”不能
我是 Android 开发新手,我正在尝试创建一个 ListFragment,它可以从 3 个 ArrayList 对象创建一个 ListView,每个对象包含 30 个字符串。每行将包含每个数组中的
我有一个具有两个 fragment (权重 3 和 2)线性布局的 Activity 。第一个 fragment 是一个列表 fragment 。它最初具有指定的权重高度,但是当列表变大时,它会变得更
我试图让我的 ListFragment 显示在我的 View 中,但是我对这段代码遇到了无穷无尽的问题,无法让它显示我的解析器的结果。我相信这很可能是我的 xml 文件的问题,我的代码中没有收到任何错
我有一个像这样的列表 fragment : private ArrayListimages; private View view; @Override public void onCreate(Bun
我已经向 ListFragment 添加了一个选项菜单,我需要在用户单击它时显示一个新的 ListFragment。这有什么问题吗?我对开始 fragment 的正确方法有点困惑.. 最佳答案 将事务
我是 android api 的新手,我在 .. 基础知识方面遇到了一些问题。我试图让列表正常工作,但列表显示了错误的文本: 每一行有两个 TextView(标题,副标题),这里是 xml:
我已经用头撞墙了大约一天了。我希望有人能提供帮助。 我的程序在异步任务中将一些 Json 加载并解析到对象数组中。这个异步任务是从我的 ListFragment 类调用的,数据在 onPostExec
我可以设法制作具有相同图标的列表 fragment 。但似乎我无法用不同的图标制作该列表。因此列表结果对于不同的项目列表将是不同的图标。我试试看。当我尝试构建它时。它没有显示错误。但是当我运行它时。它
我是 Android 开发的新手。我在使用 ListFragment 和自定义适配器时遇到问题。我的列表中没有对象。你能检查我的代码吗?如果我滚动到底部,您会看到一个对象,但随后会崩溃。 ListFr
我使用 Listfragmen,您的适配器在其中输出图像。 我需要在两列中显示图像。 我尝试放入 listView fragment ,但数据未显示。 我需要有两列来显示图像。
我正在尝试将一些数据加载到 ListFragment 中,但出现此错误: 05-07 00:05:30.533 625-625/com.myapp.android E/AndroidRuntime
我检查了有关 ListFragments 异步更新的其他答案,发现 notifyDataSetChanged() 方法不起作用的最常见问题是,开发人员倾向于覆盖初始适配器源,导致适配器丢失引用,因此不
我的应用程序中有两个列表 fragment 和一个 fragment 。我给你看第一个 fragment 。 应用程序启动,Asynctask 检索数据并将它们放入 arrayNews 中。我认为问题
我有一个 ListFragment,其中包含使用自定义 ArrayAdapter 填充的 Earthquake 对象列表。每个元素要么“突出显示”,要么不“突出显示”,具体取决于地震的震级是否高于某个
我是第一次使用 ListFragments。我想显示来自 sqlite 数据库的数据。 我的问题:我无法检索上下文。 我在扩展 ListFragment 的类中尝试了 this.getActivity
我有一个带有 ArrayAdapter 的 ListFragment 来显示对象列表。我想刷新 ListFragment 的内容。这些项目本身稍微改变了它们的文本,也改变了它们的顺序。我有一百多个项目
我有一个列表 fragment 显示我的自定义列表项,它由包含图像和文本等的相对布局组成。
我是一名优秀的程序员,十分优秀!