NekoX/TMessagesProj/src/main/java/org/telegram/ui/ActionBar/SimpleTextView.java

467 lines
16 KiB
Java

/*
* This is the source code of Telegram for Android v. 5.x.x.
* It is licensed under GNU GPL v. 2 or later.
* You should have received a copy of the license in this archive (see LICENSE).
*
* Copyright Nikolai Kudashov, 2013-2018.
*/
package org.telegram.ui.ActionBar;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.PorterDuff;
import android.graphics.PorterDuffColorFilter;
import android.graphics.Typeface;
import android.graphics.drawable.Drawable;
import android.graphics.drawable.GradientDrawable;
import android.os.SystemClock;
import android.text.Layout;
import android.text.SpannableStringBuilder;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.text.TextUtils;
import android.view.Gravity;
import android.view.View;
import org.telegram.messenger.AndroidUtilities;
public class SimpleTextView extends View implements Drawable.Callback {
private Layout layout;
private TextPaint textPaint;
private int gravity = Gravity.LEFT | Gravity.TOP;
private CharSequence text;
private SpannableStringBuilder spannableStringBuilder;
private Drawable leftDrawable;
private Drawable rightDrawable;
private int drawablePadding = AndroidUtilities.dp(4);
private int leftDrawableTopPadding;
private int rightDrawableTopPadding;
private boolean scrollNonFitText;
private boolean textDoesNotFit;
private float scrollingOffset;
private long lastUpdateTime;
private int currentScrollDelay;
private GradientDrawable fadeDrawable;
private GradientDrawable fadeDrawableBack;
private int lastWidth;
private int offsetX;
private int offsetY;
private int textWidth;
private int totalWidth;
private int textHeight;
private boolean wasLayout;
private static final int PIXELS_PER_SECOND = 50;
private static final int PIXELS_PER_SECOND_SLOW = 30;
private static final int DIST_BETWEEN_SCROLLING_TEXT = 16;
private static final int SCROLL_DELAY_MS = 500;
private static final int SCROLL_SLOWDOWN_PX = 100;
public SimpleTextView(Context context) {
super(context);
textPaint = new TextPaint(Paint.ANTI_ALIAS_FLAG);
}
public void setTextColor(int color) {
textPaint.setColor(color);
invalidate();
}
public void setLinkTextColor(int color) {
textPaint.linkColor = color;
invalidate();
}
@Override
protected void onDetachedFromWindow() {
super.onDetachedFromWindow();
wasLayout = false;
}
public void setTextSize(int size) {
int newSize = AndroidUtilities.dp(size);
if (newSize == textPaint.getTextSize()) {
return;
}
textPaint.setTextSize(newSize);
if (!recreateLayoutMaybe()) {
invalidate();
}
}
public void setScrollNonFitText(boolean value) {
if (scrollNonFitText == value) {
return;
}
scrollNonFitText = value;
if (scrollNonFitText) {
fadeDrawable = new GradientDrawable(GradientDrawable.Orientation.LEFT_RIGHT, new int[]{0xffffffff, 0});
fadeDrawableBack = new GradientDrawable(GradientDrawable.Orientation.LEFT_RIGHT, new int[]{0, 0xffffffff});
}
requestLayout();
}
public void setGravity(int value) {
gravity = value;
}
public void setTypeface(Typeface typeface) {
textPaint.setTypeface(typeface);
}
public int getSideDrawablesSize() {
int size = 0;
if (leftDrawable != null) {
size += leftDrawable.getIntrinsicWidth() + drawablePadding;
}
if (rightDrawable != null) {
size += rightDrawable.getIntrinsicWidth() + drawablePadding;
}
return size;
}
public Paint getPaint() {
return textPaint;
}
private void calcOffset(int width) {
if (layout.getLineCount() > 0) {
textWidth = (int) Math.ceil(layout.getLineWidth(0));
textHeight = layout.getLineBottom(0);
if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == Gravity.CENTER_VERTICAL) {
offsetY = (getMeasuredHeight() - textHeight) / 2;
} else {
offsetY = 0;
}
if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.LEFT) {
offsetX = -(int) layout.getLineLeft(0);
} else if (layout.getLineLeft(0) == 0) {
offsetX = width - textWidth;
} else {
offsetX = -AndroidUtilities.dp(8);
}
offsetX += getPaddingLeft();
textDoesNotFit = textWidth > width;
}
}
private boolean createLayout(int width) {
if (text != null) {
try {
if (leftDrawable != null) {
width -= leftDrawable.getIntrinsicWidth();
width -= drawablePadding;
}
if (rightDrawable != null) {
width -= rightDrawable.getIntrinsicWidth();
width -= drawablePadding;
}
CharSequence string;
if (scrollNonFitText) {
string = text;
} else {
string = TextUtils.ellipsize(text, textPaint, width, TextUtils.TruncateAt.END);
}
/*if (layout != null && TextUtils.equals(layout.getText(), string)) {
calcOffset(width);
return false;
}*/
layout = new StaticLayout(string, 0, string.length(), textPaint, scrollNonFitText ? AndroidUtilities.dp(2000) : width + AndroidUtilities.dp(8), Layout.Alignment.ALIGN_NORMAL, 1.0f, 0.0f, false);
calcOffset(width);
} catch (Exception ignore) {
}
} else {
layout = null;
textWidth = 0;
textHeight = 0;
}
invalidate();
return true;
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int width = MeasureSpec.getSize(widthMeasureSpec);
int height = MeasureSpec.getSize(heightMeasureSpec);
if (lastWidth != AndroidUtilities.displaySize.x) {
lastWidth = AndroidUtilities.displaySize.x;
scrollingOffset = 0;
currentScrollDelay = SCROLL_DELAY_MS;
}
createLayout(width - getPaddingLeft() - getPaddingRight());
int finalHeight;
if (MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.EXACTLY) {
finalHeight = height;
} else {
finalHeight = textHeight;
}
setMeasuredDimension(width, finalHeight);
}
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
wasLayout = true;
}
public int getTextWidth() {
return textWidth;
}
public int getTextHeight() {
return textHeight;
}
public void setLeftDrawableTopPadding(int value) {
leftDrawableTopPadding = value;
}
public void setRightDrawableTopPadding(int value) {
rightDrawableTopPadding = value;
}
public void setLeftDrawable(int resId) {
setLeftDrawable(resId == 0 ? null : getContext().getResources().getDrawable(resId));
}
public void setRightDrawable(int resId) {
setRightDrawable(resId == 0 ? null : getContext().getResources().getDrawable(resId));
}
public void setLeftDrawable(Drawable drawable) {
if (leftDrawable == drawable) {
return;
}
if (leftDrawable != null) {
leftDrawable.setCallback(null);
}
leftDrawable = drawable;
if (drawable != null) {
drawable.setCallback(this);
}
if (!recreateLayoutMaybe()) {
invalidate();
}
}
public void setRightDrawable(Drawable drawable) {
if (rightDrawable == drawable) {
return;
}
if (rightDrawable != null) {
rightDrawable.setCallback(null);
}
rightDrawable = drawable;
if (drawable != null) {
drawable.setCallback(this);
}
if (!recreateLayoutMaybe()) {
invalidate();
}
}
public boolean setText(CharSequence value) {
return setText(value, false);
}
public boolean setText(CharSequence value, boolean force) {
if (text == null && value == null || !force && text != null && value != null && text.equals(value)) {
return false;
}
text = value;
scrollingOffset = 0;
currentScrollDelay = SCROLL_DELAY_MS;
recreateLayoutMaybe();
return true;
}
public void setDrawablePadding(int value) {
if (drawablePadding == value) {
return;
}
drawablePadding = value;
if (!recreateLayoutMaybe()) {
invalidate();
}
}
private boolean recreateLayoutMaybe() {
if (wasLayout && getMeasuredHeight() != 0) {
return createLayout(getMeasuredWidth());
} else {
requestLayout();
}
return true;
}
public CharSequence getText() {
if (text == null) {
return "";
}
return text;
}
public int getTextStartX() {
if (layout == null) {
return 0;
}
int textOffsetX = 0;
if (leftDrawable != null) {
if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.LEFT) {
textOffsetX += drawablePadding + leftDrawable.getIntrinsicWidth();
}
}
return (int) getX() + offsetX + textOffsetX;
}
public TextPaint getTextPaint() {
return textPaint;
}
public int getTextStartY() {
if (layout == null) {
return 0;
}
return (int) getY();
}
@Override
protected void onDraw(Canvas canvas) {
int textOffsetX = 0;
totalWidth = textWidth;
if (leftDrawable != null) {
int x = (int) -scrollingOffset;
int y = (textHeight - leftDrawable.getIntrinsicHeight()) / 2 + leftDrawableTopPadding;
leftDrawable.setBounds(x, y, x + leftDrawable.getIntrinsicWidth(), y + leftDrawable.getIntrinsicHeight());
leftDrawable.draw(canvas);
if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == Gravity.LEFT) {
textOffsetX += drawablePadding + leftDrawable.getIntrinsicWidth();
}
totalWidth += drawablePadding + leftDrawable.getIntrinsicWidth();
}
if (rightDrawable != null) {
int x = textOffsetX + textWidth + drawablePadding + (int) -scrollingOffset;
int y = (textHeight - rightDrawable.getIntrinsicHeight()) / 2 + rightDrawableTopPadding;
rightDrawable.setBounds(x, y, x + rightDrawable.getIntrinsicWidth(), y + rightDrawable.getIntrinsicHeight());
rightDrawable.draw(canvas);
totalWidth += drawablePadding + rightDrawable.getIntrinsicWidth();
}
int nextScrollX = totalWidth + AndroidUtilities.dp(DIST_BETWEEN_SCROLLING_TEXT);
if (scrollingOffset != 0) {
if (leftDrawable != null) {
int x = (int) -scrollingOffset + nextScrollX;
int y = (textHeight - leftDrawable.getIntrinsicHeight()) / 2 + leftDrawableTopPadding;
leftDrawable.setBounds(x, y, x + leftDrawable.getIntrinsicWidth(), y + leftDrawable.getIntrinsicHeight());
leftDrawable.draw(canvas);
}
if (rightDrawable != null) {
int x = textOffsetX + textWidth + drawablePadding + (int) -scrollingOffset + nextScrollX;
int y = (textHeight - rightDrawable.getIntrinsicHeight()) / 2 + rightDrawableTopPadding;
rightDrawable.setBounds(x, y, x + rightDrawable.getIntrinsicWidth(), y + rightDrawable.getIntrinsicHeight());
rightDrawable.draw(canvas);
}
}
if (layout != null) {
if (offsetX + textOffsetX != 0 || offsetY != 0 || scrollingOffset != 0) {
canvas.save();
canvas.translate(offsetX + textOffsetX - scrollingOffset, offsetY);
if (scrollingOffset != 0) {
//canvas.clipRect(0, 0, getMeasuredWidth(), getMeasuredHeight());
}
}
layout.draw(canvas);
if (scrollingOffset != 0) {
canvas.translate(nextScrollX, 0);
layout.draw(canvas);
}
if (offsetX + textOffsetX != 0 || offsetY != 0 || scrollingOffset != 0) {
canvas.restore();
}
if (scrollNonFitText && textDoesNotFit) {
if (scrollingOffset < AndroidUtilities.dp(10)) {
fadeDrawable.setAlpha((int) (255 * (scrollingOffset / AndroidUtilities.dp(10))));
} else if (scrollingOffset > totalWidth + AndroidUtilities.dp(DIST_BETWEEN_SCROLLING_TEXT) - AndroidUtilities.dp(10)) {
float dist = scrollingOffset - (totalWidth + AndroidUtilities.dp(DIST_BETWEEN_SCROLLING_TEXT) - AndroidUtilities.dp(10));
fadeDrawable.setAlpha((int) (255 * (1.0f - dist / AndroidUtilities.dp(10))));
} else {
fadeDrawable.setAlpha(255);
}
fadeDrawable.setBounds(0, 0, AndroidUtilities.dp(6), getMeasuredHeight());
fadeDrawable.draw(canvas);
fadeDrawableBack.setBounds(getMeasuredWidth() - AndroidUtilities.dp(6), 0, getMeasuredWidth(), getMeasuredHeight());
fadeDrawableBack.draw(canvas);
}
updateScrollAnimation();
}
}
@Override
public void setBackgroundColor(int color) {
if (scrollNonFitText) {
if (fadeDrawable != null) {
fadeDrawable.setColorFilter(new PorterDuffColorFilter(color, PorterDuff.Mode.MULTIPLY));
fadeDrawableBack.setColorFilter(new PorterDuffColorFilter(color, PorterDuff.Mode.MULTIPLY));
}
} else {
super.setBackgroundColor(color);
}
}
private void updateScrollAnimation() {
if (!scrollNonFitText || !textDoesNotFit) {
return;
}
long newUpdateTime = SystemClock.uptimeMillis();
long dt = newUpdateTime - lastUpdateTime;
if (dt > 17) {
dt = 17;
}
if (currentScrollDelay > 0) {
currentScrollDelay -= dt;
} else {
int totalDistance = totalWidth + AndroidUtilities.dp(DIST_BETWEEN_SCROLLING_TEXT);
float pixelsPerSecond;
if (scrollingOffset < AndroidUtilities.dp(SCROLL_SLOWDOWN_PX)) {
pixelsPerSecond = PIXELS_PER_SECOND_SLOW + (PIXELS_PER_SECOND - PIXELS_PER_SECOND_SLOW) * (scrollingOffset / AndroidUtilities.dp(SCROLL_SLOWDOWN_PX));
} else if (scrollingOffset >= totalDistance - AndroidUtilities.dp(SCROLL_SLOWDOWN_PX)) {
float dist = scrollingOffset - (totalDistance - AndroidUtilities.dp(SCROLL_SLOWDOWN_PX));
pixelsPerSecond = PIXELS_PER_SECOND - (PIXELS_PER_SECOND - PIXELS_PER_SECOND_SLOW) * (dist / AndroidUtilities.dp(SCROLL_SLOWDOWN_PX));
} else {
pixelsPerSecond = PIXELS_PER_SECOND;
}
scrollingOffset += dt / 1000.0f * AndroidUtilities.dp(pixelsPerSecond);
lastUpdateTime = newUpdateTime;
if (scrollingOffset > totalDistance) {
scrollingOffset = 0;
currentScrollDelay = SCROLL_DELAY_MS;
}
}
invalidate();
}
@Override
public void invalidateDrawable(Drawable who) {
if (who == leftDrawable) {
invalidate(leftDrawable.getBounds());
} else if (who == rightDrawable) {
invalidate(rightDrawable.getBounds());
}
}
@Override
public boolean hasOverlappingRendering() {
return false;
}
}