/* * Copyright (C) 2017 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 android.support.text.emoji; import static android.support.annotation.RestrictTo.Scope.LIBRARY_GROUP; import android.graphics.Paint; import android.support.annotation.NonNull; import android.support.annotation.RequiresApi; import android.support.annotation.RestrictTo; import android.support.annotation.VisibleForTesting; import android.support.v4.util.Preconditions; import android.text.style.ReplacementSpan; /** * Base span class for the emoji replacement. When an emoji is found and needs to be replaced in a * CharSequence, an instance of this class is added to the CharSequence. */ @RequiresApi(19) public abstract class EmojiSpan extends ReplacementSpan { /** * Temporary object to calculate the size of the span. */ private final Paint.FontMetricsInt mTmpFontMetrics = new Paint.FontMetricsInt(); /** * Information about emoji. This is not parcelled since we do not want multiple objects * representing same emoji to be in memory. When unparcelled, EmojiSpan tries to set it back * using the singleton EmojiCompat instance. */ private final EmojiMetadata mMetadata; /** * Cached width of the span. Width is calculated according to the font metrics. */ private short mWidth = -1; /** * Cached height of the span. Height is calculated according to the font metrics. */ private short mHeight = -1; /** * Cached ratio of current font height to emoji image height. */ private float mRatio = 1.0f; /** * Default constructor. * * @param metadata information about the emoji, cannot be {@code null} * * @hide */ @RestrictTo(LIBRARY_GROUP) EmojiSpan(@NonNull final EmojiMetadata metadata) { Preconditions.checkNotNull(metadata, "metadata cannot be null"); mMetadata = metadata; } @Override public int getSize(@NonNull final Paint paint, final CharSequence text, final int start, final int end, final Paint.FontMetricsInt fm) { paint.getFontMetricsInt(mTmpFontMetrics); final int fontHeight = Math.abs(mTmpFontMetrics.descent - mTmpFontMetrics.ascent); mRatio = fontHeight * 1.0f / mMetadata.getHeight(); mHeight = (short) (mMetadata.getHeight() * mRatio); mWidth = (short) (mMetadata.getWidth() * mRatio); if (fm != null) { fm.ascent = mTmpFontMetrics.ascent; fm.descent = mTmpFontMetrics.descent; fm.top = mTmpFontMetrics.top; fm.bottom = mTmpFontMetrics.bottom; } return mWidth; } /** * @hide */ @RestrictTo(LIBRARY_GROUP) final EmojiMetadata getMetadata() { return mMetadata; } /** * @return width of the span * * @hide */ @RestrictTo(LIBRARY_GROUP) final int getWidth() { return mWidth; } /** * @return height of the span * * @hide */ @RestrictTo(LIBRARY_GROUP) final int getHeight() { return mHeight; } /** * @hide */ @RestrictTo(LIBRARY_GROUP) final float getRatio() { return mRatio; } /** * @return unique id for the emoji that this EmojiSpan is used for * * @hide */ @RestrictTo(LIBRARY_GROUP) @VisibleForTesting public final int getId() { return getMetadata().getId(); } }