टुकड़ों के साथ एक्शनबार नेविगेशन


88

मेरे पास टैब , बी और सी तीन टैब के साथ एक एक्शन टैब / एक्शन व्यूअर लेआउट है । टैब सी टैब (टुकड़ा) में, मैं एक और टुकड़ा जोड़ रहा हूं कहो टुकड़ा डी । साथ में

 DFragment f= new DFragment();
 ft.add(android.R.id.content, f, "");
 ft.remove(CFragment.this);
 ft.addToBackStack(null);
 ft.commit();

मैं बटन को जोड़ने के लिए DFragment के onResume में एक्शनबार को संशोधित करता हूं:

ActionBar ab = getActivity().getActionBar();
ab.setNavigationMode(ActionBar.NAVIGATION_MODE_STANDARD);
ab.setDisplayHomeAsUpEnabled(true);
ab.setDisplayShowHomeEnabled(true);

अब DFragment में, जब मैं हार्डवेयर (फोन) बैक बटन दबाता हूं, तो मैं मूल Tabbed (ABC) लेआउट के साथ CFragment चयनित होता है। मैं एक्शनबार अप बटन के साथ इस कार्यक्षमता को कैसे प्राप्त कर सकता हूं?


जवाबों:


185

लागू OnBackStackChangedListenerइस कोड को अपनी फ्रैगमेंट गतिविधि में और जोड़ें।

@Override
public void onCreate(Bundle savedInstanceState) {
    //Listen for changes in the back stack
    getSupportFragmentManager().addOnBackStackChangedListener(this);
    //Handle when activity is recreated like on orientation Change
    shouldDisplayHomeUp();
}

@Override
public void onBackStackChanged() {
    shouldDisplayHomeUp();
}

public void shouldDisplayHomeUp(){
   //Enable Up button only  if there are entries in the back stack
   boolean canGoBack = getSupportFragmentManager().getBackStackEntryCount()>0;
   getSupportActionBar().setDisplayHomeAsUpEnabled(canGoBack);
}

@Override
public boolean onSupportNavigateUp() {
    //This method is called when the up button is pressed. Just the pop back stack.
    getSupportFragmentManager().popBackStack();
    return true;
}

20
के बारे में onSupportNavigateUp(), "विधि अपने सुपरक्लास से विधि को ओवरराइड नहीं करती है"।
फ्रेड

10
यदि आपके पास पहले से ही एक है onOptionsItemSelected, तो आइटम android.R.id.homeजोड़ने के बजाय जांच करना संभव है onSupportNavigateUp
डोमसोम

1
अगर API संस्करण> = 14, onSupportNavigateUp @Override सार्वजनिक बूलियन onNavigateUp () {// के बजाय onNavigateUp का उपयोग करें, तो यह विधि तब होती है जब अप बटन दबाया जाता है। बस पॉप बैक स्टैक। getFragmentManager () popBackStack ()।; सच लौटना; }
तेजस्वी हेगड़े

1
वहाँ एक अप कैरट माना जाता है जो आपके ऐप आइकन के बगल में प्रदर्शित होता है ActionBar? जब मैं इस कोड को लागू करता हूं तो मुझे एक नहीं दिखता है। मैं केवल आइकन पर क्लिक करने में सक्षम हूं, लेकिन यह कुछ भी नहीं करता है। Android 4.0+।
Azurespot

3
यदि onBackStackChanged () ओवरराइड नहीं करता है, तो सुनिश्चित करें कि आपकी गतिविधि FragmentManager.OnBackStackChangedListener इंटरफ़ेस को लागू करती है।
19BA में CBA110

41

मैं समझ गया। बस ओवरराइड onOptionsItemSelected गतिविधि की मेजबानी में और backstack, जैसे पॉपअप

public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        case android.R.id.home: {
            FragmentManager fm = getSupportFragmentManager();
            if (fm.getBackStackEntryCount() > 0) {
                fm.popBackStack();
                return true;
            }
            break;
        }
    }
    return super.onOptionsItemSelected(item);
}

नीचे दिए गए उत्तर में कॉल करें getActionBar().setDisplayHomeAsUpEnabled(boolean);और जैसा कि बताया गया है।getActionBar().setHomeButtonEnabled(boolean);onBackStackChanged()


3
आपको getActivity () भी प्राप्त करना है। getActionBar ()। setDisplayHomeAsUpEnabled (गलत); एक बार बैक स्टैक पॉप करने के बाद अप बटन को हटाने के लिए
जोप

ऐसा करने का यह सही तरीका नहीं है। यह अप बटन को सक्षम रखता है।
रोजर गारज़ोन नीटो

1
आपको उस कोड switchको केस के साथ स्टेटमेंट के अंदर रखना चाहिए android.R.id.home
फ्रेड

18

