Android onKeyDown вернуться к основной деятельности

Я новичок в Android. Я хотел бы разработать функцию из фрагмента обратно на домашний экран для выхода из приложения, но она не работает. В настоящее время, как только я нажал кнопку «Назад», будет предложено выйти из приложения. Как вернуться к основному действию, чтобы выйти из приложения из onKeyDown?

 public class MainActivity extends AppCompatActivity
    implements NavigationView.OnNavigationItemSelectedListener, FragmentFlickrGridImage.OnFragmentInteractionListener
     {

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
    setSupportActionBar(toolbar);

    if (savedInstanceState == null) {
        Fragment fragment = null;
        Class fragmentClass = null;
        fragment = new MyFragment("0");
        try {
            fragment = (Fragment) fragmentClass.newInstance();
        } catch (Exception e) {
            e.printStackTrace();
        }

        FragmentManager fragmentManager = getSupportFragmentManager();
        fragmentManager.beginTransaction().replace(R.id.flContent, fragment).commit();
    }


    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.setItemIconTintList(null);
    navigationView.setNavigationItemSelectedListener(this);
}

@Override
public void onBackPressed() {
    DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
    if (drawer.isDrawerOpen(GravityCompat.START)) {
        drawer.closeDrawer(GravityCompat.START);
    } 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.share) {
        return true;
    }

    return super.onOptionsItemSelected(item);
}

@SuppressWarnings("StatementWithEmptyBody")
@Override
public boolean onNavigationItemSelected(MenuItem item) {
    // Handle navigation view item clicks here.
    int id = item.getItemId();
    Fragment fragment = null;
    Class fragmentClass = null;
    String title="";

    if (id == R.id.one) {
        fragment = new MyFragment("1");
        title="One";
    } else if (id == R.id.two) {
        fragment = new MyFragment("2");
        title ="two";
    } else if (id == R.id.three) {
        fragment = new MyFragment("3");
        title="three";
    }

    try {
        fragment = (Fragment) fragmentClass.newInstance();
    } catch (Exception e) {
        e.printStackTrace();
    }
    FragmentManager fragmentManager = getSupportFragmentManager();
    fragmentManager.beginTransaction().replace(R.id.flContent, fragment).commit();
    getSupportActionBar().setTitle(title);
    DrawerLayout drawer = (DrawerLayout) findViewById(R.id.drawer_layout);
    drawer.closeDrawer(GravityCompat.START);
    return true;
}

@Override
public void onFragmentInteraction(Uri uri) {

}
// confirm when user click back key
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
    if (keyCode != KeyEvent.KEYCODE_BACK)  return super.onKeyDown(keyCode, event);

    DialogInterface.OnClickListener dialogClickListener = new DialogInterface.OnClickListener() {

        public void onClick(DialogInterface dialog, int which) {
            switch (which){
                case DialogInterface.BUTTON_POSITIVE:
                    //Yes button clicked
                    finish();
                    break;

                case DialogInterface.BUTTON_NEGATIVE:
                    //No button clicked
                    break;
            }
        }
    };

    AlertDialog.Builder builder = new AlertDialog.Builder(this)
            .setMessage("Are you sure?")
            .setPositiveButton("Yes", dialogClickListener)
            .setNegativeButton("No", dialogClickListener)
            .setCancelable(false)
            .setTitle("Exit");
    builder.show();

    return super.onKeyDown(keyCode, event);
} 

Ниже приведен код ошибки, который я получил при реализации addtobackstack.

Application is not responding: AppWindowToken{bcc0a65 token=Token{1a7f775c ActivityRecord{3ccf31cf u0 com.google.android.googlequicksearchbox/com.google.android.launcher.GEL t310}}}.  It has been 5010.0ms since event, 5010.0ms since wait started.  
Reason: Waiting because no window has focus but there is a focused application that may eventually add a window when it finishes starting up. 

person michelletbs    schedule 26.01.2016    source источник
comment
Следуя решению, указанному stackoverflow.com/questions/32944798/ у меня хорошо работает :)   -  person michelletbs    schedule 27.01.2016


Ответы (3)


Попробуйте вызвать moveTaskToBack(true) в методе onBackPressed. Оберните логику, чтобы проверить, является ли фрагмент тем, из которого вы хотите выйти, вокруг этого метода. Взгляните на мой ответ на этот вопрос:

Переключение между фрагментами с помощью onNavigationItemSelected в новом навигационном ящике шаблон (Android Studio 1.4 и более поздние версии)

