NekoX/TMessagesProj/src/main/java/org/telegram/ui/Components/ClippingImageView.java

300 lines
9.7 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.Components;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapShader;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.RectF;
import android.graphics.Shader;
import androidx.annotation.Keep;
import android.view.View;
import org.telegram.messenger.AndroidUtilities;
import org.telegram.messenger.FileLog;
import org.telegram.messenger.ImageReceiver;
import java.util.Arrays;
public class ClippingImageView extends View {
private int clipBottom;
private int clipLeft;
private int clipRight;
private int clipTop;
private int orientation;
private int imageY;
private int imageX;
private RectF drawRect;
private Paint paint;
private ImageReceiver.BitmapHolder bmp;
private Matrix matrix;
private boolean needRadius;
private int[] radius = new int[4];
private BitmapShader bitmapShader;
private Paint roundPaint;
private RectF roundRect;
private RectF bitmapRect;
private Matrix shaderMatrix;
private Path roundPath = new Path();
private static float[] radii = new float[8];
private float animationProgress;
private float[][] animationValues;
private float additionalTranslationY;
private float additionalTranslationX;
public ClippingImageView(Context context) {
super(context);
paint = new Paint(Paint.FILTER_BITMAP_FLAG);
paint.setFilterBitmap(true);
matrix = new Matrix();
drawRect = new RectF();
bitmapRect = new RectF();
roundPaint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG);
roundRect = new RectF();
shaderMatrix = new Matrix();
}
public void setAnimationValues(float[][] values) {
animationValues = values;
}
public void setAdditionalTranslationY(float value) {
additionalTranslationY = value;
}
public void setAdditionalTranslationX(float value) {
additionalTranslationX = value;
}
@Override
public void setTranslationY(float translationY) {
super.setTranslationY(translationY + additionalTranslationY);
}
@Override
public float getTranslationY() {
return super.getTranslationY() - additionalTranslationY;
}
@Keep
public float getAnimationProgress() {
return animationProgress;
}
@Keep
public void setAnimationProgress(float progress) {
animationProgress = progress;
setScaleX(animationValues[0][0] + (animationValues[1][0] - animationValues[0][0]) * animationProgress);
setScaleY(animationValues[0][1] + (animationValues[1][1] - animationValues[0][1]) * animationProgress);
setTranslationX(animationValues[0][2] + additionalTranslationX + (animationValues[1][2] + additionalTranslationX - animationValues[0][2] - additionalTranslationX) * animationProgress);
setTranslationY(animationValues[0][3] + (animationValues[1][3] - animationValues[0][3]) * animationProgress);
setClipHorizontal((int) (animationValues[0][4] + (animationValues[1][4] - animationValues[0][4]) * animationProgress));
setClipTop((int) (animationValues[0][5] + (animationValues[1][5] - animationValues[0][5]) * animationProgress));
setClipBottom((int) (animationValues[0][6] + (animationValues[1][6] - animationValues[0][6]) * animationProgress));
for (int a = 0; a < radius.length; a++) {
radius[a] = (int) (animationValues[0][7 + a] + (animationValues[1][7 + a] - animationValues[0][7 + a]) * animationProgress);
setRadius(radius);
}
if (animationValues[0].length > 11) {
setImageY((int) (animationValues[0][11] + (animationValues[1][11] - animationValues[0][11]) * animationProgress));
setImageX((int) (animationValues[0][12] + (animationValues[1][12] - animationValues[0][12]) * animationProgress));
}
invalidate();
}
public void getClippedVisibleRect(RectF rect) {
rect.left = getTranslationX();
rect.top = getTranslationY();
rect.right = rect.left + getMeasuredWidth() * getScaleX();
rect.bottom = rect.top + getMeasuredHeight() * getScaleY();
rect.left += clipLeft;
rect.top += clipTop;
rect.right -= clipRight;
rect.bottom -= clipBottom;
}
public int getClipBottom() {
return clipBottom;
}
public int getClipHorizontal() {
return clipRight;
}
public int getClipLeft() {
return clipLeft;
}
public int getClipRight() {
return clipRight;
}
public int getClipTop() {
return clipTop;
}
public int[] getRadius() {
return radius;
}
public void onDraw(Canvas canvas) {
if (getVisibility() != VISIBLE) {
return;
}
if (bmp != null && !bmp.isRecycled()) {
float scaleY = getScaleY();
canvas.save();
if (needRadius) {
shaderMatrix.reset();
roundRect.set(imageX / scaleY, imageY / scaleY, getWidth() - imageX / scaleY, getHeight() - imageY / scaleY);
bitmapRect.set(0, 0, bmp.getWidth(), bmp.getHeight());
AndroidUtilities.setRectToRect(shaderMatrix, bitmapRect, roundRect, orientation, false);
bitmapShader.setLocalMatrix(shaderMatrix);
canvas.clipRect(clipLeft / scaleY, clipTop / scaleY, getWidth() - clipRight / scaleY, getHeight() - clipBottom / scaleY);
for (int a = 0; a < radius.length; a++) {
radii[a * 2] = radius[a];
radii[a * 2 + 1] = radius[a];
}
roundPath.reset();
roundPath.addRoundRect(roundRect, radii, Path.Direction.CW);
roundPath.close();
canvas.drawPath(roundPath, roundPaint);
} else {
if (orientation == 90 || orientation == 270) {
drawRect.set(-getHeight() / 2, -getWidth() / 2, getHeight() / 2, getWidth() / 2);
matrix.setRectToRect(bitmapRect, drawRect, Matrix.ScaleToFit.FILL);
matrix.postRotate(orientation, 0, 0);
matrix.postTranslate(getWidth() / 2, getHeight() / 2);
} else if (orientation == 180) {
drawRect.set(-getWidth() / 2, -getHeight() / 2, getWidth() / 2, getHeight() / 2);
matrix.setRectToRect(bitmapRect, drawRect, Matrix.ScaleToFit.FILL);
matrix.postRotate(orientation, 0, 0);
matrix.postTranslate(getWidth() / 2, getHeight() / 2);
} else {
drawRect.set(0, 0, getWidth(), getHeight());
matrix.setRectToRect(bitmapRect, drawRect, Matrix.ScaleToFit.FILL);
}
canvas.clipRect(clipLeft / scaleY, clipTop / scaleY, getWidth() - clipRight / scaleY, getHeight() - clipBottom / scaleY);
try {
canvas.drawBitmap(bmp.bitmap, matrix, paint);
} catch (Exception e) {
FileLog.e(e);
}
}
canvas.restore();
}
}
public void setClipBottom(int value) {
clipBottom = value;
invalidate();
}
public void setClipHorizontal(int value) {
clipRight = value;
clipLeft = value;
invalidate();
}
public void setClipLeft(int value) {
clipLeft = value;
invalidate();
}
public void setClipRight(int value) {
clipRight = value;
invalidate();
}
public void setClipTop(int value) {
clipTop = value;
invalidate();
}
public void setClipVertical(int value) {
clipBottom = value;
clipTop = value;
invalidate();
}
public void setImageY(int value) {
imageY = value;
}
public void setImageX(int value) {
imageX = value;
}
public void setOrientation(int angle) {
orientation = angle;
}
public float getCenterX() {
float scaleY = getScaleY();
return getTranslationX() + (clipLeft / scaleY + (getWidth() - clipRight / scaleY)) / 2 * getScaleX();
}
public float getCenterY() {
float scaleY = getScaleY();
return getTranslationY() + (clipTop / scaleY + (getHeight() - clipBottom / scaleY)) / 2 * getScaleY();
}
public void setImageBitmap(ImageReceiver.BitmapHolder bitmap) {
if (bmp != null) {
bmp.release();
bitmapShader = null;
}
bmp = bitmap;
if (bitmap != null && bitmap.bitmap != null) {
bitmapRect.set(0, 0, bitmap.getWidth(), bitmap.getHeight());
bitmapShader = new BitmapShader(bmp.bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);
roundPaint.setShader(bitmapShader);
}
invalidate();
}
public Bitmap getBitmap() {
return bmp != null ? bmp.bitmap : null;
}
public int getOrientation() {
return orientation;
}
public void setRadius(int[] value) {
if (value == null) {
needRadius = false;
Arrays.fill(radius, 0);
return;
}
System.arraycopy(value, 0, radius, 0, value.length);
needRadius = false;
for (int a = 0; a < value.length; a++) {
if (value[a] != 0) {
needRadius = true;
break;
}
}
}
}