web-dev-qa-db-ja.com

ActionModeの作成時にonPrepareActionModeが呼び出されない

アプリの1つを新しいv22.1.1サポートとappcompatライブラリに調整しました。詳細については here および here を参照してください。いくつかのテストを行ったとき、使用している ActionModes で何かがずれていました。

startSupportActionMode()呼び出しを使用して ActionMode を開始する場合、非推奨になった ActionBarActivity 基本クラスを使用するか、新しい AppCompatActivity 基本クラス-onPrepareActionMode()は呼び出されていません。

V21.0.3およびv22.0.0を含む以前のバージョンでは、onPrepareActionMode()を使用してActionModeが最初に作成されたときに、startSupportActionMode()が自動的に呼び出されました。

2.2、4.4.2、および5.0デバイスでテストしたため、バージョンに依存していないようです。

これが意図された動作である場合、v22.1.1で導入された、またはバグを知っている人はいますか?

私はこれを見つけました issue 、しかしここにはあまりフィードバックがありません...

2015年5月11日編集:

Android issue tracker 159527 で述べたように、この問題はappcompat v22.1.xとサポートライブラリだけでなく、5.1 Androidの実装にも影響します。

現時点で考えられる2つの一時的な解決策、一般的な解決策:

_@Override
public ActionMode startSupportActionMode(final ActionMode.Callback callback) {
  // Fix for bug https://code.google.com/p/Android/issues/detail?id=159527
  final ActionMode mode = super.startSupportActionMode(callback);
  if (mode != null) {
    mode.invalidate();
  }
  return mode;
}
_

そして、「素早くて汚い」もの(ActionModeをインスタンス化するとき):

_final ActionMode actionMode = startSupportActionMode(new MyActionMode());
if(actionMode != null) {
    actionMode.invalidate();
}
_

Appcompat(ActionBarActivity/AppCompatActivity)を使用しない場合は、startSupportActionMode()startActionMode()に置き換える必要があります。

残念ながら、これが意図した新しい動作なのかバグなのかはまだ明らかではありません。 API doc によると、それはバグ/回帰ですが、誰が知っている...

78
darksaga

デモを作成しましたが、正常に動作し、アクションモードが表示されるたびにonPrepareActionModeが呼び出されます。常にonCreateActionModeの後に呼び出されますが、モードが無効化されると複数回呼び出される場合があります。 [私は誰かに小さな編集を依頼します。ツールバーと同じステータスバーの色が必要ですが、動的に、この効果を達成するために不要なドロワーレイアウトが使用されていることがわかりますが、ドロワーレイアウトを削除すると、ツールバーの色に応じてステータスバーの色が変わりません。ユーティリティでは、デフォルトでデフォルトのテーマの色が赤であり、ツールバーは最初は赤になりますが、ステータスバーは赤になりません。ドロワーレイアウトを削除した場合のみです。スタイルを使用してこれを行う必要があります。]リソースレイアウトを作成し、名前を付けます=> action_mode_activity

<?xml version="1.0" encoding="utf-8"?>
<Android.support.v4.widget.DrawerLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:id="@+id/my_drawer_layout"
    Android:layout_width="match_parent"
    Android:layout_height="match_parent"
    Android:fitsSystemWindows="true">

        <LinearLayout 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:fitsSystemWindows="true"
            Android:orientation="vertical"
            app:insetForeground="#4000">

            <include
                Android:id="@+id/toolbar"
                layout="@layout/toolbar" />

            <EditText
                Android:id="@+id/editTextCopy"
                Android:layout_width="fill_parent"
                Android:layout_height="40dp"
                Android:layout_marginTop="19dp"
                Android:ems="10"
                Android:inputType="textMultiLine"
                Android:text="Long click to share!">

                <requestFocus />
            </EditText>
        </LinearLayout>

</Android.support.v4.widget.DrawerLayout>

ActionModeActivityという名前のアクティビティを作成します

import Android.os.Build;
import Android.os.Bundle;
import Android.support.v7.app.ActionBar;
import Android.support.v7.app.AppCompatActivity;
import Android.support.v7.widget.Toolbar;
import Android.view.ActionMode;
import Android.view.Menu;
import Android.view.MenuInflater;
import Android.view.MenuItem;
import Android.view.View;
import Android.widget.EditText;
import Android.widget.Toast;

import com.example.deepakpawar.demolearning.R;
import com.example.deepakpawar.demolearning.demo.load.recycler.Utils;

/**
 * Created by Deepak Pawar on 9/24/2015.
 */
public class ActionModeActivity extends AppCompatActivity implements View.OnLongClickListener, ActionMode.Callback {

