15

アクションバー互換性ライブラリを使用しています。openOptionsMenu() 関数を使用してボタンからオプション メニューを開こうとしていますが、何もしません。

携帯電話のメニュー キーを押すと、通常どおりメニューが表示されます。ここで何が問題なのですか?

public class ReadActivity extends ActionBarActivity {

    ...

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        boolean value;
        MenuInflater menuInflater = getMenuInflater();
        menuInflater.inflate(R.menu.read, menu);
        value = super.onCreateOptionsMenu(menu);

        if (Helper.SupportsNewApi()) {
            getActionBar().hide();
        } else {
            ((View) ((LinearLayout) findViewById(R.id.actionbar_compat))
                    .getParent()).setVisibility(View.GONE);
        }

        return value;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch (item.getItemId()) {
        case android.R.id.home:
            finish();
            break;
        case R.id.menu_search:
            // Toast.makeText(this, "Tapped search", Toast.LENGTH_SHORT).show();
            break;
        case R.id.menu_bookmark:
            // selectText();
            // setFullScreen(false);
            break;
        case R.id.menu_day_night_mode:
            break;
        case R.id.menu_settings:
            break;
        case R.id.menu_zoom_in:
            showOverlay(false);
            break;
        case R.id.menu_zoom_out:
            showOverlay(false);
            break;
        case R.id.menu_table_of_contents:
            Intent tocIntent = new Intent(this, TocActivity.class);
            int GET_SECTION_REFERENCE = 1;
            startActivityForResult(tocIntent, GET_SECTION_REFERENCE);
            break;
        case R.id.menu_overflow:
            Toast.makeText(this, "Tapped overflow", Toast.LENGTH_SHORT).show();

            //closeOptionsMenu();
            openOptionsMenu(); //tried the below aswell, no results
            //getWindow().openPanel(Window.FEATURE_OPTIONS_PANEL, null);
            break;
        }
        return super.onOptionsItemSelected(item);
    }


    @Override //disable volume buttons
    public boolean onKeyDown(int keyCode, KeyEvent event) {
        if (!menuShown && (keyCode == 25 || keyCode == 24)) {
            return true;
        }
        return super.onKeyDown(keyCode, event);
    }

    @Override
    public boolean onKeyUp(int keyCode, KeyEvent event) {
        Log.d(tag, "Keycode is = "+keyCode);
        if (keyCode == 82) {
            if (!menuShown) {
                //openOptionsMenu();
                showOverlay(true);
            } else {
                showOverlay(false);
            }


                    //don't want it to open when pressing menu
            return true;
        } else if (keyCode == 4 && menuShown) {
            showOverlay(false);
            return true;
        } else if (keyCode == 25 && !menuShown) {
            prevPage();
            return true;
        } else if (keyCode == 24 && !menuShown) {
            nextPage();
            return true;
        }

        return super.onKeyUp(keyCode, event);
    }

}
4

8 に答える 8

31

Android 1.6以降で実行する必要があるアプリで、このopenOptionsMenuのことを回避しようとして同じ問題が発生していました。Werner Van Belle からの回答に従って、問題を解決するための回避策を達成できるという結論に達しました。そこで私は次のコードを思いつきました。人々がメソッドを最終としてマークしないときは常に美しいので、いつでもオーバーライドできます。アプリのターゲットを最新の API (android:targetSdkVersion="17") にすることをあきらめたくない場合に最適です。気に入っていただければ幸いです。:)

@Override
public void openOptionsMenu() {

    Configuration config = getResources().getConfiguration();

    if((config.screenLayout & Configuration.SCREENLAYOUT_SIZE_MASK) 
            > Configuration.SCREENLAYOUT_SIZE_LARGE) {

        int originalScreenLayout = config.screenLayout;
        config.screenLayout = Configuration.SCREENLAYOUT_SIZE_LARGE;
        super.openOptionsMenu();
        config.screenLayout = originalScreenLayout;

    } else {
        super.openOptionsMenu();
    }
}
于 2013-07-27T22:35:13.197 に答える
9

グーグルによるこの悲しい展開に光を当てるために。Google は明らかに、誰もが新しい ActionBar を受け入れることを望んでいます。ActionBarを古いメニュー システムよりも使いやすくすることで、それを実現できたはずです。しかし、それは彼らが移行を計画した方法ではありません。いいえ、適切な下位互換性を提供せずに古いメニューを使用できないようにすることで、プログラマーを困らせるのは理にかなっていると彼らは考えました。

