向上按钮调用父级的 OnDestroy Activity
Up Button Calls OnDestroy of Parent Activity
最重要的是,我想澄清一件事:我正在努力使用的按钮不是 back
按钮。我指的是应用程序顶部 ActionBar / Toolbar 中的 up/home
按钮,而不是底部的 Android 按钮。有几个 posts 性质相似,但它们针对的是后退按钮,而不是向上按钮。
情况如下:我有一个 Activity A,它有一个 ListView 片段。当用户单击列表视图项时,它会启动 Activity B。非常典型。 Activity A 在工具栏中有一个 EditText
字段,允许用户输入搜索参数。如果用户从 Activity B 中点击 up/home
按钮,我 return 成功到 Activity A。但是,我希望 Activity A 显示相同的文本他们离开时存在的 EditText
字段。如果用户点击 back
按钮,该文本将被恢复。但是,如果他们使用 up/home
按钮导航,则 EditText
字段为空。
使用一些日志语句,我可以看到当从 activity A 中点击列表项时,onSaveInstanceState
和 onStop
都被调用(但 onDestroy
是那时没有调用。)从 activity B 开始,当 up/home
按钮被点击时,来自 activity A 的 onDestroy
立即 调用,然后是 onCreate
,等等。但是,包 savedInstanceState
是空的,大概是因为刚刚调用了 onDestroy
。
为什么在 returning 到 Activity A 时调用 onDestroy
?这对我来说毫无意义。这是清单中的内容:
<activity
android:name=".Activity.ActivityA"
android:label="@string/app_name"
android:parentActivityName=".Activity.ParentActivity"
android:theme="@style/AppTheme"
android:launchMode="singleTop"
android:windowSoftInputMode="stateVisible" />
<activity
android:name=".Activity.ActivityB"
android:label="@string/app_name"
android:parentActivityName=".Activity.ActivityA"
android:theme="@style/AppTheme" />
下面是ActivityA中的相关方法:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_search);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
actionBar = getSupportActionBar();
if (actionBar != null)
initializeActionBar();
if (getSupportActionBar() != null)
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
Log.d(TAG, "on create");
if (savedInstanceState != null) {
Log.d(TAG, "saved instance state not null");
if (savedInstanceState.getString("search_text") != null)
etSearch.setText(savedInstanceState.getString("search_text"));
}
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putString("search_text", etSearch.getText().toString());
Log.d(TAG, "on Save instance state");
}
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
Log.d(TAG, "on restore instance state");
if (savedInstanceState != null) {
if (savedInstanceState.getString("search_text") != null)
etSearch.setText(savedInstanceState.getString("search_text"));
}
}
@Override
protected void onResume() {
super.onResume();
Log.d(TAG, "on resume");
}
@Override
protected void onStop() {
super.onStop();
Log.d(TAG, "on stop");
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.d(TAG, "on destroy");
}
private void initializeActionBar() {
actionBar.setCustomView(R.layout.actionbar_with_edittext);
etSearch = (EditText) actionBar.getCustomView().findViewById(R.id.actionbar_searchfield);
etSearch.setOnEditorActionListener(new TextView.OnEditorActionListener() {
@Override
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
if(event != null && event.getKeyCode() == KeyEvent.KEYCODE_ENTER && event.getAction() == KeyEvent.ACTION_DOWN) {
initiateNewSearch();
etSearch.clearFocus();
}
return false;
}
});
InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
imm.showSoftInput(etSearch, InputMethodManager.SHOW_IMPLICIT);
etSearch.requestFocus();
}
我认为 Activity B 中的任何代码与此处无关。
这是当用户点击 Activity A:
中的列表视图项时我的控制台输出
on Save instance state
on stop
然后这是当用户点击 activity B:
中的 up/home
按钮时生成的内容
on destroy
on create
on resume
如果还有什么可以帮到您的,请告诉我。感谢您的任何建议!
我不知道为什么默认的向上按钮实现会创建一个新的 activity,但对我来说一个可行的解决方案是覆盖 onOptionsItemSelected
:
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int id = item.getItemId();
if(id== android.R.id.home ){
onBackPressed();
return true;
}
return super.onOptionsItemSelected(item);
}
这个解决方案也适用于我:
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home:
Intent intent = NavUtils.getParentActivityIntent(this);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP
| Intent.FLAG_ACTIVITY_SINGLE_TOP);
NavUtils.navigateUpTo(this, intent);
return true;
default:
break;
}
return super.onOptionsItemSelected(item);
}
我也遇到了这个问题,在进一步挖掘之后发现正确的解决方案是在清单中进行更改以将以下内容添加到您的父级 activity:
android:launchMode="singleTop"
详细说明here.
最重要的是,我想澄清一件事:我正在努力使用的按钮不是 back
按钮。我指的是应用程序顶部 ActionBar / Toolbar 中的 up/home
按钮,而不是底部的 Android 按钮。有几个 posts 性质相似,但它们针对的是后退按钮,而不是向上按钮。
情况如下:我有一个 Activity A,它有一个 ListView 片段。当用户单击列表视图项时,它会启动 Activity B。非常典型。 Activity A 在工具栏中有一个 EditText
字段,允许用户输入搜索参数。如果用户从 Activity B 中点击 up/home
按钮,我 return 成功到 Activity A。但是,我希望 Activity A 显示相同的文本他们离开时存在的 EditText
字段。如果用户点击 back
按钮,该文本将被恢复。但是,如果他们使用 up/home
按钮导航,则 EditText
字段为空。
使用一些日志语句,我可以看到当从 activity A 中点击列表项时,onSaveInstanceState
和 onStop
都被调用(但 onDestroy
是那时没有调用。)从 activity B 开始,当 up/home
按钮被点击时,来自 activity A 的 onDestroy
立即 调用,然后是 onCreate
,等等。但是,包 savedInstanceState
是空的,大概是因为刚刚调用了 onDestroy
。
为什么在 returning 到 Activity A 时调用 onDestroy
?这对我来说毫无意义。这是清单中的内容:
<activity
android:name=".Activity.ActivityA"
android:label="@string/app_name"
android:parentActivityName=".Activity.ParentActivity"
android:theme="@style/AppTheme"
android:launchMode="singleTop"
android:windowSoftInputMode="stateVisible" />
<activity
android:name=".Activity.ActivityB"
android:label="@string/app_name"
android:parentActivityName=".Activity.ActivityA"
android:theme="@style/AppTheme" />
下面是ActivityA中的相关方法:
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_search);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
actionBar = getSupportActionBar();
if (actionBar != null)
initializeActionBar();
if (getSupportActionBar() != null)
getSupportActionBar().setDisplayHomeAsUpEnabled(true);
Log.d(TAG, "on create");
if (savedInstanceState != null) {
Log.d(TAG, "saved instance state not null");
if (savedInstanceState.getString("search_text") != null)
etSearch.setText(savedInstanceState.getString("search_text"));
}
}
@Override
public void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
outState.putString("search_text", etSearch.getText().toString());
Log.d(TAG, "on Save instance state");
}
@Override
protected void onRestoreInstanceState(Bundle savedInstanceState) {
super.onRestoreInstanceState(savedInstanceState);
Log.d(TAG, "on restore instance state");
if (savedInstanceState != null) {
if (savedInstanceState.getString("search_text") != null)
etSearch.setText(savedInstanceState.getString("search_text"));
}
}
@Override
protected void onResume() {
super.onResume();
Log.d(TAG, "on resume");
}
@Override
protected void onStop() {
super.onStop();
Log.d(TAG, "on stop");
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.d(TAG, "on destroy");
}
private void initializeActionBar() {
actionBar.setCustomView(R.layout.actionbar_with_edittext);
etSearch = (EditText) actionBar.getCustomView().findViewById(R.id.actionbar_searchfield);
etSearch.setOnEditorActionListener(new TextView.OnEditorActionListener() {
@Override
public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {
if(event != null && event.getKeyCode() == KeyEvent.KEYCODE_ENTER && event.getAction() == KeyEvent.ACTION_DOWN) {
initiateNewSearch();
etSearch.clearFocus();
}
return false;
}
});
InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
imm.showSoftInput(etSearch, InputMethodManager.SHOW_IMPLICIT);
etSearch.requestFocus();
}
我认为 Activity B 中的任何代码与此处无关。
这是当用户点击 Activity A:
中的列表视图项时我的控制台输出on Save instance state
on stop
然后这是当用户点击 activity B:
中的up/home
按钮时生成的内容
on destroy
on create
on resume
如果还有什么可以帮到您的,请告诉我。感谢您的任何建议!
我不知道为什么默认的向上按钮实现会创建一个新的 activity,但对我来说一个可行的解决方案是覆盖 onOptionsItemSelected
:
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int id = item.getItemId();
if(id== android.R.id.home ){
onBackPressed();
return true;
}
return super.onOptionsItemSelected(item);
}
这个解决方案也适用于我:
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case android.R.id.home:
Intent intent = NavUtils.getParentActivityIntent(this);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP
| Intent.FLAG_ACTIVITY_SINGLE_TOP);
NavUtils.navigateUpTo(this, intent);
return true;
default:
break;
}
return super.onOptionsItemSelected(item);
}
我也遇到了这个问题,在进一步挖掘之后发现正确的解决方案是在清单中进行更改以将以下内容添加到您的父级 activity:
android:launchMode="singleTop"
详细说明here.