mirror of
https://github.com/material-components/material-components-android.git
synced 2026-01-17 10:21:51 +08:00
309 lines
12 KiB
Java
309 lines
12 KiB
Java
/*
|
|
* Copyright (C) 2019 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package com.google.android.material.badge;
|
|
|
|
import com.google.android.material.R;
|
|
|
|
import android.content.Context;
|
|
import android.content.res.Resources;
|
|
import android.graphics.Rect;
|
|
import android.os.Build.VERSION;
|
|
import android.os.Build.VERSION_CODES;
|
|
import androidx.appcompat.view.menu.ActionMenuItemView;
|
|
import androidx.appcompat.widget.Toolbar;
|
|
import android.util.Log;
|
|
import android.util.SparseArray;
|
|
import android.view.View;
|
|
import android.widget.FrameLayout;
|
|
import androidx.annotation.IdRes;
|
|
import androidx.annotation.NonNull;
|
|
import androidx.annotation.Nullable;
|
|
import androidx.annotation.VisibleForTesting;
|
|
import androidx.core.view.AccessibilityDelegateCompat;
|
|
import androidx.core.view.ViewCompat;
|
|
import androidx.core.view.accessibility.AccessibilityNodeInfoCompat;
|
|
import com.google.android.material.internal.ParcelableSparseArray;
|
|
import com.google.android.material.internal.ToolbarUtils;
|
|
|
|
/**
|
|
* Utility class for {@link BadgeDrawable}.
|
|
*
|
|
* <p>Warning: This class is experimental and the APIs are subject to change.
|
|
*/
|
|
@ExperimentalBadgeUtils
|
|
public class BadgeUtils {
|
|
|
|
private static final String LOG_TAG = "BadgeUtils";
|
|
|
|
private BadgeUtils() {
|
|
// Private constructor to prevent unwanted construction.
|
|
}
|
|
|
|
/**
|
|
* Updates a badge's bounds using its center coordinate, {@code halfWidth} and {@code halfHeight}.
|
|
*
|
|
* @param rect Holds rectangular coordinates of the badge's bounds.
|
|
* @param centerX A badge's center x coordinate.
|
|
* @param centerY A badge's center y coordinate.
|
|
* @param halfWidth Half of a badge's width.
|
|
* @param halfHeight Half of a badge's height.
|
|
*/
|
|
public static void updateBadgeBounds(
|
|
@NonNull Rect rect, float centerX, float centerY, float halfWidth, float halfHeight) {
|
|
rect.set(
|
|
(int) (centerX - halfWidth),
|
|
(int) (centerY - halfHeight),
|
|
(int) (centerX + halfWidth),
|
|
(int) (centerY + halfHeight));
|
|
}
|
|
|
|
public static void attachBadgeDrawable(
|
|
@NonNull BadgeDrawable badgeDrawable, @NonNull View anchor) {
|
|
attachBadgeDrawable(badgeDrawable, anchor, /* customBadgeParent */ null);
|
|
}
|
|
|
|
/**
|
|
* Attaches a BadgeDrawable to its associated anchor and update the BadgeDrawable's coordinates
|
|
* based on the anchor. The BadgeDrawable will be added as a view overlay as default. If it has a
|
|
* FrameLayout custom parent that is an ancestor of the anchor, then the BadgeDrawable will be set
|
|
* as the foreground of that.
|
|
*/
|
|
public static void attachBadgeDrawable(
|
|
@NonNull BadgeDrawable badgeDrawable,
|
|
@NonNull View anchor,
|
|
@Nullable FrameLayout customBadgeParent) {
|
|
setBadgeDrawableBounds(badgeDrawable, anchor, customBadgeParent);
|
|
|
|
if (badgeDrawable.getCustomBadgeParent() != null) {
|
|
badgeDrawable.getCustomBadgeParent().setForeground(badgeDrawable);
|
|
} else {
|
|
anchor.getOverlay().add(badgeDrawable);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A convenience method to attach a BadgeDrawable to the specified menu item on a toolbar, update
|
|
* the BadgeDrawable's coordinates based on its anchor and adjust the BadgeDrawable's offset so it
|
|
* is not clipped off by the toolbar.
|
|
*
|
|
* <p>Menu item views are reused by the menu, so any structural changes to the menu may require
|
|
* detaching the BadgeDrawable and re-attaching it to the correct item.
|
|
*/
|
|
public static void attachBadgeDrawable(
|
|
@NonNull BadgeDrawable badgeDrawable, @NonNull Toolbar toolbar, @IdRes int menuItemId) {
|
|
attachBadgeDrawable(badgeDrawable, toolbar, menuItemId, null /*customBadgeParent */);
|
|
}
|
|
|
|
/**
|
|
* Attaches a BadgeDrawable to its associated action menu item on a toolbar, update the
|
|
* BadgeDrawable's coordinates based on this anchor and adjust the BadgeDrawable's offset so it is
|
|
* not clipped off by the toolbar. The BadgeDrawable will be added as a view overlay as default.
|
|
* If it has a FrameLayout custom parent that is an ancestor of the anchor, then the BadgeDrawable
|
|
* will be set as the foreground of that.
|
|
*
|
|
* <p>Menu item views are reused by the menu, so any structural changes to the menu may require
|
|
* detaching the BadgeDrawable and re-attaching it to the correct item.
|
|
*/
|
|
public static void attachBadgeDrawable(
|
|
@NonNull final BadgeDrawable badgeDrawable,
|
|
@NonNull final Toolbar toolbar,
|
|
@IdRes final int menuItemId,
|
|
@Nullable final FrameLayout customBadgeParent) {
|
|
|
|
toolbar.post(
|
|
new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
ActionMenuItemView menuItemView =
|
|
ToolbarUtils.getActionMenuItemView(toolbar, menuItemId);
|
|
if (menuItemView != null) {
|
|
setToolbarOffset(badgeDrawable, toolbar.getResources());
|
|
BadgeUtils.attachBadgeDrawable(badgeDrawable, menuItemView, customBadgeParent);
|
|
attachBadgeContentDescription(badgeDrawable, menuItemView);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
private static void attachBadgeContentDescription(
|
|
@NonNull final BadgeDrawable badgeDrawable, @NonNull View view) {
|
|
if (VERSION.SDK_INT >= VERSION_CODES.Q && ViewCompat.hasAccessibilityDelegate(view)) {
|
|
ViewCompat.setAccessibilityDelegate(
|
|
view,
|
|
new AccessibilityDelegateCompat(view.getAccessibilityDelegate()) {
|
|
@Override
|
|
public void onInitializeAccessibilityNodeInfo(
|
|
View host, AccessibilityNodeInfoCompat info) {
|
|
super.onInitializeAccessibilityNodeInfo(host, info);
|
|
info.setContentDescription(getBadgeAnchorContentDescription(view, badgeDrawable));
|
|
}
|
|
});
|
|
} else {
|
|
ViewCompat.setAccessibilityDelegate(
|
|
view,
|
|
new AccessibilityDelegateCompat() {
|
|
@Override
|
|
public void onInitializeAccessibilityNodeInfo(
|
|
View host, AccessibilityNodeInfoCompat info) {
|
|
super.onInitializeAccessibilityNodeInfo(host, info);
|
|
info.setContentDescription(getBadgeAnchorContentDescription(view, badgeDrawable));
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
private static CharSequence getBadgeAnchorContentDescription(
|
|
View anchor, BadgeDrawable badgeDrawable) {
|
|
CharSequence badgeContentDescription = badgeDrawable.getContentDescription();
|
|
return badgeContentDescription != null
|
|
? badgeContentDescription
|
|
: anchor.getContentDescription();
|
|
}
|
|
|
|
/**
|
|
* Detaches a BadgeDrawable from its associated anchor. The BadgeDrawable will be removed from its
|
|
* anchor's ViewOverlay. If it has a FrameLayout custom parent that is an ancestor of the anchor,
|
|
* then the BadgeDrawable will be removed from the parent's foreground instead.
|
|
*/
|
|
public static void detachBadgeDrawable(
|
|
@Nullable BadgeDrawable badgeDrawable, @NonNull View anchor) {
|
|
if (badgeDrawable == null) {
|
|
return;
|
|
}
|
|
if (badgeDrawable.getCustomBadgeParent() != null) {
|
|
badgeDrawable.getCustomBadgeParent().setForeground(null);
|
|
} else {
|
|
anchor.getOverlay().remove(badgeDrawable);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Detaches a BadgeDrawable from its associated action menu item on a toolbar, The BadgeDrawable
|
|
* will be removed from its anchor's ViewOverlay. If it has a FrameLayout custom parent that is an
|
|
* ancestor of the anchor, then the BadgeDrawable will be removed from the parent's foreground
|
|
* instead.
|
|
*/
|
|
public static void detachBadgeDrawable(
|
|
@Nullable BadgeDrawable badgeDrawable, @NonNull Toolbar toolbar, @IdRes int menuItemId) {
|
|
if (badgeDrawable == null) {
|
|
return;
|
|
}
|
|
ActionMenuItemView menuItemView = ToolbarUtils.getActionMenuItemView(toolbar, menuItemId);
|
|
if (menuItemView != null) {
|
|
removeToolbarOffset(badgeDrawable);
|
|
detachBadgeDrawable(badgeDrawable, menuItemView);
|
|
detachBadgeContentDescription(menuItemView);
|
|
} else {
|
|
Log.w(LOG_TAG, "Trying to remove badge from a null menuItemView: " + menuItemId);
|
|
}
|
|
}
|
|
|
|
private static void detachBadgeContentDescription(@NonNull View view) {
|
|
if (VERSION.SDK_INT >= VERSION_CODES.Q && ViewCompat.hasAccessibilityDelegate(view)) {
|
|
ViewCompat.setAccessibilityDelegate(
|
|
view,
|
|
new AccessibilityDelegateCompat(view.getAccessibilityDelegate()) {
|
|
@Override
|
|
public void onInitializeAccessibilityNodeInfo(
|
|
View host, AccessibilityNodeInfoCompat info) {
|
|
super.onInitializeAccessibilityNodeInfo(host, info);
|
|
info.setContentDescription(view.getContentDescription());
|
|
}
|
|
});
|
|
} else {
|
|
ViewCompat.setAccessibilityDelegate(view, null);
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
static void setToolbarOffset(BadgeDrawable badgeDrawable, Resources resources) {
|
|
badgeDrawable.setAdditionalHorizontalOffset(
|
|
resources.getDimensionPixelOffset(
|
|
R.dimen.mtrl_badge_toolbar_action_menu_item_horizontal_offset));
|
|
badgeDrawable.setAdditionalVerticalOffset(
|
|
resources.getDimensionPixelOffset(
|
|
R.dimen.mtrl_badge_toolbar_action_menu_item_vertical_offset));
|
|
}
|
|
|
|
@VisibleForTesting
|
|
static void removeToolbarOffset(BadgeDrawable badgeDrawable) {
|
|
badgeDrawable.setAdditionalHorizontalOffset(0);
|
|
badgeDrawable.setAdditionalVerticalOffset(0);
|
|
}
|
|
|
|
/**
|
|
* Sets the bounds of a BadgeDrawable to match the bounds of its anchor or its anchor's
|
|
* FrameLayout ancestor if it has a custom parent set.
|
|
*/
|
|
public static void setBadgeDrawableBounds(
|
|
@NonNull BadgeDrawable badgeDrawable,
|
|
@NonNull View anchor,
|
|
@Nullable FrameLayout compatBadgeParent) {
|
|
Rect badgeBounds = new Rect();
|
|
anchor.getDrawingRect(badgeBounds);
|
|
badgeDrawable.setBounds(badgeBounds);
|
|
badgeDrawable.updateBadgeCoordinates(anchor, compatBadgeParent);
|
|
}
|
|
|
|
/**
|
|
* Given a map of int keys to {@code BadgeDrawable BadgeDrawables}, creates a parcelable map of
|
|
* unique int keys to {@code BadgeDrawable.SavedState SavedStates}. Useful for state restoration.
|
|
*
|
|
* @param badgeDrawables A {@link SparseArray} that contains a map of int keys (e.g. menuItemId)
|
|
* to {@code BadgeDrawable BadgeDrawables}.
|
|
* @return A parcelable {@link SparseArray} that contains a map of int keys (e.g. menuItemId) to
|
|
* {@code BadgeDrawable.SavedState SavedStates}.
|
|
*/
|
|
@NonNull
|
|
public static ParcelableSparseArray createParcelableBadgeStates(
|
|
@NonNull SparseArray<BadgeDrawable> badgeDrawables) {
|
|
ParcelableSparseArray badgeStates = new ParcelableSparseArray();
|
|
for (int i = 0; i < badgeDrawables.size(); i++) {
|
|
int key = badgeDrawables.keyAt(i);
|
|
BadgeDrawable badgeDrawable = badgeDrawables.valueAt(i);
|
|
badgeStates.put(key, badgeDrawable != null ? badgeDrawable.getSavedState() : null);
|
|
}
|
|
return badgeStates;
|
|
}
|
|
|
|
/**
|
|
* Given a map of int keys to {@link BadgeState.State SavedStates}, creates a parcelable
|
|
* map of int keys to {@link BadgeDrawable BadgeDrawbles}. Useful for state restoration.
|
|
*
|
|
* @param context Current context
|
|
* @param badgeStates A parcelable {@link SparseArray} that contains a map of int keys (e.g.
|
|
* menuItemId) to {@link BadgeState.State states}.
|
|
* @return A {@link SparseArray} that contains a map of int keys (e.g. menuItemId)
|
|
* to {@link BadgeDrawable BadgeDrawables}.
|
|
*/
|
|
@NonNull
|
|
public static SparseArray<BadgeDrawable> createBadgeDrawablesFromSavedStates(
|
|
Context context, @NonNull ParcelableSparseArray badgeStates) {
|
|
SparseArray<BadgeDrawable> badgeDrawables = new SparseArray<>(badgeStates.size());
|
|
for (int i = 0; i < badgeStates.size(); i++) {
|
|
int key = badgeStates.keyAt(i);
|
|
BadgeState.State savedState = (BadgeState.State) badgeStates.valueAt(i);
|
|
BadgeDrawable badgeDrawable = null;
|
|
if (savedState != null) {
|
|
badgeDrawable = BadgeDrawable.createFromSavedState(context, savedState);
|
|
}
|
|
badgeDrawables.put(key, badgeDrawable);
|
|
}
|
|
return badgeDrawables;
|
|
}
|
|
}
|