以下は、optionsMenu パネルを作成することになっている com.android.internal.policy.impl から取得したコードです。ご覧のとおり、コードは単にオプション パネルの作成を拒否しています。しかし、能力は明らかにそこにあります。ですから、あなたの質問に答えるために: 忘れてください。Google は、あなたがその optionsPanel をもう使用することを望んでいません。

 // Don't open an options panel for honeycomb apps on xlarge devices.
 // (The app should be using an action bar for menu items.)
 if (st.featureId == FEATURE_OPTIONS_PANEL) {
            Context context = getContext();
            Configuration config = context.getResources().getConfiguration();
            boolean isXLarge = (config.screenLayout & Configuration.SCREENLAYOUT_SIZE_MASK) ==
                    Configuration.SCREENLAYOUT_SIZE_XLARGE;
            boolean isHoneycombApp = context.getApplicationInfo().targetSdkVersion >=
                    android.os.Build.VERSION_CODES.HONEYCOMB;

            if (isXLarge && isHoneycombApp) {
                return;
            }
        }
于 2013-06-06T15:33:32.990 に答える
4

メニューボタンの使用を強制的にブロックすることも理解できません。ただし、次のトリックは、「制限された」タイプのデバイスでメニューを表示するのに役立ちました.

まず、次のハックが必要かどうかを定義する必要があります。

boolean requireDirtyMenuButtonHack = Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB && (activity.getResources().getConfiguration().screenLayout & Configuration.SCREENLAYOUT_SIZE_XLARGE) > 0;

それで:

protected final OnClickListener mMenuButtonClickListener = new OnClickListener() {

    @Override
    public void onClick(View v) {
        if (requireDirtyMenuButtonHack) {
            Configuration config = getContext().getResources().getConfiguration();
            config.screenLayout &= ~Configuration.SCREENLAYOUT_SIZE_XLARGE;
            config.screenLayout |= Configuration.SCREENLAYOUT_SIZE_LARGE;
        }

        getActivity().openOptionsMenu();
    }
};

掃除もお忘れなく!(必要かどうかはわかりませんが、優しく遊んだほうがいいです)

public void onPrepareOptionsMenu(Menu menu) {
    if (requireDirtyMenuButtonHack) {
        Configuration config = getContext().getResources().getConfiguration();
        config.screenLayout &= ~Configuration.SCREENLAYOUT_SIZE_LARGE;
        config.screenLayout |= Configuration.SCREENLAYOUT_SIZE_XLARGE;
    }

    //do the preparing...
}
于 2013-09-05T10:16:06.043 に答える
3

これがどれほど関連性があるかはわかりませんが、私が見つけたこのフォーラムの投稿は、あなたと同じ質問に答えているようです。

この投稿があなたの問題を解決するのに十分に関連していることを願っています。

幸運を!

于 2012-04-19T14:57:37.667 に答える
2

マニフェストの android:targetSdkVersion="10" が役に立ちました。openOptionsMenu() は ICS+ で期待どおりに動作するようになりました。さらに、画面の下部 (デバイス ボタン パネル上) に「オーバーフロー」メニュー ボタンが表示されます。

ps: NoTitleBar テーマ (SDK 11 以降の場合は NoActionBar) + Jake Wharton によるViewPagerIndicatorを使用します。

于 2012-09-24T21:52:37.833 に答える
2

これは私にとってはうまくいきました。私のコードはあなたのものとかなり似ています。私がやりたいことは、アクションバーのボタンからオーバーフローメニューを開くことです:

public boolean onOptionsItemSelected(MenuItem item) {  
        switch (item.getItemId()) {  
        case R.id.menu_home_about:  
            dialog = new Dialog(this);
            dialog.requestWindowFeature(Window.FEATURE_NO_TITLE);
            dialog.setContentView(R.layout.customdialog);
            dialog.show();
            break;

            default:

        case R.id.menu_home_refresh:
            updateLists(true);
            break;

        case R.id.menu_home_menu:
            new Handler().postDelayed(new Runnable() {
                public void run() {                
                    openOptionsMenu();        
                }
            }, 0); 
            return true;

        }
        return false;  
    }
于 2013-08-30T16:43:20.780 に答える
0

アクションバーの右側にボタンを表示したいということですか?

これが私のやり方です:

res/メニュー/main.xml

<menu xmlns:android="http://schemas.android.com/apk/res/android">
<item android:id="@+id/menu_share"
    android:title="Logout"
    android:icon="@android:drawable/ic_lock_power_off"
    android:orderInCategory="1"
    android:showAsAction="always" />
</menu>

アクティビティ
1) ActionBarActivity に注意してください。2) onCreateOptionsMenu の MenuInflater 3) onOptionsItemsSelected ( super.onOptionsItemSelected(item) を返す必要があると思います)

public class BaseActivity extends ActionBarActivity {
    ....

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getMenuInflater();
    inflater.inflate(R.menu.main, menu);

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

    switch (item.getItemId()) {
    case R.id.menu_share:


       //Do something
        break;
    }

    return super.onOptionsItemSelected(item);
}
于 2012-04-03T15:30:30.477 に答える