web-dev-qa-db-fra.com

Comment implémenter OnFragmentInteractionListener

J'ai une application générée par un assistant avec un tiroir de navigation dans Android studio 0.8.2

J'ai créé un fragment et l'a ajouté à newInstance () et j'obtiens cette erreur:

com.domain.myapp E/AndroidRuntime EX EXCEPTION FATALE: exception principale Java.lang.ClassCast: com.domain.myapp.MainActivity@422fb8f0 doit implémenter OnFragmentInteractionListener

Je ne trouve nulle part comment implémenter ce OnFragmentInteractionListener ?? Il est introuvable, même dans la documentation Android sdk!

MainActivity.Java

import Android.app.Activity;

import Android.app.ActionBar;
import Android.app.Fragment;
import Android.app.FragmentManager;
import Android.os.Bundle;
import Android.view.LayoutInflater;
import Android.view.Menu;
import Android.view.MenuItem;
import Android.view.View;
import Android.view.ViewGroup;
import Android.support.v4.widget.DrawerLayout;


public class MainActivity extends Activity
    implements NavigationDrawerFragment.NavigationDrawerCallbacks {

/**
 * Fragment managing the behaviors, interactions and presentation of the navigation drawer.
 */
private NavigationDrawerFragment mNavigationDrawerFragment;

/**
 * Used to store the last screen title. For use in {@link #restoreActionBar()}.
 */
private CharSequence mTitle;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    mNavigationDrawerFragment = (NavigationDrawerFragment)
            getFragmentManager().findFragmentById(R.id.navigation_drawer);
    mTitle = getTitle();

    // Set up the drawer.
    mNavigationDrawerFragment.setUp(
            R.id.navigation_drawer,
            (DrawerLayout) findViewById(R.id.drawer_layout));
}

@Override
public void onNavigationDrawerItemSelected(int position) {
    // update the main content by replacing fragments
    FragmentManager fragmentManager = getFragmentManager();

    switch (position) {
        case 0: fragmentManager.beginTransaction()
                .replace(R.id.container, PlaceholderFragment.newInstance(position + 1))
                .commit(); break; 
        case 1: fragmentManager.beginTransaction() 
                .replace(R.id.container, AboutFragment.newInstance("test1", "test2"))
                .commit(); break; // this crashes the app
        case 2: fragmentManager.beginTransaction()
                .replace(R.id.container, BrowseQuotesFragment.newInstance("test1", "test2"))
                .commit(); break; // this crashes the app
    }
}


public void onSectionAttached(int number) {
    switch (number) {
        case 1:
            mTitle = getString(R.string.title_section1);
            break;
        case 2:
            mTitle = getString(R.string.title_section2);
            break;
        case 3:
            mTitle = getString(R.string.title_section3);
            break;
    }
}

public void restoreActionBar() {
    ActionBar actionBar = getActionBar();
    actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_STANDARD);
    actionBar.setDisplayShowTitleEnabled(true);
    actionBar.setTitle(mTitle);
}


@Override
public boolean onCreateOptionsMenu(Menu menu) {
    if (!mNavigationDrawerFragment.isDrawerOpen()) {
        // Only show items in the action bar relevant to this screen
        // if the drawer is not showing. Otherwise, let the drawer
        // decide what to show in the action bar.
        getMenuInflater().inflate(R.menu.main, menu);
        restoreActionBar();
        return true;
    }
    return super.onCreateOptionsMenu(menu);
}

@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();
    if (id == R.id.action_settings) {
        return true;
    }
    return super.onOptionsItemSelected(item);
}

/**
 * A placeholder fragment containing a simple view.
 */
public static class PlaceholderFragment extends Fragment {
    /**
     * The fragment argument representing the section number for this
     * fragment.
     */
    private static final String ARG_SECTION_NUMBER = "section_number";

    /**
     * Returns a new instance of this fragment for the given section
     * number.
     */
    public static PlaceholderFragment newInstance(int sectionNumber) {
        PlaceholderFragment fragment = new PlaceholderFragment();
        Bundle args = new Bundle();
        args.putInt(ARG_SECTION_NUMBER, sectionNumber);
        fragment.setArguments(args);
        return fragment;
    }

    public PlaceholderFragment() {
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View rootView = inflater.inflate(R.layout.fragment_main, container, false);
        return rootView;
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        ((MainActivity) activity).onSectionAttached(
                getArguments().getInt(ARG_SECTION_NUMBER));
    }
}

}

133
Mario M

Les réponses publiées ici n'ont pas aidé, mais le lien suivant l'a fait:

http://developer.Android.com/training/basics/fragments/communicating.html

Définir une interface

public class HeadlinesFragment extends ListFragment {
    OnHeadlineSelectedListener mCallback;

    // Container Activity must implement this interface
    public interface OnHeadlineSelectedListener {
        public void onArticleSelected(int position);
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);

        // This makes sure that the container activity has implemented
        // the callback interface. If not, it throws an exception
        try {
            mCallback = (OnHeadlineSelectedListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString()
                    + " must implement OnHeadlineSelectedListener");
        }
    }

    ...
}

Par exemple, la méthode suivante du fragment est appelée lorsque l'utilisateur clique sur un élément de la liste. Le fragment utilise l'interface de rappel pour transmettre l'événement à l'activité parent.

@Override
public void onListItemClick(ListView l, View v, int position, long id) {
    // Send the event to the Host activity
    mCallback.onArticleSelected(position);
}

Implémenter l'interface

Par exemple, l'activité suivante implémente l'interface de l'exemple ci-dessus.

public static class MainActivity extends Activity
        implements HeadlinesFragment.OnHeadlineSelectedListener{
    ...

    public void onArticleSelected(int position) {
        // The user selected the headline of an article from the HeadlinesFragment
        // Do something here to display that article
    }
}

Mise à jour pour l'API 23: 8/31/2015

La méthode surchargée onAttach(Activity activity) est désormais obsolète dans Android.app.Fragment, le code doit être mis à niveau vers onAttach(Context context).

@Override
public void onAttach(Context context) {
    super.onAttach(context);
}


@Override
public void onStart() {
    super.onStart();
    try {
        mListener = (OnFragmentInteractionListener) getActivity();
    } catch (ClassCastException e) {
        throw new ClassCastException(getActivity().toString()
                + " must implement OnFragmentInteractionListener");
    }
}
113
meda

Pour ceux d'entre vous qui ne comprennent toujours pas après avoir lu la réponse de @meda, voici mon explication concise et complète de ce problème:

Supposons que vous avez 2 fragments, Fragment_A et Fragment_B qui sont générés automatiquement à partir de l'application. Dans la partie inférieure de vos fragments générés, vous allez trouver ce code:

public class Fragment_A extends Fragment {

    //rest of the code is omitted

    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        public void onFragmentInteraction(Uri uri);
    }
}

public class Fragment_B extends Fragment {

    //rest of the code is omitted

    public interface OnFragmentInteractionListener {
        // TODO: Update argument type and name
        public void onFragmentInteraction(Uri uri);
    }
}

Pour résoudre ce problème, vous devez ajouter la méthode onFragmentInteraction à votre activité, qui dans mon cas s'appelle MainActivity2. Après cela, vous devez implements tous les fragments de la MainActivity comme ceci:

public class MainActivity2 extends ActionBarActivity
        implements Fragment_A.OnFragmentInteractionListener, 
                   Fragment_B.OnFragmentInteractionListener, 
                   NavigationDrawerFragment.NavigationDrawerCallbacks {
    //rest code is omitted

    @Override
    public void onFragmentInteraction(Uri uri){
        //you can leave it empty
    }
}

P.S .: En bref, cette méthode pourrait être utilisée pour communiquer entre fragments. Pour ceux d'entre vous qui veulent en savoir plus sur cette méthode, veuillez vous référer à ceci lien .

200
Bla...

Consultez votre Fragment auto-généré créé par Android Studio. Lorsque vous avez créé la nouvelle Fragment, Studio a écrasé un tas de code pour vous. Au bas du modèle généré automatiquement, il y a une définition d'interface interne appelée OnFragmentInteractionListener. Votre Activity doit implémenter cette interface. Il s'agit du modèle recommandé pour que votre Fragment informe votre Activity d'événements afin qu'il puisse ensuite prendre les mesures appropriées, telles que charger un autre Fragment. Consultez cette page pour plus de détails. Recherchez la section "Création de rappels d'événement pour l'activité": http://developer.Android.com/guide/components/fragments.html

41
Larry Schiefer

Pour ceux d'entre vous qui visitent cette page à la recherche d'éclaircissements supplémentaires sur cette erreur, dans mon cas, l'activité faisant l'appel au fragment devait avoir 2 implémentations dans ce cas, comme ceci:

public class MyActivity extends Activity implements 
    MyFragment.OnFragmentInteractionListener, 
    NavigationDrawerFragment.NaviationDrawerCallbacks {
    ...// rest of the code
}
26
Bwvolleyball