यदि आपके पास एक माता-पिता की गतिविधि है और चाहते हैं कि यह बटन बैक बटन के रूप में काम करे, तो आप इस कोड का उपयोग कर सकते हैं:

इसे अपने मुख्य गतिविधि वर्ग में onCreate में जोड़ें

getSupportFragmentManager().addOnBackStackChangedListener(new FragmentManager.OnBackStackChangedListener() {
        @Override
        public void onBackStackChanged() {
            int stackHeight = getSupportFragmentManager().getBackStackEntryCount();
            if (stackHeight > 0) { // if we have something on the stack (doesn't include the current shown fragment)
                getSupportActionBar().setHomeButtonEnabled(true);
                getSupportActionBar().setDisplayHomeAsUpEnabled(true);
            } else {
                getSupportActionBar().setDisplayHomeAsUpEnabled(false);
                getSupportActionBar().setHomeButtonEnabled(false);
            }
        }

    });

और फिर onOptionsItemSelected जैसे जोड़ें:

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        case android.R.id.home:
            getSupportFragmentManager().popBackStack();
            return true;
     ....
 }

मैं आम तौर पर हर समय इसका उपयोग करता हूं और काफी कानूनी लगता है


1
मैंने इस कोड का बिल्कुल उपयोग करने की कोशिश की, Activityमेरी एक उम्मीद में, कि यह उस टुकड़े पर वापस आ जाएगा जो इससे शुरू हुआ था। बैक बटन मेरे ऐप आइकन के पास दिखाई देता है जब मैं अपने पास जाता हूं Activity, लेकिन मैं आइकन पर क्लिक करता हूं और कुछ भी नहीं होता है (यह टुकड़े में वापस जाना चाहिए)। कोई विचार क्यों? धन्यवाद।
Azurespot

1
@Daniel आपका कोड वैध है .. यह काम करता है। आप बस इसे कैच ऑप्शन के साथ सॉर्स करना चाह सकते हैं .. आप किसी भी अप्रत्याशित अपवाद और क्रैश को रोकने के लिए जानते हैं
Zuko

1
@NoniA। यह केवल एक पिछले टुकड़े (जैसे, टुकड़ा बी -> टुकड़ा ए) पर वापस जाने वाला है, यदि आप एक नई गतिविधि में 1 टुकड़ा फुला रहे हैं, तो यह पूर्व गतिविधि पर वापस नहीं जाएगा।
डैनियल जोकर

10

आप बैक बटन जैसे बैक बटन के साथ वापस जा सकते हैं;

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        case android.R.id.home:
            super.onBackPressed();
            return true;
    }
    return super.onOptionsItemSelected(item);
}

8

मुझे पता है कि यह सवाल पुराना है, लेकिन हो सकता है कि कोई (मेरे जैसा) को भी इसकी आवश्यकता हो।

यदि आपकी गतिविधि AppCompatActivity का विस्तार करती है , तो आप एक सरल (दो-चरण) समाधान का उपयोग कर सकते हैं:

1 - जब भी आप एक गैर-घर का टुकड़ा जोड़ते हैं, तो केवल ऊपर का बटन दिखाते हैं, ठीक खंड के लेन-देन के बाद। इस कदर:

    // ... add a fragment
    // Commit the transaction
    transaction.commit();

    getSupportActionBar().setDisplayHomeAsUpEnabled(true);

2 - फिर जब यूपी बटन दबाया जाता है, तो आप इसे छिपाते हैं।

@Override
public boolean onSupportNavigateUp() {
    getSupportActionBar().setDisplayHomeAsUpEnabled(false);        
    return true;
}

बस।


7

मैंने रोजर गारज़ोन नीटो और सोहेलज़िज़ के उत्तरों के संयोजन का उपयोग किया । मेरे ऐप में एक ही MainActivity है, और इसमें A, B, C के टुकड़े हैं। मेरा "घर" टुकड़ा (ए) ओनबैकस्टैकचैनडेलिस्टनर को लागू करता है, और बैकस्ट के आकार की जांच करता है; यदि यह एक से कम है, तो यह यूपी बटन को छुपाता है। टुकड़े बी और सी हमेशा बैक बटन को लोड करते हैं (मेरे डिज़ाइन में, बी को ए से लॉन्च किया गया है, और सी को बी से लॉन्च किया गया है)। MainActivity ही यूपी बटन टैप पर बैकस्टैक को पॉप करती है, और बटन को दिखाने / छिपाने के तरीके हैं, जिसे टुकड़े कहते हैं:

मुख्य गतिविधि:

public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
        // Respond to the action bar's Up/Home button
        case android.R.id.home:
            getSupportFragmentManager().popBackStack();
            return true;
    }
    return super.onOptionsItemSelected(item);
}

public void showUpButton() { getSupportActionBar().setDisplayHomeAsUpEnabled(true); }
public void hideUpButton() { getSupportActionBar().setDisplayHomeAsUpEnabled(false); }

