/* * 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 com.google.android.material.badge.BadgeDrawable.SavedState; import com.google.android.material.internal.ParcelableSparseArray; import com.google.android.material.internal.ToolbarUtils; /** * Utility class for {@link BadgeDrawable}. * *

Warning: This class is experimental and the APIs are subject to change. */ @ExperimentalBadgeUtils public class BadgeUtils { public static final boolean USE_COMPAT_PARENT = VERSION.SDK_INT < VERSION_CODES.JELLY_BEAN_MR2; 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. For API 18+, the BadgeDrawable will be added as a view overlay. For * pre-API 18, the BadgeDrawable will be set as the foreground of a FrameLayout that is an * ancestor of the anchor. */ 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 { if (USE_COMPAT_PARENT) { throw new IllegalArgumentException("Trying to reference null customBadgeParent"); } 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. */ 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. For API 18+, the BadgeDrawable will be added as a view overlay. * For pre-API 18, the BadgeDrawable will be set as the foreground of a FrameLayout that is an * ancestor of the anchor. */ 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); } } }); } /** * Detaches a BadgeDrawable from its associated anchor. For API 18+, the BadgeDrawable will be * removed from its anchor's ViewOverlay. For pre-API 18, the BadgeDrawable will be removed from * the foreground of a FrameLayout that is an ancestor of the anchor. */ public static void detachBadgeDrawable( @Nullable BadgeDrawable badgeDrawable, @NonNull View anchor) { if (badgeDrawable == null) { return; } if (USE_COMPAT_PARENT || badgeDrawable.getCustomBadgeParent() != null) { badgeDrawable.getCustomBadgeParent().setForeground(null); } else { anchor.getOverlay().remove(badgeDrawable); } } /** * Detaches a BadgeDrawable from its associated action menu item on a toolbar, For API 18+, the * BadgeDrawable will be removed from its anchor's ViewOverlay. For pre-API 18, the BadgeDrawable * will be removed from the foreground of a FrameLayout that is an ancestor of the anchor. */ 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); } else { Log.w(LOG_TAG, "Trying to remove badge from a null menuItemView: " + menuItemId); } } @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 (for API 18+) or its * anchor's FrameLayout ancestor (pre-API 18). */ 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 badgeDrawables) { ParcelableSparseArray badgeStates = new ParcelableSparseArray(); for (int i = 0; i < badgeDrawables.size(); i++) { int key = badgeDrawables.keyAt(i); BadgeDrawable badgeDrawable = badgeDrawables.valueAt(i); if (badgeDrawable == null) { throw new IllegalArgumentException("badgeDrawable cannot be null"); } badgeStates.put(key, badgeDrawable.getSavedState()); } return badgeStates; } /** * Given a map of int keys to {@link BadgeDrawable.SavedState 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 BadgeDrawable.SavedState states}. * @return A {@link SparseArray} that contains a map of int keys (e.g. menuItemId) to {@code * BadgeDrawable BadgeDrawbles}. */ @NonNull public static SparseArray createBadgeDrawablesFromSavedStates( Context context, @NonNull ParcelableSparseArray badgeStates) { SparseArray badgeDrawables = new SparseArray<>(badgeStates.size()); for (int i = 0; i < badgeStates.size(); i++) { int key = badgeStates.keyAt(i); BadgeDrawable.SavedState savedState = (SavedState) badgeStates.valueAt(i); if (savedState == null) { throw new IllegalArgumentException("BadgeDrawable's savedState cannot be null"); } BadgeDrawable badgeDrawable = BadgeDrawable.createFromSavedState(context, savedState); badgeDrawables.put(key, badgeDrawable); } return badgeDrawables; } }