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));
}
}
}
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
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);
}
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");
}
}
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 .
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
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
}
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.
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
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()
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
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");
}
}
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.
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;
}
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);
}