person Ojonugwa Jude Ochalifu    schedule 03.02.2016

При замене фрагмента добавить fragmentManager.addToBackStack(null);

    FragmentManager fragmentManager = getSupportFragmentManager(); 
    fragmentManager.beginTransaction().replace(R.id.flContent, fragment);
    fragmentManager.addToBackStack(null);
    fragmentManager.commit();

При вызове addToBackStack() транзакция замены сохраняется в обратном стеке, поэтому пользователь может отменить транзакцию и вернуть предыдущий фрагмент, нажав кнопку «Назад».

И в вашем основном действии/домашнем действии переопределите onBackPressed()

    public boolean onKeyDown(int keyCode, KeyEvent event) {
      if (keyCode == KeyEvent.KEYCODE_BACK && event.getRepeatCount() == 0) {
     // Add your Dialogue or whatever to alert 
       moveTaskToBack(true);
      }
      return super.onKeyDown(keyCode, event);
     }

     @Override
     public void onBackPressed() {

      super.onBackPressed();
      onKeyDown(KeyEvent.KEYCODE_BACK, null);
     }
person Pavan Bilagi    schedule 26.01.2016
comment
я тестировал в эмуляторе, но не удалось. вот код ошибки: Отмена события из-за отсутствия фокуса окна. Проблема с эмулятором? - person michelletbs; 26.01.2016
comment
Я тестировал на реальных устройствах. он выйдет прямо из фрагмента. Но то, что я хочу, было из фрагмента, вернуться на главный экран, а затем выйти... хм... - person michelletbs; 26.01.2016

Ниже мое решение

Определите переменную с именем backstact_count, чтобы сохранить количество замененных fragments.

Вам нужно добавить текущий fragment к backstack, когда вы замените его другим

public void replaceFragment(Fragment fragment, boolean isAddToBackStack) {
    FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
    transaction.replace(R.id.flContent, fragment);
    if (isAddToBackStack) {
        transaction.addToBackStack(fragment.getClass().toString());
        backstack_count ++;
    }
    transaction.commit();
}

затем переопределить onBackPressed()

@Override
    public void onBackPressed() {
if (backstack_count > 0) {
            FragmentManager manager = getSupportFragmentManager();
            manager.popBackStack();
            backstack_count--;
        } else {
           super.onBackPressed();
        }
    }

наконец, отредактируйте замененный фрагмент в вашем onCreate(), как показано ниже:

replaceFragment(fragment, true);
person Thai Phan Dinh    schedule 26.01.2016
comment
я добавил эту строку кода в onCreate fragmentManager.beginTransaction().replace(R.id.flContent, fragment).commit(); хотел бы спросить, следует ли заменить эту строку кода на replaceFragment(fragment,true)? - person michelletbs; 26.01.2016
comment
Да, ты должен. Все, что вам нужно, это вызвать replaceFragment(fragment, true); Таким образом, ваш код будет таким: try { fragment = (Fragment) fragmentClass.newInstance(); } catch (Исключение e) { e.printStackTrace(); } replaceFragment(фрагмент, правда); - person Thai Phan Dinh; 26.01.2016
comment
Я пробовал в эмуляторе, приложение не отвечало сообщением об ошибке: Отмена события из-за отсутствия фокуса окна. Вы знаете, что означает эта ошибка? - person michelletbs; 26.01.2016
comment
попробуйте удалить всю свою реализацию onKeyDown. Я имею в виду удалить их. - person Thai Phan Dinh; 26.01.2016
comment
Я пробовал на реальных устройствах, он не возвращается на главный экран для выхода. он выйдет, как только я дважды нажму кнопку «Назад» - person michelletbs; 26.01.2016
comment
ах, твоя проблема в том, что у тебя есть только один фрагмент. Таким образом, вам нечего отображать при нажатии. попробуйте создать main_fragment для главного экрана действия, установите его как первый фрагмент, а затем добавьте свой пользовательский фрагмент, как указано выше. - person Thai Phan Dinh; 26.01.2016
comment
хочу задать еще один вопрос. Я действительно хочу запустить один из фрагментов в качестве основного экрана действия. могу ли я установить его как первый фрагмент? - person michelletbs; 26.01.2016
comment
конечно вы можете. Помните, что НЕ добавляйте фрагмент в стопку, если хотите установить его в качестве первого фрагмента. - person Thai Phan Dinh; 26.01.2016
comment
привет @Phan Dinh Thai, эта ссылка хорошо работает для меня - заголовок stackoverflow.com/questions/32944798/ - person michelletbs; 27.01.2016