如何在活动和片段上正确使用返回堆栈
How to use back stack properly on activities AND fragments
我有 2 个活动:
- LoginActivity:稍后将处理登录内容
- MainActivity:主应用程序,所有功能将在片段上提供。
我想正确使用返回栈,但是我不会。
我想要(但不能)实现的:
应用程序从登录屏幕开始。我登录(现在按“开始”按钮),然后出现主屏幕并加载第一个片段。
现在,如果我按下后退按钮,应用程序将关闭,它应该是这样工作的,它不能通过简单的后退按钮返回到登录屏幕。
但现在如果我点击注销(出现登录屏幕),然后按下后退按钮,应用程序会将我带回主屏幕,这真的很不对劲。如果我注销,登录屏幕上的后退按钮应该会关闭应用程序。
还有另一个问题,关于 backstack 和 fragments:
我有 3 个片段:第一、第二和第三。如果我依次打开它们(当然是在主 activity 内),然后按下后退按钮,我希望它能带我回到之前打开的片段,或者如果没有之前的片段,则关闭应用程序。
现在它成功了,打开了前一个片段,但同时它关闭了整个应用程序。 (我可以看到它们紧接着发生)
缺少什么?应该修改什么以实现此(正常)行为?
谢谢!
我的代码:
主要活动:
public class MainActivity extends AppCompatActivity
implements NavigationView.OnNavigationItemSelectedListener {
public static final String SP_DATA = "SP_DATA";
public static final String FB_LOGIN_STATUS = "FB_LOGIN_STATUS";
public static final String FLRT = "FLRT";
public SharedPreferences sharedPreferences;
@Override
protected void onCreate(Bundle savedInstanceState) {
sharedPreferences = getSharedPreferences(SP_DATA, MODE_PRIVATE);
boolean bAlreadyLoggedIn = sharedPreferences.getBoolean(FB_LOGIN_STATUS, false);
if (bAlreadyLoggedIn) {
Log.d(FLRT, "Already logged in");
}
else {
Log.d(FLRT, "Not logged in");
Intent intent = new Intent(MainActivity.this, LoginActivity.class);
startActivity(intent);
}
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
showFragment(new FirstFragment(),"FirstFragment");
Button btnLogout = (Button) findViewById(R.id.btnLogout);
btnLogout.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
logout();
showActivity(MainActivity.this, LoginActivity.class);
}
});
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
}
});
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
drawer.setDrawerListener(toggle);
toggle.syncState();
NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
navigationView.setNavigationItemSelectedListener(this);
}
private void showActivity(Context context_fromActivity, Class<?> class_toActivty){
Intent intent = new Intent(context_fromActivity, class_toActivty);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
}
private void logout() {
Log.d(FLRT, "Logging out...");
sharedPreferences = getSharedPreferences(SP_DATA, MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putBoolean(FB_LOGIN_STATUS, false);
editor.commit();
}
@Override
public void onBackPressed() {
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
if (drawer.isDrawerOpen(GravityCompat.START)) {
drawer.closeDrawer(GravityCompat.START);
} else {
super.onBackPressed();
}
if (getFragmentManager().getBackStackEntryCount() > 0 ){
getFragmentManager().popBackStack();
} else {
super.onBackPressed();
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@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);
}
@SuppressWarnings("StatementWithEmptyBody")
@Override
public boolean onNavigationItemSelected(MenuItem item) {
// Handle navigation view item clicks here.
int id = item.getItemId();
if (id == R.id.nav_first_layout) {
showFragment(new FirstFragment(), "FirstFragment");
}
else if (id == R.id.nav_second_layout) {
showFragment(new SecondFragment(), "SecondFragment");
}
else if (id == R.id.nav_third_layout) {
showFragment(new ThirdFragment(), "ThirdFragment");
}
else if (id == R.id.nav_share) {
}
else if (id == R.id.nav_send) {
}
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
drawer.closeDrawer(GravityCompat.START);
return true;
}
private void showFragment(Fragment fragment, String sFragmentTAG) {
FragmentManager fragmentManager = getFragmentManager();
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
if (fragmentManager.findFragmentByTag(sFragmentTAG) != null) {
Log.d(FLRT, "Fragment found, using existing one: " + sFragmentTAG);
fragment = fragmentManager.findFragmentByTag(sFragmentTAG);
}
fragmentTransaction.replace(R.id.fragmentContainer, fragment, sFragmentTAG);
fragmentTransaction.addToBackStack(null);
fragmentTransaction.commit();
}
}
登录活动:
public class LoginActivity extends AppCompatActivity {
public static final String SP_DATA = "SP_DATA";
public static final String FB_LOGIN_STATUS = "FB_LOGIN_STATUS";
public static final String FLRT = "FLRT";
public SharedPreferences sharedPreferences;
private Button btn_start;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
btn_start = (Button) findViewById(R.id.btn_start);
btn_start.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
login();
showActivity(LoginActivity.this, MainActivity.class);;
}
});
}
private void showActivity(Context context_fromActivity, Class<?> class_toActivty){
Intent intent = new Intent(context_fromActivity, class_toActivty);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
}
private void login() {
Log.d(FLRT, "Logging in...");
sharedPreferences = getSharedPreferences(SP_DATA, MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putBoolean(FB_LOGIN_STATUS, true);
editor.commit();
}
}
关于Activities
.
之间的切换
一件重要的事。
在 showActivity()
中,使用 finish()
去另一个时杀死当前 Activity
。如果没有帮助 - 请参阅下文。
尝试将 FLAG_ACTIVITY_NEW_TASK
与您当前的标志一起使用,因为 FLAG_ACTIVITY_CLEAR_TOP
将仅关闭高于当前 Activity 的任务中的活动,但不会关闭低于当前的任务。
所以,它会是这样的:intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_CLEAR_TASK);
另一种选择是使用:仅 FLAG_ACTIVITY_CLEAR_TASK
- new Activity 将清除任务并成为其中的 root。
如果两者都无济于事 - 在清单中为每个 Actitivy
.
使用 android:launchMode = "singleInstance"
关于 Fragments
- 我在您的代码中看到一个小错误:它在从 BackStack
[=51= 弹出 Fragment
之前调用 super.onBackPressed();
]
我会这样改变:
@Override
public void onBackPressed() {
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
if (drawer.isDrawerOpen(GravityCompat.START)) {
drawer.closeDrawer(GravityCompat.START);
} else if (getFragmentManager().getBackStackEntryCount() > 0 ){
getFragmentManager().popBackStack();
} else {
super.onBackPressed();
}
}
!! 您还应该测试此行:getFragmentManager().getBackStackEntryCount() > 0
条件为 0
和 1
(因为在我的当前实现中项目 FragmentContainer
正在考虑 BackStack
中的第一个零值)。只需测试 0 和 1,看看哪个有效。
谢谢。
PS。还有一些提示。
您可以在 Activity
中声明 FragmentManager fragmentManager
,以便仅在 onCreate()
中创建一次。总比每次都得到好。
考虑改变这个:
if (id == R.id.nav_first_layout) {
showFragment(new FirstFragment(), "FirstFragment");
}
else if (id == R.id.nav_second_layout) {
showFragment(new SecondFragment(), "SecondFragment");
}
else if (id == R.id.nav_third_layout) {
showFragment(new ThirdFragment(), "ThirdFragment");
}
else if (id == R.id.nav_share) {
}
else if (id == R.id.nav_send) {
}
收件人:
switch (item.getItemId()) {
case R.id.nav_first_layout:
showFragment(new FirstFragment(), "FirstFragment");
break;
case ...
}
等等。代码会更漂亮:)
我有 2 个活动: - LoginActivity:稍后将处理登录内容 - MainActivity:主应用程序,所有功能将在片段上提供。
我想正确使用返回栈,但是我不会。
我想要(但不能)实现的: 应用程序从登录屏幕开始。我登录(现在按“开始”按钮),然后出现主屏幕并加载第一个片段。 现在,如果我按下后退按钮,应用程序将关闭,它应该是这样工作的,它不能通过简单的后退按钮返回到登录屏幕。
但现在如果我点击注销(出现登录屏幕),然后按下后退按钮,应用程序会将我带回主屏幕,这真的很不对劲。如果我注销,登录屏幕上的后退按钮应该会关闭应用程序。
还有另一个问题,关于 backstack 和 fragments: 我有 3 个片段:第一、第二和第三。如果我依次打开它们(当然是在主 activity 内),然后按下后退按钮,我希望它能带我回到之前打开的片段,或者如果没有之前的片段,则关闭应用程序。 现在它成功了,打开了前一个片段,但同时它关闭了整个应用程序。 (我可以看到它们紧接着发生)
缺少什么?应该修改什么以实现此(正常)行为?
谢谢!
我的代码:
主要活动:
public class MainActivity extends AppCompatActivity
implements NavigationView.OnNavigationItemSelectedListener {
public static final String SP_DATA = "SP_DATA";
public static final String FB_LOGIN_STATUS = "FB_LOGIN_STATUS";
public static final String FLRT = "FLRT";
public SharedPreferences sharedPreferences;
@Override
protected void onCreate(Bundle savedInstanceState) {
sharedPreferences = getSharedPreferences(SP_DATA, MODE_PRIVATE);
boolean bAlreadyLoggedIn = sharedPreferences.getBoolean(FB_LOGIN_STATUS, false);
if (bAlreadyLoggedIn) {
Log.d(FLRT, "Already logged in");
}
else {
Log.d(FLRT, "Not logged in");
Intent intent = new Intent(MainActivity.this, LoginActivity.class);
startActivity(intent);
}
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
showFragment(new FirstFragment(),"FirstFragment");
Button btnLogout = (Button) findViewById(R.id.btnLogout);
btnLogout.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
logout();
showActivity(MainActivity.this, LoginActivity.class);
}
});
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
.setAction("Action", null).show();
}
});
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
ActionBarDrawerToggle toggle = new ActionBarDrawerToggle(
this, drawer, toolbar, R.string.navigation_drawer_open, R.string.navigation_drawer_close);
drawer.setDrawerListener(toggle);
toggle.syncState();
NavigationView navigationView = (NavigationView) findViewById(R.id.nav_view);
navigationView.setNavigationItemSelectedListener(this);
}
private void showActivity(Context context_fromActivity, Class<?> class_toActivty){
Intent intent = new Intent(context_fromActivity, class_toActivty);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
}
private void logout() {
Log.d(FLRT, "Logging out...");
sharedPreferences = getSharedPreferences(SP_DATA, MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putBoolean(FB_LOGIN_STATUS, false);
editor.commit();
}
@Override
public void onBackPressed() {
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
if (drawer.isDrawerOpen(GravityCompat.START)) {
drawer.closeDrawer(GravityCompat.START);
} else {
super.onBackPressed();
}
if (getFragmentManager().getBackStackEntryCount() > 0 ){
getFragmentManager().popBackStack();
} else {
super.onBackPressed();
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main, menu);
return true;
}
@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);
}
@SuppressWarnings("StatementWithEmptyBody")
@Override
public boolean onNavigationItemSelected(MenuItem item) {
// Handle navigation view item clicks here.
int id = item.getItemId();
if (id == R.id.nav_first_layout) {
showFragment(new FirstFragment(), "FirstFragment");
}
else if (id == R.id.nav_second_layout) {
showFragment(new SecondFragment(), "SecondFragment");
}
else if (id == R.id.nav_third_layout) {
showFragment(new ThirdFragment(), "ThirdFragment");
}
else if (id == R.id.nav_share) {
}
else if (id == R.id.nav_send) {
}
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
drawer.closeDrawer(GravityCompat.START);
return true;
}
private void showFragment(Fragment fragment, String sFragmentTAG) {
FragmentManager fragmentManager = getFragmentManager();
FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
if (fragmentManager.findFragmentByTag(sFragmentTAG) != null) {
Log.d(FLRT, "Fragment found, using existing one: " + sFragmentTAG);
fragment = fragmentManager.findFragmentByTag(sFragmentTAG);
}
fragmentTransaction.replace(R.id.fragmentContainer, fragment, sFragmentTAG);
fragmentTransaction.addToBackStack(null);
fragmentTransaction.commit();
}
}
登录活动:
public class LoginActivity extends AppCompatActivity {
public static final String SP_DATA = "SP_DATA";
public static final String FB_LOGIN_STATUS = "FB_LOGIN_STATUS";
public static final String FLRT = "FLRT";
public SharedPreferences sharedPreferences;
private Button btn_start;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
btn_start = (Button) findViewById(R.id.btn_start);
btn_start.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
login();
showActivity(LoginActivity.this, MainActivity.class);;
}
});
}
private void showActivity(Context context_fromActivity, Class<?> class_toActivty){
Intent intent = new Intent(context_fromActivity, class_toActivty);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(intent);
}
private void login() {
Log.d(FLRT, "Logging in...");
sharedPreferences = getSharedPreferences(SP_DATA, MODE_PRIVATE);
SharedPreferences.Editor editor = sharedPreferences.edit();
editor.putBoolean(FB_LOGIN_STATUS, true);
editor.commit();
}
}
关于Activities
.
之间的切换
一件重要的事。
在 showActivity()
中,使用 finish()
去另一个时杀死当前 Activity
。如果没有帮助 - 请参阅下文。
尝试将 FLAG_ACTIVITY_NEW_TASK
与您当前的标志一起使用,因为 FLAG_ACTIVITY_CLEAR_TOP
将仅关闭高于当前 Activity 的任务中的活动,但不会关闭低于当前的任务。
所以,它会是这样的:intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_CLEAR_TASK);
另一种选择是使用:仅 FLAG_ACTIVITY_CLEAR_TASK
- new Activity 将清除任务并成为其中的 root。
如果两者都无济于事 - 在清单中为每个 Actitivy
.
使用 android:launchMode = "singleInstance"
关于 Fragments
- 我在您的代码中看到一个小错误:它在从 BackStack
[=51= 弹出 Fragment
之前调用 super.onBackPressed();
]
我会这样改变:
@Override
public void onBackPressed() {
DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
if (drawer.isDrawerOpen(GravityCompat.START)) {
drawer.closeDrawer(GravityCompat.START);
} else if (getFragmentManager().getBackStackEntryCount() > 0 ){
getFragmentManager().popBackStack();
} else {
super.onBackPressed();
}
}
!! 您还应该测试此行:getFragmentManager().getBackStackEntryCount() > 0
条件为 0
和 1
(因为在我的当前实现中项目 FragmentContainer
正在考虑 BackStack
中的第一个零值)。只需测试 0 和 1,看看哪个有效。
谢谢。
PS。还有一些提示。
您可以在 Activity
中声明 FragmentManager fragmentManager
,以便仅在 onCreate()
中创建一次。总比每次都得到好。
考虑改变这个:
if (id == R.id.nav_first_layout) {
showFragment(new FirstFragment(), "FirstFragment");
}
else if (id == R.id.nav_second_layout) {
showFragment(new SecondFragment(), "SecondFragment");
}
else if (id == R.id.nav_third_layout) {
showFragment(new ThirdFragment(), "ThirdFragment");
}
else if (id == R.id.nav_share) {
}
else if (id == R.id.nav_send) {
}
收件人:
switch (item.getItemId()) {
case R.id.nav_first_layout:
showFragment(new FirstFragment(), "FirstFragment");
break;
case ...
}
等等。代码会更漂亮:)