टुकड़ा (लागू करता है FragmentManager.OnBackStackChangedListener):

public void onCreate(Bundle savedinstanceSate) {
    // listen to backstack changes
    getActivity().getSupportFragmentManager().addOnBackStackChangedListener(this);

    // other fragment init stuff
    ...
}

public void onBackStackChanged() {
    // enable Up button only  if there are entries on the backstack
    if(getActivity().getSupportFragmentManager().getBackStackEntryCount() < 1) {
        ((MainActivity)getActivity()).hideUpButton();
    }
}

टुकड़ा, टुकड़ा:

public void onCreate(Bundle savedinstanceSate) {
    // show the UP button
    ((MainActivity)getActivity()).showUpButton();

    // other fragment init stuff
    ...
}

5

इसने मेरे लिए काम किया। उदाहरण के लिए (ओवर कोटलिन में कोड) ओवरसुपोर्टनैवेटअप और ऑनबैकपैक को ओवरराइड करें;

override fun onBackPressed() {
    val count = supportFragmentManager.backStackEntryCount
    if (count == 0) {
        super.onBackPressed()
    } else {
        supportFragmentManager.popBackStack()
    }
}

override fun onSupportNavigateUp(): Boolean {
    super.onSupportNavigateUp()
    onBackPressed()
    return true
}

अब टुकड़े में, यदि आप ऊपर तीर को प्रदर्शित करते हैं

activity.supportActionBar?.setDisplayHomeAsUpEnabled(true)

इस पर क्लिक करने से आपको पिछली गतिविधि वापस मिल जाती है।


5

Kotlin:

class MyActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        ...
        supportFragmentManager.addOnBackStackChangedListener { setupHomeAsUp() }
        setupHomeAsUp()
    }

    private fun setupHomeAsUp() {
        val shouldShow = 0 < supportFragmentManager.backStackEntryCount
        supportActionBar?.setDisplayHomeAsUpEnabled(shouldShow)
    }

    override fun onSupportNavigateUp(): Boolean = 
        supportFragmentManager.popBackStack().run { true }

    ...
}

2

यह एक बहुत अच्छा और विश्वसनीय उपाय है: http://vinsol.com/blog/2014/10/01/handling-back-button-press-inside-fragments/

आदमी ने एक अमूर्त टुकड़ा बनाया है जो बैकस्पेस व्यवहार को संभालता है और रणनीति पैटर्न का उपयोग करके सक्रिय टुकड़ों के बीच स्विच कर रहा है।

आप में से कुछ के लिए शायद अमूर्त वर्ग में एक छोटी सी खामी है ...

कुछ ही समय में, लिंक से समाधान इस प्रकार है:

// Abstract Fragment handling the back presses

public abstract class BackHandledFragment extends Fragment {
    protected BackHandlerInterface backHandlerInterface;
    public abstract String getTagText();
    public abstract boolean onBackPressed();

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if(!(getActivity()  instanceof BackHandlerInterface)) {
            throw new ClassCastException("Hosting activity must implement BackHandlerInterface");
        } else {
            backHandlerInterface = (BackHandlerInterface) getActivity();
        }
    }

    @Override
    public void onStart() {
        super.onStart();

        // Mark this fragment as the selected Fragment.
        backHandlerInterface.setSelectedFragment(this);
    }

    public interface BackHandlerInterface {
        public void setSelectedFragment(BackHandledFragment backHandledFragment);
    }
}   

और गतिविधि में उपयोग:

// BASIC ACTIVITY CODE THAT LETS ITS FRAGMENT UTILIZE onBackPress EVENTS 
// IN AN ADAPTIVE AND ORGANIZED PATTERN USING BackHandledFragment

public class TheActivity extends FragmentActivity implements BackHandlerInterface {
    private BackHandledFragment selectedFragment;

    @Override
    public void onBackPressed() {
        if(selectedFragment == null || !selectedFragment.onBackPressed()) {
            // Selected fragment did not consume the back press event.
            super.onBackPressed();
        }
    }

    @Override
    public void setSelectedFragment(BackHandledFragment selectedFragment) {
        this.selectedFragment = selectedFragment;
    }
}

इस लिंक पर सवाल का जवाब कर सकते हैं, यह जवाब के आवश्यक हिस्से शामिल करने के लिए बेहतर है यहाँ और संदर्भ के लिए लिंक प्रदान करते हैं। लिंक-केवल उत्तर अमान्य हो सकते हैं यदि लिंक किए गए पृष्ठ बदल जाते हैं।
बुम्मी

BTW: यदि आप डुप्लिकेट की पहचान करते हैं, तो कृपया उन्हें इस तरह ध्वजांकित करें। धन्यवाद।
बुम्मी

क्या यह महत्वपूर्ण है कि ऑनस्टार्ट के अंदर सेटिफ़्लेगमेंट सेट किया जाए?
वीएलोनोव्स
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.