1 /* 2 * Copyright (C) 2010 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.view; 18 19 20 import android.annotation.StringRes; 21 import android.annotation.TestApi; 22 import android.graphics.Rect; 23 24 /** 25 * Represents a contextual mode of the user interface. Action modes can be used to provide 26 * alternative interaction modes and replace parts of the normal UI until finished. 27 * Examples of good action modes include text selection and contextual actions. 28 * <div class="special reference"> 29 * <h3>Developer Guides</h3> 30 * <p>For information about how to provide contextual actions with {@code ActionMode}, 31 * read the <a href="{@docRoot}guide/topics/ui/menus.html#context-menu">Menus</a> 32 * developer guide.</p> 33 * </div> 34 */ 35 public abstract class ActionMode { 36 37 /** 38 * The action mode is treated as a Primary mode. This is the default. 39 * Use with {@link #setType}. 40 */ 41 public static final int TYPE_PRIMARY = 0; 42 /** 43 * The action mode is treated as a Floating Toolbar. 44 * Use with {@link #setType}. 45 */ 46 public static final int TYPE_FLOATING = 1; 47 48 /** 49 * Default value to hide the action mode for 50 * {@link ViewConfiguration#getDefaultActionModeHideDuration()}. 51 */ 52 public static final int DEFAULT_HIDE_DURATION = -1; 53 54 private Object mTag; 55 private boolean mTitleOptionalHint; 56 private int mType = TYPE_PRIMARY; 57 58 /** 59 * Set a tag object associated with this ActionMode. 60 * 61 * <p>Like the tag available to views, this allows applications to associate arbitrary 62 * data with an ActionMode for later reference. 63 * 64 * @param tag Tag to associate with this ActionMode 65 * 66 * @see #getTag() 67 */ setTag(Object tag)68 public void setTag(Object tag) { 69 mTag = tag; 70 } 71 72 /** 73 * Retrieve the tag object associated with this ActionMode. 74 * 75 * <p>Like the tag available to views, this allows applications to associate arbitrary 76 * data with an ActionMode for later reference. 77 * 78 * @return Tag associated with this ActionMode 79 * 80 * @see #setTag(Object) 81 */ getTag()82 public Object getTag() { 83 return mTag; 84 } 85 86 /** 87 * Set the title of the action mode. This method will have no visible effect if 88 * a custom view has been set. 89 * 90 * @param title Title string to set 91 * 92 * @see #setTitle(int) 93 * @see #setCustomView(View) 94 */ setTitle(CharSequence title)95 public abstract void setTitle(CharSequence title); 96 97 /** 98 * Set the title of the action mode. This method will have no visible effect if 99 * a custom view has been set. 100 * 101 * @param resId Resource ID of a string to set as the title 102 * 103 * @see #setTitle(CharSequence) 104 * @see #setCustomView(View) 105 */ setTitle(@tringRes int resId)106 public abstract void setTitle(@StringRes int resId); 107 108 /** 109 * Set the subtitle of the action mode. This method will have no visible effect if 110 * a custom view has been set. 111 * 112 * @param subtitle Subtitle string to set 113 * 114 * @see #setSubtitle(int) 115 * @see #setCustomView(View) 116 */ setSubtitle(CharSequence subtitle)117 public abstract void setSubtitle(CharSequence subtitle); 118 119 /** 120 * Set the subtitle of the action mode. This method will have no visible effect if 121 * a custom view has been set. 122 * 123 * @param resId Resource ID of a string to set as the subtitle 124 * 125 * @see #setSubtitle(CharSequence) 126 * @see #setCustomView(View) 127 */ setSubtitle(@tringRes int resId)128 public abstract void setSubtitle(@StringRes int resId); 129 130 /** 131 * Set whether or not the title/subtitle display for this action mode 132 * is optional. 133 * 134 * <p>In many cases the supplied title for an action mode is merely 135 * meant to add context and is not strictly required for the action 136 * mode to be useful. If the title is optional, the system may choose 137 * to hide the title entirely rather than truncate it due to a lack 138 * of available space.</p> 139 * 140 * <p>Note that this is merely a hint; the underlying implementation 141 * may choose to ignore this setting under some circumstances.</p> 142 * 143 * @param titleOptional true if the title only presents optional information. 144 */ setTitleOptionalHint(boolean titleOptional)145 public void setTitleOptionalHint(boolean titleOptional) { 146 mTitleOptionalHint = titleOptional; 147 } 148 149 /** 150 * @return true if this action mode has been given a hint to consider the 151 * title/subtitle display to be optional. 152 * 153 * @see #setTitleOptionalHint(boolean) 154 * @see #isTitleOptional() 155 */ getTitleOptionalHint()156 public boolean getTitleOptionalHint() { 157 return mTitleOptionalHint; 158 } 159 160 /** 161 * @return true if this action mode considers the title and subtitle fields 162 * as optional. Optional titles may not be displayed to the user. 163 */ isTitleOptional()164 public boolean isTitleOptional() { 165 return false; 166 } 167 168 /** 169 * Set a custom view for this action mode. The custom view will take the place of 170 * the title and subtitle. Useful for things like search boxes. 171 * 172 * @param view Custom view to use in place of the title/subtitle. 173 * 174 * @see #setTitle(CharSequence) 175 * @see #setSubtitle(CharSequence) 176 */ setCustomView(View view)177 public abstract void setCustomView(View view); 178 179 /** 180 * Set a type for this action mode. This will affect how the system renders the action mode if 181 * it has to. 182 * 183 * @param type One of {@link #TYPE_PRIMARY} or {@link #TYPE_FLOATING}. 184 */ setType(int type)185 public void setType(int type) { 186 mType = type; 187 } 188 189 /** 190 * Returns the type for this action mode. 191 * 192 * @return One of {@link #TYPE_PRIMARY} or {@link #TYPE_FLOATING}. 193 */ getType()194 public int getType() { 195 return mType; 196 } 197 198 /** 199 * Invalidate the action mode and refresh menu content. The mode's 200 * {@link ActionMode.Callback} will have its 201 * {@link Callback#onPrepareActionMode(ActionMode, Menu)} method called. 202 * If it returns true the menu will be scanned for updated content and any relevant changes 203 * will be reflected to the user. 204 */ invalidate()205 public abstract void invalidate(); 206 207 /** 208 * Invalidate the content rect associated to this ActionMode. This only makes sense for 209 * action modes that support dynamic positioning on the screen, and provides a more efficient 210 * way to reposition it without invalidating the whole action mode. 211 * 212 * @see Callback2#onGetContentRect(ActionMode, View, Rect) . 213 */ invalidateContentRect()214 public void invalidateContentRect() {} 215 216 /** 217 * Hide the action mode view from obstructing the content below for a short duration. 218 * This only makes sense for action modes that support dynamic positioning on the screen. 219 * If this method is called again before the hide duration expires, the later hide call will 220 * cancel the former and then take effect. 221 * NOTE that there is an internal limit to how long the mode can be hidden for. It's typically 222 * about a few seconds. 223 * 224 * @param duration The number of milliseconds to hide for. 225 * @see #DEFAULT_HIDE_DURATION 226 */ hide(long duration)227 public void hide(long duration) {} 228 229 /** 230 * Finish and close this action mode. The action mode's {@link ActionMode.Callback} will 231 * have its {@link Callback#onDestroyActionMode(ActionMode)} method called. 232 */ finish()233 public abstract void finish(); 234 235 /** 236 * Returns the menu of actions that this action mode presents. 237 * @return The action mode's menu. 238 */ getMenu()239 public abstract Menu getMenu(); 240 241 /** 242 * Returns the current title of this action mode. 243 * @return Title text 244 */ getTitle()245 public abstract CharSequence getTitle(); 246 247 /** 248 * Returns the current subtitle of this action mode. 249 * @return Subtitle text 250 */ getSubtitle()251 public abstract CharSequence getSubtitle(); 252 253 /** 254 * Returns the current custom view for this action mode. 255 * @return The current custom view 256 */ getCustomView()257 public abstract View getCustomView(); 258 259 /** 260 * Returns a {@link MenuInflater} with the ActionMode's context. 261 */ getMenuInflater()262 public abstract MenuInflater getMenuInflater(); 263 264 /** 265 * Called when the window containing the view that started this action mode gains or loses 266 * focus. 267 * 268 * @param hasWindowFocus True if the window containing the view that started this action mode 269 * now has focus, false otherwise. 270 * 271 */ onWindowFocusChanged(boolean hasWindowFocus)272 public void onWindowFocusChanged(boolean hasWindowFocus) {} 273 274 /** 275 * Returns whether the UI presenting this action mode can take focus or not. 276 * This is used by internal components within the framework that would otherwise 277 * present an action mode UI that requires focus, such as an EditText as a custom view. 278 * 279 * @return true if the UI used to show this action mode can take focus 280 * @hide Internal use only 281 */ 282 @TestApi isUiFocusable()283 public boolean isUiFocusable() { 284 return true; 285 } 286 287 /** 288 * Callback interface for action modes. Supplied to 289 * {@link View#startActionMode(Callback)}, a Callback 290 * configures and handles events raised by a user's interaction with an action mode. 291 * 292 * <p>An action mode's lifecycle is as follows: 293 * <ul> 294 * <li>{@link Callback#onCreateActionMode(ActionMode, Menu)} once on initial 295 * creation</li> 296 * <li>{@link Callback#onPrepareActionMode(ActionMode, Menu)} after creation 297 * and any time the {@link ActionMode} is invalidated</li> 298 * <li>{@link Callback#onActionItemClicked(ActionMode, MenuItem)} any time a 299 * contextual action button is clicked</li> 300 * <li>{@link Callback#onDestroyActionMode(ActionMode)} when the action mode 301 * is closed</li> 302 * </ul> 303 */ 304 public interface Callback { 305 /** 306 * Called when action mode is first created. The menu supplied will be used to 307 * generate action buttons for the action mode. 308 * 309 * @param mode ActionMode being created 310 * @param menu Menu used to populate action buttons 311 * @return true if the action mode should be created, false if entering this 312 * mode should be aborted. 313 */ onCreateActionMode(ActionMode mode, Menu menu)314 public boolean onCreateActionMode(ActionMode mode, Menu menu); 315 316 /** 317 * Called to refresh an action mode's action menu whenever it is invalidated. 318 * 319 * @param mode ActionMode being prepared 320 * @param menu Menu used to populate action buttons 321 * @return true if the menu or action mode was updated, false otherwise. 322 */ onPrepareActionMode(ActionMode mode, Menu menu)323 public boolean onPrepareActionMode(ActionMode mode, Menu menu); 324 325 /** 326 * Called to report a user click on an action button. 327 * 328 * @param mode The current ActionMode 329 * @param item The item that was clicked 330 * @return true if this callback handled the event, false if the standard MenuItem 331 * invocation should continue. 332 */ onActionItemClicked(ActionMode mode, MenuItem item)333 public boolean onActionItemClicked(ActionMode mode, MenuItem item); 334 335 /** 336 * Called when an action mode is about to be exited and destroyed. 337 * 338 * @param mode The current ActionMode being destroyed 339 */ onDestroyActionMode(ActionMode mode)340 public void onDestroyActionMode(ActionMode mode); 341 } 342 343 /** 344 * Extension of {@link ActionMode.Callback} to provide content rect information. This is 345 * required for ActionModes with dynamic positioning such as the ones with type 346 * {@link ActionMode#TYPE_FLOATING} to ensure the positioning doesn't obscure app content. If 347 * an app fails to provide a subclass of this class, a default implementation will be used. 348 */ 349 public static abstract class Callback2 implements ActionMode.Callback { 350 351 /** 352 * Called when an ActionMode needs to be positioned on screen, potentially occluding view 353 * content. Note this may be called on a per-frame basis. 354 * 355 * @param mode The ActionMode that requires positioning. 356 * @param view The View that originated the ActionMode, in whose coordinates the Rect should 357 * be provided. 358 * @param outRect The Rect to be populated with the content position. Use this to specify 359 * where the content in your app lives within the given view. This will be used 360 * to avoid occluding the given content Rect with the created ActionMode. 361 */ onGetContentRect(ActionMode mode, View view, Rect outRect)362 public void onGetContentRect(ActionMode mode, View view, Rect outRect) { 363 if (view != null) { 364 outRect.set(0, 0, view.getWidth(), view.getHeight()); 365 } else { 366 outRect.set(0, 0, 0, 0); 367 } 368 } 369 370 } 371 } 372