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 package com.android.internal.widget; 17 18 import android.compat.annotation.UnsupportedAppUsage; 19 import android.content.Context; 20 import android.content.res.TypedArray; 21 import android.graphics.drawable.Drawable; 22 import android.os.Build; 23 import android.text.TextUtils; 24 import android.util.AttributeSet; 25 import android.view.ActionMode; 26 import android.view.LayoutInflater; 27 import android.view.View; 28 import android.view.ViewGroup; 29 import android.view.accessibility.AccessibilityEvent; 30 import android.widget.ActionMenuPresenter; 31 import android.widget.ActionMenuView; 32 import android.widget.LinearLayout; 33 import android.widget.TextView; 34 35 import com.android.internal.R; 36 import com.android.internal.view.menu.MenuBuilder; 37 38 /** 39 * @hide 40 */ 41 public class ActionBarContextView extends AbsActionBarView { 42 private static final String TAG = "ActionBarContextView"; 43 44 private CharSequence mTitle; 45 private CharSequence mSubtitle; 46 47 private View mClose; 48 private View mCustomView; 49 private LinearLayout mTitleLayout; 50 private TextView mTitleView; 51 private TextView mSubtitleView; 52 private int mTitleStyleRes; 53 private int mSubtitleStyleRes; 54 private Drawable mSplitBackground; 55 private boolean mTitleOptional; 56 private int mCloseItemLayout; 57 ActionBarContextView(Context context)58 public ActionBarContextView(Context context) { 59 this(context, null); 60 } 61 62 @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553) ActionBarContextView(Context context, AttributeSet attrs)63 public ActionBarContextView(Context context, AttributeSet attrs) { 64 this(context, attrs, com.android.internal.R.attr.actionModeStyle); 65 } 66 ActionBarContextView(Context context, AttributeSet attrs, int defStyleAttr)67 public ActionBarContextView(Context context, AttributeSet attrs, int defStyleAttr) { 68 this(context, attrs, defStyleAttr, 0); 69 } 70 ActionBarContextView( Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)71 public ActionBarContextView( 72 Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { 73 super(context, attrs, defStyleAttr, defStyleRes); 74 75 final TypedArray a = context.obtainStyledAttributes( 76 attrs, R.styleable.ActionMode, defStyleAttr, defStyleRes); 77 setBackground(a.getDrawable( 78 com.android.internal.R.styleable.ActionMode_background)); 79 mTitleStyleRes = a.getResourceId( 80 com.android.internal.R.styleable.ActionMode_titleTextStyle, 0); 81 mSubtitleStyleRes = a.getResourceId( 82 com.android.internal.R.styleable.ActionMode_subtitleTextStyle, 0); 83 84 mContentHeight = a.getLayoutDimension( 85 com.android.internal.R.styleable.ActionMode_height, 0); 86 87 mSplitBackground = a.getDrawable( 88 com.android.internal.R.styleable.ActionMode_backgroundSplit); 89 90 mCloseItemLayout = a.getResourceId( 91 com.android.internal.R.styleable.ActionMode_closeItemLayout, 92 R.layout.action_mode_close_item); 93 94 a.recycle(); 95 } 96 97 @Override onDetachedFromWindow()98 public void onDetachedFromWindow() { 99 super.onDetachedFromWindow(); 100 if (mActionMenuPresenter != null) { 101 mActionMenuPresenter.hideOverflowMenu(); 102 mActionMenuPresenter.hideSubMenus(); 103 } 104 } 105 106 @Override setSplitToolbar(boolean split)107 public void setSplitToolbar(boolean split) { 108 if (mSplitActionBar != split) { 109 if (mActionMenuPresenter != null) { 110 // Mode is already active; move everything over and adjust the menu itself. 111 final LayoutParams layoutParams = new LayoutParams(LayoutParams.WRAP_CONTENT, 112 LayoutParams.MATCH_PARENT); 113 if (!split) { 114 mMenuView = (ActionMenuView) mActionMenuPresenter.getMenuView(this); 115 mMenuView.setBackground(null); 116 final ViewGroup oldParent = (ViewGroup) mMenuView.getParent(); 117 if (oldParent != null) oldParent.removeView(mMenuView); 118 addView(mMenuView, layoutParams); 119 } else { 120 // Allow full screen width in split mode. 121 mActionMenuPresenter.setWidthLimit( 122 getContext().getResources().getDisplayMetrics().widthPixels, true); 123 // No limit to the item count; use whatever will fit. 124 mActionMenuPresenter.setItemLimit(Integer.MAX_VALUE); 125 // Span the whole width 126 layoutParams.width = LayoutParams.MATCH_PARENT; 127 layoutParams.height = mContentHeight; 128 mMenuView = (ActionMenuView) mActionMenuPresenter.getMenuView(this); 129 mMenuView.setBackground(mSplitBackground); 130 final ViewGroup oldParent = (ViewGroup) mMenuView.getParent(); 131 if (oldParent != null) oldParent.removeView(mMenuView); 132 mSplitView.addView(mMenuView, layoutParams); 133 } 134 } 135 super.setSplitToolbar(split); 136 } 137 } 138 setContentHeight(int height)139 public void setContentHeight(int height) { 140 mContentHeight = height; 141 } 142 setCustomView(View view)143 public void setCustomView(View view) { 144 if (mCustomView != null) { 145 removeView(mCustomView); 146 } 147 mCustomView = view; 148 if (view != null && mTitleLayout != null) { 149 removeView(mTitleLayout); 150 mTitleLayout = null; 151 } 152 if (view != null) { 153 addView(view); 154 } 155 requestLayout(); 156 } 157 setTitle(CharSequence title)158 public void setTitle(CharSequence title) { 159 mTitle = title; 160 initTitle(); 161 } 162 setSubtitle(CharSequence subtitle)163 public void setSubtitle(CharSequence subtitle) { 164 mSubtitle = subtitle; 165 initTitle(); 166 } 167 getTitle()168 public CharSequence getTitle() { 169 return mTitle; 170 } 171 getSubtitle()172 public CharSequence getSubtitle() { 173 return mSubtitle; 174 } 175 initTitle()176 private void initTitle() { 177 if (mTitleLayout == null) { 178 LayoutInflater inflater = LayoutInflater.from(getContext()); 179 inflater.inflate(R.layout.action_bar_title_item, this); 180 mTitleLayout = (LinearLayout) getChildAt(getChildCount() - 1); 181 mTitleView = (TextView) mTitleLayout.findViewById(R.id.action_bar_title); 182 mSubtitleView = (TextView) mTitleLayout.findViewById(R.id.action_bar_subtitle); 183 if (mTitleStyleRes != 0) { 184 mTitleView.setTextAppearance(mTitleStyleRes); 185 } 186 if (mSubtitleStyleRes != 0) { 187 mSubtitleView.setTextAppearance(mSubtitleStyleRes); 188 } 189 } 190 191 mTitleView.setText(mTitle); 192 mSubtitleView.setText(mSubtitle); 193 194 final boolean hasTitle = !TextUtils.isEmpty(mTitle); 195 final boolean hasSubtitle = !TextUtils.isEmpty(mSubtitle); 196 mSubtitleView.setVisibility(hasSubtitle ? VISIBLE : GONE); 197 mTitleLayout.setVisibility(hasTitle || hasSubtitle ? VISIBLE : GONE); 198 if (mTitleLayout.getParent() == null) { 199 addView(mTitleLayout); 200 } 201 } 202 initForMode(final ActionMode mode)203 public void initForMode(final ActionMode mode) { 204 if (mClose == null) { 205 LayoutInflater inflater = LayoutInflater.from(mContext); 206 mClose = inflater.inflate(mCloseItemLayout, this, false); 207 addView(mClose); 208 } else if (mClose.getParent() == null) { 209 addView(mClose); 210 } 211 212 View closeButton = mClose.findViewById(R.id.action_mode_close_button); 213 closeButton.setOnClickListener(new OnClickListener() { 214 public void onClick(View v) { 215 mode.finish(); 216 } 217 }); 218 219 final MenuBuilder menu = (MenuBuilder) mode.getMenu(); 220 if (mActionMenuPresenter != null) { 221 mActionMenuPresenter.dismissPopupMenus(); 222 } 223 mActionMenuPresenter = new ActionMenuPresenter(mContext); 224 mActionMenuPresenter.setReserveOverflow(true); 225 226 final LayoutParams layoutParams = new LayoutParams(LayoutParams.WRAP_CONTENT, 227 LayoutParams.MATCH_PARENT); 228 if (!mSplitActionBar) { 229 menu.addMenuPresenter(mActionMenuPresenter, mPopupContext); 230 mMenuView = (ActionMenuView) mActionMenuPresenter.getMenuView(this); 231 mMenuView.setBackground(null); 232 addView(mMenuView, layoutParams); 233 } else { 234 // Allow full screen width in split mode. 235 mActionMenuPresenter.setWidthLimit( 236 getContext().getResources().getDisplayMetrics().widthPixels, true); 237 // No limit to the item count; use whatever will fit. 238 mActionMenuPresenter.setItemLimit(Integer.MAX_VALUE); 239 // Span the whole width 240 layoutParams.width = LayoutParams.MATCH_PARENT; 241 layoutParams.height = mContentHeight; 242 menu.addMenuPresenter(mActionMenuPresenter, mPopupContext); 243 mMenuView = (ActionMenuView) mActionMenuPresenter.getMenuView(this); 244 mMenuView.setBackgroundDrawable(mSplitBackground); 245 mSplitView.addView(mMenuView, layoutParams); 246 } 247 } 248 closeMode()249 public void closeMode() { 250 if (mClose == null) { 251 killMode(); 252 return; 253 } 254 255 } 256 killMode()257 public void killMode() { 258 removeAllViews(); 259 if (mSplitView != null) { 260 mSplitView.removeView(mMenuView); 261 } 262 mCustomView = null; 263 mMenuView = null; 264 } 265 266 @Override showOverflowMenu()267 public boolean showOverflowMenu() { 268 if (mActionMenuPresenter != null) { 269 return mActionMenuPresenter.showOverflowMenu(); 270 } 271 return false; 272 } 273 274 @Override hideOverflowMenu()275 public boolean hideOverflowMenu() { 276 if (mActionMenuPresenter != null) { 277 return mActionMenuPresenter.hideOverflowMenu(); 278 } 279 return false; 280 } 281 282 @Override isOverflowMenuShowing()283 public boolean isOverflowMenuShowing() { 284 if (mActionMenuPresenter != null) { 285 return mActionMenuPresenter.isOverflowMenuShowing(); 286 } 287 return false; 288 } 289 290 @Override generateDefaultLayoutParams()291 protected ViewGroup.LayoutParams generateDefaultLayoutParams() { 292 // Used by custom views if they don't supply layout params. Everything else 293 // added to an ActionBarContextView should have them already. 294 return new MarginLayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT); 295 } 296 297 @Override generateLayoutParams(AttributeSet attrs)298 public ViewGroup.LayoutParams generateLayoutParams(AttributeSet attrs) { 299 return new MarginLayoutParams(getContext(), attrs); 300 } 301 302 @Override onMeasure(int widthMeasureSpec, int heightMeasureSpec)303 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 304 final int widthMode = MeasureSpec.getMode(widthMeasureSpec); 305 if (widthMode != MeasureSpec.EXACTLY) { 306 throw new IllegalStateException(getClass().getSimpleName() + " can only be used " + 307 "with android:layout_width=\"match_parent\" (or fill_parent)"); 308 } 309 310 final int heightMode = MeasureSpec.getMode(heightMeasureSpec); 311 if (heightMode == MeasureSpec.UNSPECIFIED) { 312 throw new IllegalStateException(getClass().getSimpleName() + " can only be used " + 313 "with android:layout_height=\"wrap_content\""); 314 } 315 316 final int contentWidth = MeasureSpec.getSize(widthMeasureSpec); 317 318 int maxHeight = mContentHeight > 0 ? 319 mContentHeight : MeasureSpec.getSize(heightMeasureSpec); 320 321 final int verticalPadding = getPaddingTop() + getPaddingBottom(); 322 int availableWidth = contentWidth - getPaddingLeft() - getPaddingRight(); 323 final int height = maxHeight - verticalPadding; 324 final int childSpecHeight = MeasureSpec.makeMeasureSpec(height, MeasureSpec.AT_MOST); 325 326 if (mClose != null) { 327 availableWidth = measureChildView(mClose, availableWidth, childSpecHeight, 0); 328 MarginLayoutParams lp = (MarginLayoutParams) mClose.getLayoutParams(); 329 availableWidth -= lp.leftMargin + lp.rightMargin; 330 } 331 332 if (mMenuView != null && mMenuView.getParent() == this) { 333 availableWidth = measureChildView(mMenuView, availableWidth, 334 childSpecHeight, 0); 335 } 336 337 if (mTitleLayout != null && mCustomView == null) { 338 if (mTitleOptional) { 339 final int titleWidthSpec = MeasureSpec.makeSafeMeasureSpec(contentWidth, 340 MeasureSpec.UNSPECIFIED); 341 mTitleLayout.measure(titleWidthSpec, childSpecHeight); 342 final int titleWidth = mTitleLayout.getMeasuredWidth(); 343 final boolean titleFits = titleWidth <= availableWidth; 344 if (titleFits) { 345 availableWidth -= titleWidth; 346 } 347 mTitleLayout.setVisibility(titleFits ? VISIBLE : GONE); 348 } else { 349 availableWidth = measureChildView(mTitleLayout, availableWidth, childSpecHeight, 0); 350 } 351 } 352 353 if (mCustomView != null) { 354 ViewGroup.LayoutParams lp = mCustomView.getLayoutParams(); 355 final int customWidthMode = lp.width != LayoutParams.WRAP_CONTENT ? 356 MeasureSpec.EXACTLY : MeasureSpec.AT_MOST; 357 final int customWidth = lp.width >= 0 ? 358 Math.min(lp.width, availableWidth) : availableWidth; 359 final int customHeightMode = lp.height != LayoutParams.WRAP_CONTENT ? 360 MeasureSpec.EXACTLY : MeasureSpec.AT_MOST; 361 final int customHeight = lp.height >= 0 ? 362 Math.min(lp.height, height) : height; 363 mCustomView.measure(MeasureSpec.makeMeasureSpec(customWidth, customWidthMode), 364 MeasureSpec.makeMeasureSpec(customHeight, customHeightMode)); 365 } 366 367 if (mContentHeight <= 0) { 368 int measuredHeight = 0; 369 final int count = getChildCount(); 370 for (int i = 0; i < count; i++) { 371 View v = getChildAt(i); 372 int paddedViewHeight = v.getMeasuredHeight() + verticalPadding; 373 if (paddedViewHeight > measuredHeight) { 374 measuredHeight = paddedViewHeight; 375 } 376 } 377 setMeasuredDimension(contentWidth, measuredHeight); 378 } else { 379 setMeasuredDimension(contentWidth, maxHeight); 380 } 381 } 382 383 @Override onLayout(boolean changed, int l, int t, int r, int b)384 protected void onLayout(boolean changed, int l, int t, int r, int b) { 385 final boolean isLayoutRtl = isLayoutRtl(); 386 int x = isLayoutRtl ? r - l - getPaddingRight() : getPaddingLeft(); 387 final int y = getPaddingTop(); 388 final int contentHeight = b - t - getPaddingTop() - getPaddingBottom(); 389 390 if (mClose != null && mClose.getVisibility() != GONE) { 391 MarginLayoutParams lp = (MarginLayoutParams) mClose.getLayoutParams(); 392 final int startMargin = (isLayoutRtl ? lp.rightMargin : lp.leftMargin); 393 final int endMargin = (isLayoutRtl ? lp.leftMargin : lp.rightMargin); 394 x = next(x, startMargin, isLayoutRtl); 395 x += positionChild(mClose, x, y, contentHeight, isLayoutRtl); 396 x = next(x, endMargin, isLayoutRtl); 397 398 } 399 400 if (mTitleLayout != null && mCustomView == null && mTitleLayout.getVisibility() != GONE) { 401 x += positionChild(mTitleLayout, x, y, contentHeight, isLayoutRtl); 402 } 403 404 if (mCustomView != null) { 405 x += positionChild(mCustomView, x, y, contentHeight, isLayoutRtl); 406 } 407 408 x = isLayoutRtl ? getPaddingLeft() : r - l - getPaddingRight(); 409 410 if (mMenuView != null) { 411 x += positionChild(mMenuView, x, y, contentHeight, !isLayoutRtl); 412 } 413 } 414 415 @Override shouldDelayChildPressedState()416 public boolean shouldDelayChildPressedState() { 417 return false; 418 } 419 420 @Override onInitializeAccessibilityEventInternal(AccessibilityEvent event)421 public void onInitializeAccessibilityEventInternal(AccessibilityEvent event) { 422 if (event.getEventType() == AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED) { 423 // Action mode started 424 event.setSource(this); 425 event.setClassName(getClass().getName()); 426 event.setPackageName(getContext().getPackageName()); 427 event.setContentDescription(mTitle); 428 } else { 429 super.onInitializeAccessibilityEventInternal(event); 430 } 431 } 432 setTitleOptional(boolean titleOptional)433 public void setTitleOptional(boolean titleOptional) { 434 if (titleOptional != mTitleOptional) { 435 requestLayout(); 436 } 437 mTitleOptional = titleOptional; 438 } 439 isTitleOptional()440 public boolean isTitleOptional() { 441 return mTitleOptional; 442 } 443 } 444