onOptionsItemSelected() 方法在 Fragment 中未被调用

4
我有一个 AppCompatActivity 和一些片段。在 AboutFragment 中,操作栏上的返回按钮不起作用。
该 Activity 没有膨胀菜单,而 Fragment 只在操作栏中有一个返回按钮。
片段菜单可见,但点击时返回按钮根本没有反应。 AboutFragment 在从操作栏中的信息图标上单击后显示。

the <code>AboutFragment</code> is getting showed on clicking the info icon from the actionbar

在单击图标时,下面的方法在MainActivity中起作用。
 @Override
    public void onInfoSelected() {

        abf = (AboutFragment) getSupportFragmentManager().findFragmentByTag(ABOUT_FRAGMENT_TAG);

        FragmentTransaction ft = fm.beginTransaction();
        ft.setCustomAnimations(android.R.anim.slide_in_left, android.R.anim.slide_out_right,
                android.R.anim.slide_in_left, android.R.anim.slide_out_right);
       // ft.hide(pf);

        //ft.hide(cf);
        ft.hide(tlf);
        ft.show(abf);
        ft.addToBackStack(null);
        ft.commit();
    }

那么,在 AboutFragment 中的后退按钮不会调用 Fragment 内的 onOptionsItemSelected() 方法。

enter image description here

当调试时,我可以看到Fragment和Activity的onCreateOptionsMenu()都被调用了,但是在点击按钮时,无论是从Activity还是从Fragment中都没有调用onOptionsItemSelected()
我已经搜索和谷歌了两天,但是对我来说都没用。任何帮助将不胜感激。 Activity Java代码
package me.declangao.jiasazsales.app;

import android.content.Intent;
import android.os.Bundle;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentTransaction;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;

import me.declangao.jiasazsales.R;
import me.declangao.jiasazsales.model.Post;

public class MainActivity extends AppCompatActivity implements
        RecyclerViewFragment.PostListListener, PostFragment.PostListener,
        TabLayoutFragment.TabLayoutListener, SearchResultFragment.SearchResultListener,
        CommentFragment.CommentListener,AboutFragment.AboutListener {

    private static final String TAG = MainActivity.class.getSimpleName();
    public static final String TAB_LAYOUT_FRAGMENT_TAG = "TabLayoutFragment";
    public static final String POST_FRAGMENT_TAG = "PostFragment";
    public static final String COMMENT_FRAGMENT_TAG = "CommentFragment";

    public static final String ABOUT_FRAGMENT_TAG = "AboutFragment";


    private FragmentManager fm = null;
    private TabLayoutFragment tlf;
    private PostFragment pf;
    private CommentFragment cf;
    private SearchResultFragment srf;
    private AboutFragment abf;


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



        fm = getSupportFragmentManager();

        // Setup fragments
        tlf = new TabLayoutFragment();
        pf = new PostFragment();
        cf = new CommentFragment();
        srf = new SearchResultFragment();
        abf=new AboutFragment();
        FragmentTransaction ft = fm.beginTransaction();
        ft.add(android.R.id.content, abf, ABOUT_FRAGMENT_TAG);
        ft.add(android.R.id.content, pf, POST_FRAGMENT_TAG);
        ft.add(android.R.id.content, cf, COMMENT_FRAGMENT_TAG);
        ft.add(android.R.id.content, tlf, TAB_LAYOUT_FRAGMENT_TAG);


        ft.hide(pf);
        ft.hide(cf);
        ft.hide(abf);
        ft.show(tlf);
        ft.commit();
    }

    /**
     * Invoked when a post in the list is selected
     *
     * @param post Selected Post object
     */
    @Override
    public void onPostSelected(Post post, boolean isSearch) {
        // Find the fragment in order to set it up later
        pf = (PostFragment) getSupportFragmentManager().findFragmentByTag(POST_FRAGMENT_TAG);

        // Set necessary arguments
        Bundle args = new Bundle();
        args.putInt("id", post.getId());
        args.putString("title", post.getTitle());
        args.putString("date", post.getDate());
        args.putString("author", post.getAuthor());
        args.putString("content", post.getContent());
        args.putString("url", post.getUrl());
        //args.putString("thumbnailUrl", post.getThumbnailUrl());
        args.putString("featuredImage", post.getFeaturedImageUrl());

        // Configure PostFragment to display the right post
        pf.setUIArguments(args);

        // Show the fragment
        FragmentTransaction ft = fm.beginTransaction();
        ft.setCustomAnimations(android.R.anim.slide_in_left, android.R.anim.slide_out_right,
                android.R.anim.slide_in_left, android.R.anim.slide_out_right);
        if (!isSearch) { // Hide TabLayoutFragment if this is not search result
            ft.hide(tlf);

        } else { // Otherwise, hide the search result, ie. SearchResultFragment.
            ft.hide(srf);
        }
        ft.show(pf);
        ft.addToBackStack(null);
        ft.commit();
    }
    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
            case android.R.id.home:
                //do something here like
