Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

onPrepareActionMode not called when creating ActionMode

I just finished adjusting one of my apps to the new v22.1.1 support & appcompat libraries, see here and here for more details. When I did some testing, something was off with the ActionModes I'm using.

When you start a ActionMode using a startSupportActionMode() call - it does not matter if you use the now deprecated ActionBarActivity base class or the new AppCompatActivity base class - onPrepareActionMode() is not being called.

In previous versions, including v21.0.3 & v22.0.0, onPrepareActionMode() was called automatically, when the ActionMode was initially created using startSupportActionMode().

I tested it on a 2.2, 4.4.2 and 5.0 device, so it seems not to be version dependent.

Does anyone know, if this is intended behavior, that's been introduced in v22.1.1, or a bug?

I found this issue, but there's not a lot of feedback here...

Edit May 11 2015:

As mentioned in the Android issue tracker 159527, this issue not only affects the v22.1.x of appcompat and the support library, but also the 5.1 Android implementation.

Two possible temporary solutions at the moment, a general one:

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

and a 'quick and dirty' one (when you instantiate your ActionMode):

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

If you don't use appcompat (ActionBarActivity/AppCompatActivity) you need to replace startSupportActionMode() with startActionMode().

Unfortunately it's still not clear if this is intended new behavior or a bug. According to the API doc it's a bug/regression, but who knows...

like image 240
darksaga Avatar asked Apr 28 '15 18:04

darksaga


1 Answers

I have created a demo and its working fine,onPrepareActionMode gets called each time the action mode is shown. Always called after onCreateActionMode, but may be called multiple times if the mode is invalidated. [ I request any one to make a small edit. I need status bar color same as that of toolbar but dynamically, you can see unnecessary Drawyer Layout is used to achieve this effect,But if i remove drawyer layout the status bar color not changes according to toolbar color. In utility you can see by default default theme color is red, toolbar intially gets red color but status bar does not, only and only if i remove drawyer layout. I need to do this using style. ] Create a resource layout and name it = > 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> 

Create a Activity name it 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;     } } 

//Utils class having method to change theme // i created it because need to change app theme dynamically import 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> 
like image 156
DeepakPanwar Avatar answered Sep 22 '22 10:09

DeepakPanwar