    EditText editTextCopy;
    Android.view.ActionMode mActionMode;
    private Toolbar toolbar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Utils.onActivityCreateSetTheme(this);
        setContentView(R.layout.action_mode_activity);

        // 1. Get the editText
        editTextCopy = (EditText) findViewById(R.id.editTextCopy);

        // 2. add long-click listener
        editTextCopy.setOnLongClickListener(this);

        toolbar = (Toolbar) findViewById(R.id.toolbar);
        if (toolbar != null) {
            setSupportActionBar(toolbar);
            ActionBar actionBar = getSupportActionBar();
            actionBar.setDisplayHomeAsUpEnabled(true);
            actionBar.setHomeButtonEnabled(true);
            actionBar.setTitle("Android Students");
        }

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Lollipop) {

            getWindow().getDecorView().setSystemUiVisibility(
                    View.SYSTEM_UI_FLAG_LAYOUT_STABLE
                            | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN);
        } 
    }

    @Override
    public boolean onLongClick(View view) {

        // if actionmode is null "not started"
        if (mActionMode != null) {
            return false;
        }
        // Start the CAB
        mActionMode = this.startActionMode(this);
        view.setSelected(true);
        return true;

    }

    // 4. Called when the action mode is created; startActionMode() was called
    @Override
    public boolean onCreateActionMode(Android.view.ActionMode mode, Menu menu) {

        // Inflate a menu resource providing context menu items
        MenuInflater inflater = mode.getMenuInflater();
        inflater.inflate(R.menu.action_menu, menu);
        return true;
    }

    // 5. Called when the user click share item
    @Override
    public boolean onActionItemClicked(Android.view.ActionMode mode, MenuItem item) {
        switch (item.getItemId()) {
            case R.id.action_share:
                Toast.makeText(this, "Shared!", Toast.LENGTH_SHORT).show();

                mode.finish(); // Action picked, so close the CAB
                return true;
            default:
                return false;
        }
    }

    // 6. Called each time the action mode is shown. Always called after onCreateActionMode, but
    // may be called multiple times if the mode is invalidated.
    @Override
    public boolean onPrepareActionMode(Android.view.ActionMode mode, Menu menu) {

        Toast.makeText(ActionModeActivity.this,"onPrepareActionMode Called ",Toast.LENGTH_SHORT).show();
        return false; // Return false if nothing is done
    }

    // 7. Called when the user exits the action mode
    @Override
    public void onDestroyActionMode(Android.view.ActionMode mode) {
        mActionMode = null;
    }
}

//テーマを変更するメソッドを持つユーティリティクラス//アプリのテーマを動的に変更する必要があるため、Android.app.Activityをインポートして作成しました。

public class Utils {
    private static int sTheme;
    public final static int THEME_DEFAULT = 0;
    public final static int THEME_WHITE = 1;
    public final static int THEME_BLUE = 2;

    /**
     * Set the theme of the Activity, and restart it by creating a new Activity of the same type.
     */

    public static int getsTheme() {
        return sTheme;
    }

    public static void changeToTheme(Activity activity, int theme) {
        sTheme = theme;
        activity.recreate();
//        activity.finish();
//        activity.startActivity(new Intent(activity, activity.getClass()));
    }

    /**
     * Set the theme of the activity, according to the configuration.
     */
    public static void onActivityCreateSetTheme(Activity activity) {
        switch (sTheme) {
            default:
            case THEME_DEFAULT:
                activity.setTheme(R.style.FirstTheme);
                break;
            case THEME_WHITE:
                activity.setTheme(R.style.SecondTheme);
                break;
            case THEME_BLUE:
                activity.setTheme(R.style.Thirdheme);
                break;
        }
    }
}

v21-themes.xml

<resources>

    <style name="AppTheme" parent="AppTheme.Base">
        <item name="Android:windowContentTransitions">true</item>
        <item name="Android:windowAllowEnterTransitionOverlap">true</item>
        <item name="Android:windowAllowReturnTransitionOverlap">true</item>
        <item name="Android:windowSharedElementEnterTransition">@Android:transition/move</item>
        <item name="Android:windowSharedElementExitTransition">@Android:transition/move</item>

        <item name="Android:actionOverflowButtonStyle">@style/Widget.ActionButton.Overflow</item>
       <!-- <item name="Android:navigationBarColor">@color/PrimaryColor</item>-->

        <item name="windowActionBar">false</item>
        <item name="windowActionModeOverlay">true</item>

        <!-- To Make Navigation Drawer Fill Status Bar and become Transparent Too -->
        <item name="Android:windowDrawsSystemBarBackgrounds">true</item>
        <item name="Android:statusBarColor">@Android:color/transparent</item>