Log.e("menu Test","rom Main activity");
                int backStackEntryCount
                        =getSupportFragmentManager().getBackStackEntryCount();

                if (backStackEntryCount > 0) {

                    getSupportFragmentManager().popBackStack();

                }

                return true;
        }
        return false;
    }
    /**
     * Invoked when a search query is submitted
     *
     * @param query Selected Post object
     */
    @Override
    public void onSearchSubmitted(String query) {
        FragmentTransaction ft = fm.beginTransaction();
        ft.setCustomAnimations(android.R.anim.slide_in_left, android.R.anim.slide_out_right,
                android.R.anim.slide_in_left, android.R.anim.slide_out_right);

        // Send query to fragment using factory method
        srf = SearchResultFragment.newInstance(query);
        ft.add(android.R.id.content, srf);
        ft.hide(tlf);
        ft.addToBackStack(null);
        ft.commit();
    }


    @Override
    public void onInfoSelected() {

        abf = (AboutFragment) getSupportFragmentManager().findFragmentByTag(ABOUT_FRAGMENT_TAG);

        FragmentTransaction ft = fm.beginTransaction();
        ft.setCustomAnimations(android.R.anim.slide_in_left, android.R.anim.slide_out_right,
                android.R.anim.slide_in_left, android.R.anim.slide_out_right);
       // ft.hide(pf);

        //ft.hide(cf);
        ft.hide(tlf);
        ft.show(abf);
        ft.addToBackStack(null);
        ft.commit();
    }
    /**
     * Invoked when comment menu is selected
     *
     * @param id ID of the article, assigned by WordPress
     */
    @Override
    public void onCommentSelected(int id) {
        cf = (CommentFragment) getSupportFragmentManager().findFragmentByTag(COMMENT_FRAGMENT_TAG);
        Bundle args = new Bundle();
        args.putInt("id", id);
        // Setup CommentFragment to display the right comments page
        cf.setUIArguments(args);

        FragmentTransaction ft = fm.beginTransaction();
        ft.setCustomAnimations(android.R.anim.slide_in_left, android.R.anim.slide_out_right,
                android.R.anim.slide_in_left, android.R.anim.slide_out_right);
        ft.hide(pf);
        //ft.hide(abf);
        ft.show(cf);
        ft.addToBackStack(null);
        ft.commit();
    }

    /**
     * Intercept back button event, reset ActionBar if necessary
     */
    @Override
    public void onBackPressed() {
        resetActionBarIfApplicable();
        super.onBackPressed();
    }

    /**
     * Simulate a back button press when home is selected
     */
    @Override
    public void onHomePressed() {
        resetActionBarIfApplicable();
        fm.popBackStack();
    }

    /**
     * Reset TabLayoutFragment's ActionBar if necessary
     */
    private void resetActionBarIfApplicable() {
        Log.d(TAG, "SearchResultFragment is visible: " + srf.isHidden());
        if (srf.isVisible()) {
            tlf.resetActionBar();
        }
    }

    // Commented out coz we will let fragments handle their own Options Menus
/*
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        //Log.e("Erroraaa","aaaaa");

        return true;
    }



    @Override
    public boolean onOptionsItemSelected(MenuItem item) {

        Log.e("menu","activity: action home has clicked");
        switch (item.getItemId()){
            case android.R.id.home:

                onBackPressed();
                return false;

        }

        return super.onOptionsItemSelected(item);
    }
*/

}

AboutFragment

 package me.declangao.jiasazsales.app;

import android.app.Activity;

import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentTransaction;
import android.support.v7.widget.Toolbar;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuInflater;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;

import me.declangao.jiasazsales.R;

/**
 * Fragment to display a Info about Jiasaz company.
 * Activities that contain this fragment must implement the
 * {@link AboutFragment.AboutListener} interface
 * to handle interaction events.
 */
public class AboutFragment extends Fragment {



    private AboutListener mListener;
    private Toolbar toolbar;


    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        this.setRetainInstance(true);

        this.setHasOptionsMenu(true);

    }

