У меня есть приложение, созданное мастером, с навигационным ящиком в Android-студии 0.8.2
Я создал фрагмент и добавил его с помощью newInstance(), и я получаю эту ошибку:
com.domain.myapp E/AndroidRuntime: FATAL EXCEPTION: main java.lang.ClassCastException: com.domain.myapp.MainActivity@422fb8f0 должен реализовать OnFragmentInteractionListener
Я не могу найти нигде, как реализовать этот OnFragmentInteractionListener?? Он не может быть найден даже в документации 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));
}
}
}
Ответ 1
Ответы, отправленные здесь, не помогли, но следующая ссылка сделала:
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");
}
}
...
}
Например, следующий метод в фрагменте вызывается, когда пользователь нажимает на элемент списка. Фрагмент использует интерфейс обратного вызова для доставки события родительской активности.
@Override
public void onListItemClick(ListView l, View v, int position, long id) {
// Send the event to the host activity
mCallback.onArticleSelected(position);
}
Внедрить интерфейс
Например, следующее действие реализует интерфейс из приведенного выше примера.
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
}
}
Обновление для API 23: 8/31/2015
Переопределенный метод onAttach(Activity activity)
теперь устарел в android.app.Fragment
, код должен быть обновлен до 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");
}
}
Ответ 2
Для тех из вас, кто все еще не понимает после прочтения ответа @meda, вот мое краткое и полное объяснение этой проблемы:
Допустим, у вас есть 2 фрагмента, Fragment_A
и Fragment_B
которые автоматически генерируются из приложения. В нижней части сгенерированных фрагментов вы найдете этот код:
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);
}
}
Чтобы решить эту проблему, вы должны добавить метод onFragmentInteraction
в вашу деятельность, которая в моем случае называется MainActivity2
. После этого вам нужно implements
все фрагменты в MainActivity
следующим образом:
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
}
}
PS: Короче говоря, этот метод может быть использован для связи между фрагментами. Для тех из вас, кто хочет узнать больше об этом методе, пожалуйста, перейдите по этой ссылке.
Ответ 3
См. ваш автогенератор Fragment
, созданный Android Studio. Когда вы создали новый Fragment
, Studio запустил кучу кода для вас. В нижней части автогенерированного шаблона есть определение внутреннего интерфейса, называемое OnFragmentInteractionListener
. Ваш Activity
должен реализовать этот интерфейс. Это рекомендуемый шаблон для Fragment
, чтобы уведомить ваш Activity
о событиях, чтобы он мог предпринять соответствующие действия, например загрузить еще один Fragment
. Более подробную информацию см. на этой странице. Найдите раздел "Создание обратных вызовов событий для Activity": http://developer.android.com/guide/components/fragments.html
Ответ 4
Для тех из вас, кто посещает эту страницу, ища дальнейшие разъяснения по этой ошибке, в моем случае активность, вызывающая вызов фрагмента, должна иметь в этом случае 2 инструмента, например:
public class MyActivity extends Activity implements
MyFragment.OnFragmentInteractionListener,
NavigationDrawerFragment.NaviationDrawerCallbacks {
...// rest of the code
}
Ответ 5
Вы должны попробовать удалить следующий код из своих фрагментов
try {
mListener = (OnFragmentInteractionListener) activity;
} catch (ClassCastException e) {
throw new ClassCastException(activity.toString()
+ " must implement OnFragmentInteractionListener");
}
Интерфейс/прослушиватель по умолчанию создан так, чтобы ваша активность и фрагменты могли легче общаться
Ответ 6
В дополнение к @user26409021 ответ, если вы добавили ItemFragment, сообщение в ItemFragment:
Activities containing this fragment MUST implement the {@link OnListFragmentInteractionListener} interface.
И вы должны добавить в свою деятельность:
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
}
Здесь фиктивный элемент - это то, что у вас есть в нижней части вашего ItemFragment
Ответ 7
Вместо использования контекста использования. Он работает для меня.
@Override
public void onAttach(Context context) {
super.onAttach(context);
try {
mListener = (OnFragmentInteractionListener) context;
} catch (ClassCastException e) {
throw new ClassCastException(context.toString()
+ " must implement OnFragmentInteractionListener");
}
}
Ответ 8
Просто добавление:
OnFragmentInteractionListener обрабатывает связь между Activity и Fragment с помощью интерфейса (OnFragmentInteractionListener) и создается по умолчанию Android Studio, но если вам не нужно общаться с вашей деятельностью, вы можете просто получить ездить на нем.
Цель состоит в том, что вы можете прикрепить свой фрагмент к нескольким действиям и по-прежнему использовать один и тот же коммуникационный подход. Каждый процесс может иметь собственный OnFragmentInteractionListener для каждого фрагмента.
Но если я уверен, что мой фрагмент будет привязан только к одному типу активности, и я хочу общаться с этим действием?
Затем, если вы не хотите использовать OnFragmentInteractionListener из-за своей многословия, вы можете получить доступ к своим методам активности, используя:
((MyActivityClass) getActivity()).someMethod()
Ответ 9
OnFragmentInteractionListener
является реализацией по умолчанию для обработки фрагмента для связи активности. Это может быть реализовано в зависимости от ваших потребностей. Предположим, что если вам нужно, чтобы функция в вашей деятельности выполнялась во время определенного действия в вашем фрагменте, вы можете использовать этот метод обратного вызова. Если вам не нужно это взаимодействие между вашей хостинговой activity
и fragment
, вы можете удалить эту реализацию.
Короче говоря, вы должны implement
слушателя в вашу деятельность по размещению фрагментов, если вам нужно взаимодействие фрагмента с активностью, подобное
public class MainActivity extends Activity implements
YourFragment.OnFragmentInteractionListener {..}
и ваш фрагмент должен иметь такое определение
public interface OnFragmentInteractionListener {
// TODO: Update argument type and name
void onFragmentInteraction(Uri uri);
}
также предоставьте определение для void onFragmentInteraction(Uri uri);
в вашей деятельности
или просто удалите инициализацию listener
из вашего фрагмента onAttach
если у вас нет взаимодействия между фрагментом и активностью
Ответ 10
Просто перейдите в свой фрагмент Activity и удалите весь метод..... вместо этого на метод createview.
ваш фрагмент имеет только метод oncreateview.
//только этот метод реализует другой метод delete
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View rootView = inflater.inflate(R.layout.fragment_main, container, false);
return rootView;
}
и убедитесь, что ваш макет - это демо для u.
Ответ 11
Я хотел бы добавить уничтожение слушателя, когда фрагмент отсоединен от действия или уничтожен.
@Override
public void onDetach() {
super.onDetach();
mListener = null;
}
и при использовании нового метода onStart() с Context
@Override
public void onDestroy() {
super.onDestroy();
mListener = null;
}
Ответ 12
У меня получилось удалить этот код:
@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");
}
}
Заканчивается так:
@Override
public void onAttach(Context context) {
super.onAttach(context);
}