Vous devriez essayer de supprimer le code suivant de vos fragments

    try {
        mListener = (OnFragmentInteractionListener) activity;
    } catch (ClassCastException e) {
        throw new ClassCastException(activity.toString()
                + " must implement OnFragmentInteractionListener");
    }

L’interface/écouteur est une valeur créée par défaut afin que votre activité et vos fragments puissent communiquer plus facilement.

9
Joe Plante

En plus de la réponse de @ user26409021, si vous avez ajouté un ItemFragment, le message dans ItemFragment est;

Activities containing this fragment MUST implement the {@link OnListFragmentInteractionListener} interface.

Et vous devriez ajouter dans votre activité;

public class MainActivity extends AppCompatActivity
    implements NavigationView.OnNavigationItemSelectedListener, ItemFragment.OnListFragmentInteractionListener {

//the code is omitted

 public void onListFragmentInteraction(DummyContent.DummyItem uri){
    //you can leave it empty
}

Ici, l'élément factice est ce que vous avez au bas de votre ItemFragment

4
oneNiceFriend

Juste un addenda:

OnFragmentInteractionListener gère la communication entre Activity et Fragment à l'aide d'une interface (OnFragmentInteractionListener) et est créé par défaut par Android Studio, mais si vous n'avez pas besoin de communiquer avec votre activité, vous pouvez simplement vous en débarrasser.

L'objectif est que vous puissiez attacher votre fragment à plusieurs activités tout en réutilisant la même approche de communication (Chaque activité pourrait avoir son propre OnFragmentInteractionListener pour chaque fragment).

Mais et si je suis sûr que mon fragment sera attaché à un seul type d'activité et que je veux communiquer avec cette activité?

Ensuite, si vous ne voulez pas utiliser OnFragmentInteractionListener à cause de sa verbosité, vous pouvez accéder à vos méthodes d’activité en utilisant:

((MyActivityClass) getActivity()).someMethod()
3
sagits

OnFragmentInteractionListener est l'implémentation par défaut pour la gestion de la communication fragment à activité. Cela peut être mis en œuvre en fonction de vos besoins. Supposons que si vous avez besoin qu'une fonction de votre activité soit exécutée lors d'une action particulière de votre fragment, vous pouvez utiliser cette méthode de rappel. Si vous n'avez pas besoin de cette interaction entre votre hébergement activity et fragment, vous pouvez supprimer cette implémentation.

En bref, vous devriez implement l'écouteur dans votre activité d'hébergement de fragments si vous avez besoin de l'interaction fragment-activité comme celle-ci.

public class MainActivity extends Activity implements 
YourFragment.OnFragmentInteractionListener {..}

et votre fragment devrait l'avoir défini comme ceci

public interface OnFragmentInteractionListener {
    // TODO: Update argument type and name
    void onFragmentInteraction(Uri uri);
}

fournissez également la définition de void onFragmentInteraction(Uri uri); dans votre activité

ou bien simplement supprimer l'initialisation listener de votre onAttach de votre fragment si vous n'avez aucune interaction fragment-activity

3
Navneet Krishna

Au lieu d'utiliser Activity use context.It fonctionne pour moi.

@Override
    public void onAttach(Context context) {
        super.onAttach(context);
        try {
            mListener = (OnFragmentInteractionListener) context;
        } catch (ClassCastException e) {
            throw new ClassCastException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
}
3
KCN

Allez simplement dans votre fragment Activity et supprimez toute méthode ..... à la place de la méthode createview.

votre fragment n'a que sur la méthode oncreateview c'est tout.

// seule cette méthode implémente une autre méthode delete

 @Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    View rootView = inflater.inflate(R.layout.fragment_main, container, false);
    return rootView;
}

et assurez-vous que votre mise en page est une démo pour vous.

2
Kalpesh A. Nikam

J'aimerais ajouter la destruction de l'auditeur lorsque le fragment est détaché de l'activité ou détruit.

@Override
public void onDetach() {
    super.onDetach();
    mListener = null;
}

et lors de l'utilisation de la nouvelle méthode onStart () avec Context

@Override
public void onDestroy() {
    super.onDestroy();
    mListener = null;
}
1
rexxar

Avec moi cela a fonctionné supprimer ce code:

@Override
    public void onAttach(Context context) {
        super.onAttach(context);
        if (context instanceof OnFragmentInteractionListener) {
            mListener = (OnFragmentInteractionListener) context;
        } else {
            throw new RuntimeException(context.toString()
                    + " must implement OnFragmentInteractionListener");
        }
    }

Se terminant comme ça:

@Override
public void onAttach(Context context) {
    super.onAttach(context);
}
0
Vinicius Petrachin