//    @Override
//    public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
//        Log.e("Menu:created","Menu");
//
////        super.onCreateOptionsMenu(menu, inflater);
////        menu.clear();
//        inflater.inflate(R.menu.menu_post, menu);
//        super.onCreateOptionsMenu(menu, inflater);
//    }
    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        Log.e("Menu:selected","Menu");
        if (item.getItemId() == android.R.id.home) {
            mListener.onHomePressed();
        }
        return false;
    }


    public AboutFragment() {
        // Required empty public constructor
    }

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {

        // Inflate the layout for this fragment
        View rootView = inflater.inflate(R.layout.about_layout, container, false);

        toolbar = (Toolbar) rootView.findViewById(R.id.toolbar);
        ((MainActivity) getActivity()).setSupportActionBar(toolbar);
        ((MainActivity) getActivity()).getSupportActionBar().setHomeButtonEnabled(true);
        ((MainActivity) getActivity()).getSupportActionBar().setDisplayHomeAsUpEnabled(true);
        ((MainActivity) getActivity()).getSupportActionBar().setTitle(  "www.Jiasaz.com");
         Log.e("Menu: fragment created","onCreaate()");

        return rootView;
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        try {
            mListener = (AboutListener) activity;
        } catch (ClassCastException e) {
            throw new ClassCastException(activity.toString()
                    + " must implement AboutListener");
        }
    }

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

    /**
     * This interface must be implemented by activities that contain this
     * fragment to allow an interaction in this fragment to be communicated
     * to the activity and potentially other fragments contained in that
     * activity.
     * <p/>
     * See the Android Training lesson <a href=
     * "http://developer.android.com/training/basics/fragments/communicating.html"
     * >Communicating with Other Fragments</a> for more information.
     */
    public interface AboutListener {
       void onHomePressed();
        //void onInfoSelected();
    }




}

这是我的 AboutFragment xml 代码

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/colorAccent"
    android:layoutDirection="rtl"
    android:textDirection="rtl"

    tools:context="me.declangao.jiasazsales.app.AboutFragment">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layoutDirection="rtl"
        android:textDirection="rtl"
        android:orientation="vertical">

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:background="@color/colorPrimary"
            android:theme="@style/ThemeOverlay.AppCompat.Dark"
            android:layoutDirection="rtl"
            android:textDirection="rtl"
            />

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layoutDirection="rtl"
        android:textDirection="rtl"
        android:padding="15dp"
        android:orientation="vertical">

        <TextView
            android:id="@+id/textView2"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="20dp"
            android:layout_marginBottom="10dp"
            android:text="ئه‌م ئاپه‌ له‌لایه‌ن كۆمپانیای جیاساز دروست كراوه‌"
            android:textAppearance="@style/TextAppearance.AppCompat.Light.SearchResult.Title" />

        <ImageView
            android:id="@+id/imageView"
            android:layout_width="match_parent"
            android:layout_height="214dp"
            android:src="@drawable/jiasazlogo" />

        <TextView
            android:id="@+id/textView"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="20dp"
            android:layout_marginBottom="10dp"
            android:text="كۆمپانیای جیاساز بۆ خزمه‌تگوزاری و چاره‌سه‌ری ته‌كنه‌لۆجی، دروستكردنی وێبسایت و ئاپی مۆبایل و سیسته‌می دام و ده‌زگاكان و ماركێته‌كان"
            android:textAppearance="@style/TextAppearance.AppCompat.Light.SearchResult.Title" />

        <TextView
            android:id="@+id/textView3"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_gravity="center_horizontal"
            android:autoLink="all"
            android:clickable="true"
            android:text="@string/link"
            android:textAlignment="center"
            android:textAppearance="@style/TextAppearance.AppCompat.Body1"
            android:textSize="18sp" />

    </LinearLayout>
    </LinearLayout>

</FrameLayout>
4个回答

3

你可以直接在Toolbar上设置一个OnMenuItemClickListener实例。请记住,不应该使用Activity.setSupportActionBar(Toolbar)

public class AboutFragment extends Fragment {

    // ...

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

        View rootView = inflater.inflate(R.layout.about_layout, container, false);
        toolbar = (Toolbar) rootView.findViewById(R.id.toolbar);

        toolbar.inflateMenu(R.menu.menu_about_fragment);
        toolbar.setOnMenuItemClickListener(new Toolbar.OnMenuItemClickListener() {

            @Override
            public boolean onMenuItemClick(MenuItem item) {
                if (item.getItemId() == R.id.some_menu) {
                    // do sth...
                    return true; // event is handled.
                }
                return false;
            }
        });

        // ...
    }

    // ...
}

更新 2018年11月21日:

要为独立的工具栏启用返回按钮,请在xml文件中使用app:navigationIcon="?attr/homeAsUpIndicator"设置其默认图标:

