วิธีการใช้ OnFragmentInteractionListener


151

ฉันมีแอพตัวช่วยสร้างพร้อมลิ้นชักนำทางใน android studio 0.8.2

ฉันได้สร้างส่วนและเพิ่มด้วย newInstance () และฉันได้รับข้อผิดพลาดนี้:

com.domain.myapp E / AndroidRuntime﹕ ข้อยกเว้นทาง FATAL: หลัก 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));
    }
}

}

คำตอบ:


120

คำตอบที่โพสต์ที่นี่ไม่ได้ช่วย แต่ลิงค์ต่อไปนี้ทำ:

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

7
โปรดทราบว่าonAttach(Activity activity);เลิกใช้แล้วและแทนที่ด้วยonAttach(Context context)
EpicPandaForce

1
@EpicPandaForce แน่นอนคุณพูดถูกฉันอัปเดตโพสต์ของฉันเพื่อสะท้อนให้เห็นว่า
meda

1
มีเหตุผลใดบ้างที่ทำให้มันย้ายจาก onAttach ไปยัง onStart? ฉันยังสามารถวางมันลงบนแค่เพียงใช้บริบทแทนกิจกรรมได้หรือไม่
Louis Tsai

ฉันคิดว่าเพียงแค่ใช้onAttach(context)จะทำงานได้ดี
EpicPandaForce

212

สำหรับบรรดาของคุณที่ยังไม่เข้าใจหลังจากอ่านคำตอบ @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: ในระยะสั้นวิธีนี้สามารถใช้สำหรับการสื่อสารระหว่างชิ้นส่วน สำหรับผู้ที่ต้องการทราบข้อมูลเพิ่มเติมเกี่ยวกับวิธีนี้โปรดดูที่ลิงก์นี้


10
กับรุ่นปัจจุบันของ SDK ใน Android สตูดิโอก็ต้องคุณจะใช้onFragmentIntereactior(Uri)วิธีการที่ไม่ได้กล่าวถึงในใด ๆ ของคำตอบอื่น ๆ +1

2
ขอบคุณมาก!
ofir_aghai

ขอบคุณที่พูดถึงสิ่งนี้ ช่วยฉันมาก
hablema

2
ต้องมี? คุณสามารถลบรหัสที่เกี่ยวข้องกับผู้ฟังได้ ... หากคุณมีแฟรกเมนต์ที่ไม่จำเป็นต้องโต้ตอบกับแฟรกเมนต์อื่นผู้ฟังเหล่านั้นจะไร้ประโยชน์
ติดตาม

4
ง่ายต่อการติดตามและเข้าใจมากกว่าคำตอบที่ยอมรับได้
jerrythebum

44

ดูที่สร้างFragmentขึ้นอัตโนมัติโดย Android Studio ของคุณ เมื่อคุณสร้างใหม่Fragmentสตูดิโอ stubbed รหัสสำหรับคุณ OnFragmentInteractionListenerที่ด้านล่างของแม่แบบสร้างขึ้นโดยอัตโนมัติมีความหมายอินเตอร์เฟซชั้นในที่เรียกว่า คุณActivityต้องใช้อินเทอร์เฟซนี้ นี่เป็นรูปแบบที่แนะนำให้คุณFragmentแจ้งActivityเหตุการณ์ของคุณเพื่อให้สามารถดำเนินการที่เหมาะสมเช่นโหลดอีกFragmentครั้ง ดูหน้านี้สำหรับรายละเอียดค้นหาส่วน "การสร้างการเรียกกลับเหตุการณ์สำหรับกิจกรรม": http://developer.android.com/guide/components/fragments.html


ในเอกสารแสดงให้เห็นถึงวิธีการใช้ฟังในส่วน (ซึ่งถูกสร้างขึ้นโดยตัวช่วยสร้าง) แต่ไม่ได้อยู่ใน mainactivity ซึ่งทำให้แอปผิดพลาด
Mario M

3
ไม่แน่นอน เอกสาร (และรหัสที่สร้างขึ้น) กำหนดอินเตอร์เฟซและตรวจสอบกับมันเมื่อถูกแนบไปกับActivity Fragmentความผิดพลาดที่คุณเห็นนั้นเป็นเพราะคุณActivityยังไม่ได้ใช้งานอินเทอร์เฟซ คุณจำเป็นต้องเข้าไปในของคุณActivityและเพิ่มimplements YourFragment.OnFragmentInteractionListenerแล้วเพิ่มการใช้งานของวิธีการที่กำหนดไว้ในส่วนต่อประสาน
Larry Schiefer

ตกลง แต่ฉันไม่ทราบวิธีการเพิ่มการใช้งานเพราะมันไม่มีที่ไหนเลยในเอกสารประกอบ sdk ของ Android
Mario M

1
มันไม่ได้เป็นส่วนหนึ่งของ SDK แต่เป็นแนวปฏิบัติที่ดีที่สุด รหัสเทมเพลตที่สร้างโดยตัวช่วยสร้างเป็นเพียงการวางรากฐานสำหรับคุณ อินเทอร์เฟซonFragmentInteraction(Uri uri)เป็นเพียงแค่ส่วนที่เหลือ คุณสามารถทำวิธีนี้ได้ทุกอย่างที่คุณต้องการและActivityความต้องการของคุณที่จะใช้มัน ดูว่าสิ่งนี้จะช่วยให้
Larry Schiefer