<!--//if darker status bar needed-->
       <!-- <item name="Android:windowTranslucentStatus">true</item>-->
    </style>


    <style name="AppTheme.Base" parent="Theme.AppCompat.Light.NoActionBar">
        <!-- Customize your theme here. -->
        <item name="colorPrimary">@color/PrimaryColor</item>
        <item name="colorPrimaryDark">@color/PrimaryDarkColor</item>
        <item name="colorAccent">@color/AccentColor</item>
        <item name="Android:textColorPrimary">@color/TextPrimaryColor</item>
        <item name="Android:windowBackground">@color/WindowBackground</item>
    </style>

    <style name="Widget.ActionButton.Overflow" parent="@Android:style/Widget.Holo.ActionButton.Overflow">
        <item name="Android:contentDescription">@string/accessibility_overflow</item>
    </style>


    <!-- style for the tool bar backgrounds -->
    <style name="ToolBarStyle" parent="ToolBarStyle.Base" />

    <style name="ToolBarStyle.Base" parent="">
        <item name="popupTheme">@style/ThemeOverlay.AppCompat.Light</item>
        <item name="theme">@style/ThemeOverlay.AppCompat.Dark.ActionBar</item>
    </style>

    <style name="ToolBarStyle.Event" parent="ToolBarStyle">
        <item name="titleTextAppearance">@style/TextAppearance.Widget.Event.Toolbar.Title</item>
    </style>

    <style name="TextAppearance.Widget.Event.Toolbar.Title" parent="TextAppearance.Widget.AppCompat.Toolbar.Title">
        <!--Any text styling can be done here-->
        <item name="Android:textStyle">normal</item>
        <item name="Android:textSize">18sp</item>
        <item name="Android:textColor">#000000</item>
    </style>

    <!-- Customize your theme example here. -->

    <style name="FirstTheme">
        <item name="Android:textColor">#FF0000</item>
        <item name="colorPrimary">#FF0000</item>
        <item name="colorPrimaryDark">#ff0000</item>
        <item name="colorAccent">#ff0087</item>
        <item name="Android:shadowColor">#00ccff</item>
        <item name="Android:shadowRadius">1.5</item>
        <item name="Android:shadowDy">1</item>
    </style>

    <style name="SecondTheme">
        <item name="Android:textColor">#00FF00</item>
        <item name="colorPrimary">#00FF00</item>
        <item name="colorPrimaryDark">#00FF00</item>
        <item name="colorAccent">#00FF90</item>
        <item name="Android:shadowColor">#00ccff</item>
        <item name="Android:shadowRadius">1.5</item>
        <item name="Android:shadowDy">1</item>
    </style>

    <style name="Thirdheme">
        <item name="Android:textColor">#0000F0</item>
        <item name="colorPrimary">#0000F0</item>
        <item name="colorPrimaryDark">#0000F0</item>
        <item name="colorAccent">#0090F0</item>
        <item name="Android:shadowColor">#00ccff</item>
        <item name="Android:shadowRadius">1.5</item>
        <item name="Android:shadowDy">1</item>
    </style>


    <style name="AppCompatAlertDialogStyle" parent="Theme.AppCompat.Light.Dialog.Alert">
        <item name="colorAccent">#FFCC00</item>
        <item name="Android:textColorPrimary">#FFFFFF</item>
        <item name="Android:background">#5fa3d0</item>
    </style>

</resources>
1
DeepakPanwar

同様の問題がありました。

Build.gradleの「compileSdkVersion」と「buildToolsVersion」の値を高めた後、onPrepareActionModeが呼び出されないことがわかりました。

  • compileSdkVersion:21〜26
  • buildToolsVersion:21.1.2から26.0.2

そこで、コードを(A)から(B)に移動しました。 (下記を参照してください)

それが正しい解決策かどうかはわかりませんが、うまくいきます。

これが私のコードの抜粋です。

list1 = findViewById(R.id.listView1);

list1.setMultiChoiceModeListener(new AbsListView.MultiChoiceModeListener() {

    @Override
    public boolean onPrepareActionMode(ActionMode mode, Menu menu) {
        //(A)
        //MenuItem menuItem1 = menu.findItem(R.id.menu_item1);
        //menuItem1.setVisible(false);
        return false;
    }

    @Override
    public boolean onCreateActionMode(ActionMode mode, Menu menu) {
        MenuInflater inflater = getMenuInflater();
        inflater.inflate(R.menu.menu_action_mode, menu);

        //(B)
        MenuItem menuItem1 = menu.findItem(R.id.menu_item1);
        menuItem1.setVisible(false);

        return true;
    }
0
Sankame