<android.support.v7.widget.Toolbar
    android:id="@+id/toolbar"
    android:layout_width="match_parent"
    android:layout_height="?attr/actionBarSize"
    android:background="@color/colorPrimary"
    android:theme="@style/ThemeOverlay.AppCompat.Dark"
    android:layoutDirection="rtl"
    android:textDirection="rtl"
    app:navigationIcon="?attr/homeAsUpIndicator" />

然后为其设置监听器。
toolbar.setNavigationOnClickListener(new View.OnClickListener() {

    @Override
    public void onClick(View v) {
        // back button is pressed
        mListener.onHomePressed();
    }
});

你是在 toolbar.setOnMenuItemClickListener 之后调用 setSupportActionBar(toolbar) 吗? - aminography
它导致我的应用程序崩溃了。 - rabar kareem
是的,我做了,还是一样的。 - rabar kareem
1
请查看 https://github.com/aminography/FragmentToolbar 并与您自己的进行比较。 - aminography
这是一个好消息。这意味着与操作栏的冲突已经解决。现在,您可以使用 toolbar.inflateMenu 显示菜单(作为答案的第一部分),并通过 toolbar.setOnMenuItemClickListener 监听项目点击。 - aminography
显示剩余10条评论

0

你的描述很准确,这些都是平台的限制。为了解决这个问题,你需要在MainActivity中使用onOptionsItemSelected()方法,在这里插入你已经在另一个方法中使用过的代码:

abf = (AboutFragment) getSupportFragmentManager().findFragmentByTag(ABOUT_FRAGMENT_TAG);

然后,由于您拥有片段,因此可以从片段中调用某些方法,并可能获得返回值以了解是否已处理后退,以及是否需要执行其他操作...


正如您所看到的,被调用的是Activity onOptionsItemSelected()。它包含有关菜单项ID的信息。由于ID是唯一的,因此您可以知道正在发生什么,无论是来自活动还是片段。如果您需要在片段中处理它,则可以使用上面的代码。所以这个方法是有效的,我不确定您想要表达什么,可能从某些理论角度来说并不完美,但这是使其正常工作的方法,因为SDK并不像人们期望的那样工作。 - Ivan Ičin
我甚至无法在onOptionsItemSelected()活动中获取操作,它在任何地方都没有显示任何操作。 - rabar kareem

0

你可以在你的Fragment类中使用onPrepareOptionsMenu(Menu menu)来监听你的操作栏菜单...

@Override
public void onPrepareOptionsMenu(Menu menu)
{
    super.onPrepareOptionsMenu(menu);

    // to get an item of menu from action bar you can do it like...
    // this below line will find an item of menu with id action_search...
    MenuItem searchMenuItem = menu.findItem(R.id.action_search);

    // like this you can find other menu items
    // another example is
    MenuItem infoMenuItem = menu.findItem(R.id.action_info);
}

我在我的片段类中使用搜索视图来实现搜索...


只有在片段启动时才能正常工作,而不能在单击菜单项时工作。 - rabar kareem

-1

第一个原因是您没有正确实现onCreateOptionsMenu()onOptionsItemSelected() @Override方法在Activity中,所以取消注释Activity也可以调用它们各自的super方法。

第二个原因是您的onCreateOptionsMenu()代码被注释了,这不需要。重要的是必须调用super.onCreateOptionsMenu(menu, inflater);来绘制Fragment的菜单。

@Override
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
    super.onCreateOptionsMenu(menu, inflater);
}

同时将Fragment中的onOptionsItemSelected()更改为以下内容:

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    Log.e("Menu:selected","Menu");
    if (item.getItemId() == android.R.id.home) {
        mListener.onHomePressed();
        return true;
    }else{
        return super.onOptionsItemSelected(item);
    }

}

现在在Activity中将你的onCreateOptionsMenu()更改为:

@Override
    public boolean onCreateOptionsMenu(Menu menu) {
        return super.onCreateOptionsMenu(menu);
    }

onOptionsItemSelected()转化为:

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    return super.onOptionsItemSelected(item);
}

谢谢您的回复,如果您查看问题,我已经完成了这个。 - rabar kareem
已经测试过了,和之前一样,什么也没发生。onCreateOptionsMenu被调用了,但问题出在永远不会被调用的onOptionsItemSelected()函数上。 - rabar kareem
@rabarkareem 还需要取消注释 onCreateOptionsMenuonOptionsItemSelected 方法,并在它们各自的类中添加 super。已更新答案,请在相应的类中替换上述给出的方法并尝试。 - Muhammad waris
还是不行,你要我把源代码发给你吗? - rabar kareem

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接