3
คำใบ้นี้ช่วยได้หลายชั่วโมง ขณะที่สร้างแฟรกเมนต์ UN-check "รวมวิธีการแฟรกเมนต์จากโรงงาน" และ "รวมการเรียกกลับส่วนต่อประสาน" และคุณไม่ต้องใช้ OnFragmentInteractionListener ฉันใช้ Android studio 1.3.2 กับ Java sdk 8 Android 6.0 (API 23) และ sdk-platform 23 คือ ขอบคุณ Larry Schiefer
ผู้เรียน

28

สำหรับบรรดาของคุณที่เข้าชมหน้านี้กำลังมองหาการชี้แจงเพิ่มเติมเกี่ยวกับข้อผิดพลาดนี้ในกรณีของฉันกิจกรรมที่ทำให้การเรียกไปยังแฟรกเมนต์จำเป็นต้องมี 2 การดำเนินการในกรณีนี้เช่นนี้:

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

9

คุณควรลองลบรหัสต่อไปนี้ออกจากแฟรกเมนต์ของคุณ

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

อินเตอร์เฟส / ผู้ฟังเป็นค่าเริ่มต้นที่สร้างขึ้นเพื่อให้กิจกรรมและชิ้นส่วนของคุณสามารถสื่อสารได้ง่ายขึ้น


2
นี่เป็นจุดที่ดีมากเนื่องจากผู้ฟังไม่จำเป็นต้องใช้แอพเริ่มต้นส่วนใหญ่
รหัส - ฝึกงาน

5

นอกจากคำตอบของ @ 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 ของคุณ


5

กับฉันมันทำงานลบรหัสนี้:

@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);
}

4

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คุณถ้าคุณไม่ได้มีปฏิสัมพันธ์ส่วนกิจกรรมใด ๆ


3

แทนที่จะใช้บริบทของกิจกรรมมันใช้งานได้สำหรับฉัน

@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

เพียงแค่ภาคผนวก:

OnFragmentInteractionListener จัดการการสื่อสารระหว่าง Activity และ Fragment โดยใช้อินเทอร์เฟซ (OnFragmentInteractionListener) และสร้างโดยค่าเริ่มต้นโดย Android Studio แต่ถ้าคุณไม่ต้องการสื่อสารกับกิจกรรมของคุณคุณก็สามารถทำได้

เป้าหมายคือคุณสามารถแนบแฟรกเมนต์ของคุณกับหลาย ๆ กิจกรรมและยังคงใช้วิธีการสื่อสารแบบเดียวกัน (ทุกกิจกรรมสามารถมี OnFragmentInteractionListener สำหรับแต่ละแฟรกเมนต์ของตัวเอง)

แต่ถ้าฉันมั่นใจว่าชิ้นส่วนของฉันจะถูกแนบกับกิจกรรมประเภทเดียวและฉันต้องการสื่อสารกับกิจกรรมนั้นหรือไม่?

จากนั้นถ้าคุณไม่ต้องการใช้ OnFragmentInteractionListener เนื่องจากความฟุ้งซ่านคุณสามารถเข้าถึงวิธีการกิจกรรมของคุณโดยใช้:

((MyActivityClass) getActivity()).someMethod()

แม้ว่านี่จะใช้งานได้ในกรณีส่วนใหญ่บางครั้ง getActivity () สามารถคืนค่า Null ได้หากชิ้นส่วนแยกออกจากกิจกรรมในขณะที่มีการเรียกใช้ตัวอย่างเช่นในเมธอด postExecute ของ asynctask ถ้าคุณเรียกใช้รับกิจกรรม แต่ได้ทิ้งส่วนแล้ว ก่อนที่ asyncTask จะเสร็จสมบูรณ์คุณจะได้รับข้อยกเว้นตัวชี้ null ด้วยเหตุนี้เอกสาร Android โดยเฉพาะบอกว่าจะใช้ส่วนติดต่อผู้ฟังส่วนการโต้ตอบ
MichaelStoddart

2

เพียงไปที่กิจกรรมชิ้นส่วนของคุณและลบวิธีการทั้งหมด ..... แทนบนวิธีการ createview

ส่วนของคุณมีเพียงวิธี oncreateview เท่านั้น

// เฉพาะวิธีนี้ใช้วิธีอื่น ๆ ในการลบ

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

และตรวจสอบให้แน่ใจว่าเลย์เอาต์ของคุณเป็นตัวอย่างสำหรับคุณ


ขอบคุณ ... หากต้องการบางสิ่งบางอย่าง kalpeshnikam1080@gmail.com อีเมลแบบเลื่อน
Kalpesh A. Nikam

1

ฉันต้องการเพิ่มการทำลายของผู้ฟังเมื่อส่วนแยกออกจากกิจกรรมหรือถูกทำลาย

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

และเมื่อใช้เมธอด onStart () ใหม่พร้อมบริบท

@Override
public void onDestroy() {
    super.onDestroy();
    mListener = null;
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.