mirror of
https://github.com/material-components/material-components-android.git
synced 2026-01-19 19:41:35 +08:00
Resolves https://github.com/material-components/material-components-android/issues/2011 PiperOrigin-RevId: 359360955
288 lines
10 KiB
Java
288 lines
10 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.appbar;
|
|
|
|
import com.google.android.material.R;
|
|
|
|
import static com.google.android.material.theme.overlay.MaterialThemeOverlay.wrap;
|
|
import static java.lang.Math.max;
|
|
|
|
import android.content.Context;
|
|
import android.content.res.ColorStateList;
|
|
import android.content.res.TypedArray;
|
|
import android.graphics.Color;
|
|
import android.graphics.drawable.ColorDrawable;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.os.Build.VERSION_CODES;
|
|
import androidx.core.graphics.drawable.DrawableCompat;
|
|
import androidx.core.view.ViewCompat;
|
|
import androidx.appcompat.widget.Toolbar;
|
|
import android.util.AttributeSet;
|
|
import android.util.Pair;
|
|
import android.view.View;
|
|
import android.widget.TextView;
|
|
import androidx.annotation.ColorInt;
|
|
import androidx.annotation.NonNull;
|
|
import androidx.annotation.Nullable;
|
|
import androidx.annotation.RequiresApi;
|
|
import com.google.android.material.internal.ThemeEnforcement;
|
|
import com.google.android.material.internal.ToolbarUtils;
|
|
import com.google.android.material.shape.MaterialShapeDrawable;
|
|
import com.google.android.material.shape.MaterialShapeUtils;
|
|
|
|
/**
|
|
* {@code MaterialToolbar} is a {@link Toolbar} that implements certain Material features, such as
|
|
* elevation overlays for Dark Themes and centered titles.
|
|
*
|
|
* <p>Regarding the Dark Theme elevation overlays, it's important to note that the Material {@link
|
|
* AppBarLayout} component also provides elevation overlay support, and operates under the
|
|
* assumption that the child {@code Toolbar} does not have a background. While a {@code
|
|
* MaterialToolbar} with a transparent background can be used within an {@link AppBarLayout}, in
|
|
* terms of elevation overlays its main value comes into play with the standalone {@code Toolbar}
|
|
* case, when using the {@code Widget.MaterialComponents.Toolbar.Surface} style with elevation.
|
|
*
|
|
* <p>To get started with the {@code MaterialToolbar} component, use {@code
|
|
* com.google.android.material.appbar.MaterialToolbar} in your layout XML instead of {@code
|
|
* androidx.appcompat.widget.Toolbar} or {@code Toolbar}. E.g.,:
|
|
*
|
|
* <pre>
|
|
* <com.google.android.material.appbar.MaterialToolbar
|
|
* android:layout_width="match_parent"
|
|
* android:layout_height="wrap_content"/>
|
|
* </pre>
|
|
*/
|
|
public class MaterialToolbar extends Toolbar {
|
|
|
|
private static final int DEF_STYLE_RES = R.style.Widget_MaterialComponents_Toolbar;
|
|
|
|
@Nullable private Integer navigationIconTint;
|
|
private boolean titleCentered;
|
|
private boolean subtitleCentered;
|
|
|
|
public MaterialToolbar(@NonNull Context context) {
|
|
this(context, null);
|
|
}
|
|
|
|
public MaterialToolbar(@NonNull Context context, @Nullable AttributeSet attrs) {
|
|
this(context, attrs, R.attr.toolbarStyle);
|
|
}
|
|
|
|
public MaterialToolbar(@NonNull Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
|
|
super(wrap(context, attrs, defStyleAttr, DEF_STYLE_RES), attrs, defStyleAttr);
|
|
// Ensure we are using the correctly themed context rather than the context that was passed in.
|
|
context = getContext();
|
|
|
|
final TypedArray a =
|
|
ThemeEnforcement.obtainStyledAttributes(
|
|
context, attrs, R.styleable.MaterialToolbar, defStyleAttr, DEF_STYLE_RES);
|
|
|
|
if (a.hasValue(R.styleable.MaterialToolbar_navigationIconTint)) {
|
|
setNavigationIconTint(a.getColor(R.styleable.MaterialToolbar_navigationIconTint, -1));
|
|
}
|
|
|
|
titleCentered = a.getBoolean(R.styleable.MaterialToolbar_titleCentered, false);
|
|
subtitleCentered = a.getBoolean(R.styleable.MaterialToolbar_subtitleCentered, false);
|
|
|
|
a.recycle();
|
|
|
|
initBackground(context);
|
|
}
|
|
|
|
@Override
|
|
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
|
|
super.onLayout(changed, left, top, right, bottom);
|
|
|
|
maybeCenterTitleViews();
|
|
}
|
|
|
|
private void maybeCenterTitleViews() {
|
|
if (!titleCentered && !subtitleCentered) {
|
|
return;
|
|
}
|
|
|
|
TextView titleTextView = ToolbarUtils.getTitleTextView(this);
|
|
TextView subtitleTextView = ToolbarUtils.getSubtitleTextView(this);
|
|
if (titleTextView == null && subtitleTextView == null) {
|
|
return;
|
|
}
|
|
|
|
Pair<Integer, Integer> titleBoundLimits =
|
|
calculateTitleBoundLimits(titleTextView, subtitleTextView);
|
|
|
|
if (titleCentered && titleTextView != null) {
|
|
layoutTitleCenteredHorizontally(titleTextView, titleBoundLimits);
|
|
}
|
|
|
|
if (subtitleCentered && subtitleTextView != null) {
|
|
layoutTitleCenteredHorizontally(subtitleTextView, titleBoundLimits);
|
|
}
|
|
}
|
|
|
|
private Pair<Integer, Integer> calculateTitleBoundLimits(
|
|
@Nullable TextView titleTextView, @Nullable TextView subtitleTextView) {
|
|
int width = getMeasuredWidth();
|
|
int midpoint = width / 2;
|
|
int leftLimit = getPaddingLeft();
|
|
int rightLimit = width - getPaddingRight();
|
|
|
|
for (int i = 0; i < getChildCount(); i++) {
|
|
View child = getChildAt(i);
|
|
if (child.getVisibility() != GONE && child != titleTextView && child != subtitleTextView) {
|
|
if (child.getRight() < midpoint && child.getRight() > leftLimit) {
|
|
leftLimit = child.getRight();
|
|
}
|
|
if (child.getLeft() > midpoint && child.getLeft() < rightLimit) {
|
|
rightLimit = child.getLeft();
|
|
}
|
|
}
|
|
}
|
|
|
|
return new Pair<>(leftLimit, rightLimit);
|
|
}
|
|
|
|
private void layoutTitleCenteredHorizontally(
|
|
View titleView, Pair<Integer, Integer> titleBoundLimits) {
|
|
int width = getMeasuredWidth();
|
|
int titleWidth = titleView.getMeasuredWidth();
|
|
|
|
int titleLeft = width / 2 - titleWidth / 2;
|
|
int titleRight = titleLeft + titleWidth;
|
|
|
|
int leftOverlap = max(titleBoundLimits.first - titleLeft, 0);
|
|
int rightOverlap = max(titleRight - titleBoundLimits.second, 0);
|
|
int overlap = max(leftOverlap, rightOverlap);
|
|
|
|
if (overlap > 0) {
|
|
titleLeft += overlap;
|
|
titleRight -= overlap;
|
|
titleWidth = titleRight - titleLeft;
|
|
titleView.measure(
|
|
MeasureSpec.makeMeasureSpec(titleWidth, MeasureSpec.EXACTLY),
|
|
titleView.getMeasuredHeightAndState());
|
|
}
|
|
|
|
titleView.layout(titleLeft, titleView.getTop(), titleRight, titleView.getBottom());
|
|
}
|
|
|
|
@Override
|
|
protected void onAttachedToWindow() {
|
|
super.onAttachedToWindow();
|
|
|
|
MaterialShapeUtils.setParentAbsoluteElevation(this);
|
|
}
|
|
|
|
@RequiresApi(VERSION_CODES.LOLLIPOP)
|
|
@Override
|
|
public void setElevation(float elevation) {
|
|
super.setElevation(elevation);
|
|
|
|
MaterialShapeUtils.setElevation(this, elevation);
|
|
}
|
|
|
|
@Override
|
|
public void setNavigationIcon(@Nullable Drawable drawable) {
|
|
super.setNavigationIcon(maybeTintNavigationIcon(drawable));
|
|
}
|
|
|
|
/**
|
|
* Sets the color of the toolbar's navigation icon.
|
|
*
|
|
* @see #setNavigationIcon
|
|
*/
|
|
public void setNavigationIconTint(@ColorInt int navigationIconTint) {
|
|
this.navigationIconTint = navigationIconTint;
|
|
Drawable navigationIcon = getNavigationIcon();
|
|
if (navigationIcon != null) {
|
|
// Causes navigation icon to be tinted if needed.
|
|
setNavigationIcon(navigationIcon);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Sets whether the title text corresponding to the {@link #setTitle(int)} method should be
|
|
* centered horizontally within the toolbar.
|
|
*
|
|
* <p>Note: it is not recommended to use centered titles in conjunction with a nested custom view,
|
|
* as there may be positioning and overlap issues.
|
|
*/
|
|
public void setTitleCentered(boolean titleCentered) {
|
|
if (this.titleCentered != titleCentered) {
|
|
this.titleCentered = titleCentered;
|
|
requestLayout();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns whether the title text corresponding to the {@link #setTitle(int)} method should be
|
|
* centered horizontally within the toolbar.
|
|
*
|
|
* @see #setTitleCentered(boolean)
|
|
*/
|
|
public boolean isTitleCentered() {
|
|
return titleCentered;
|
|
}
|
|
|
|
/**
|
|
* Sets whether the subtitle text corresponding to the {@link #setSubtitle(int)} method should be
|
|
* centered horizontally within the toolbar.
|
|
*
|
|
* <p>Note: it is not recommended to use centered titles in conjunction with a nested custom view,
|
|
* as there may be positioning and overlap issues.
|
|
*/
|
|
public void setSubtitleCentered(boolean subtitleCentered) {
|
|
if (this.subtitleCentered != subtitleCentered) {
|
|
this.subtitleCentered = subtitleCentered;
|
|
requestLayout();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns whether the subtitle text corresponding to the {@link #setSubtitle(int)} method should
|
|
* be centered horizontally within the toolbar.
|
|
*
|
|
* @see #setSubtitleCentered(boolean)
|
|
*/
|
|
public boolean isSubtitleCentered() {
|
|
return subtitleCentered;
|
|
}
|
|
|
|
private void initBackground(Context context) {
|
|
Drawable background = getBackground();
|
|
if (background != null && !(background instanceof ColorDrawable)) {
|
|
return;
|
|
}
|
|
MaterialShapeDrawable materialShapeDrawable = new MaterialShapeDrawable();
|
|
int backgroundColor =
|
|
background != null ? ((ColorDrawable) background).getColor() : Color.TRANSPARENT;
|
|
materialShapeDrawable.setFillColor(ColorStateList.valueOf(backgroundColor));
|
|
materialShapeDrawable.initializeElevationOverlay(context);
|
|
materialShapeDrawable.setElevation(ViewCompat.getElevation(this));
|
|
ViewCompat.setBackground(this, materialShapeDrawable);
|
|
}
|
|
|
|
@Nullable
|
|
private Drawable maybeTintNavigationIcon(@Nullable Drawable navigationIcon) {
|
|
if (navigationIcon != null && navigationIconTint != null) {
|
|
Drawable wrappedNavigationIcon = DrawableCompat.wrap(navigationIcon);
|
|
DrawableCompat.setTint(wrappedNavigationIcon, navigationIconTint);
|
|
return wrappedNavigationIcon;
|
|
} else {
|
|
return navigationIcon;
|
|
}
|
|
}
|
|
}
|