自定义文本闪烁效果

效果

在这里插入图片描述
github传送门:https://github.com/drkingwater/SplashTextView

思路

  1. 使用LinearGradient生成文本的高亮部分
  2. LinearGradient.setLocalMatrix(mMatrix)设置Matrix,并不断通过修改Matrix的位置来修改高亮部分的位置

实现

package com.pxq.myapplication.widget;

import android.animation.ValueAnimator;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.LinearGradient;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.Shader;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.util.Log;
import android.util.TypedValue;
import android.view.View;

/**
 * 文本闪烁效果
 */
public class SplashTextView extends View {

    private static final String TAG = "SplashTextView";
    //动画执行时间
    private static final int DURATION = 1500;

    private Paint mPaint;

    //闪烁的宽度
    private int mSplashWidth;

    //渐变色,用来绘制闪烁效果
    private LinearGradient mLinearGradient;
    private Matrix mMatrix;

    //获取文本宽高
    private Rect mTextRect;

    //要画的文本
    private String mText = "";
    private int mTextSize = 20;

    //动画
    private ValueAnimator mValueAnimator;

    public SplashTextView(Context context) {
        super(context);
        init();
    }

    public SplashTextView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        init();
    }
    //初始化画笔等
    private void init() {

        mPaint = new Paint();
        mPaint.setTextSize(TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, mTextSize, getContext().getResources().getDisplayMetrics()));
        mPaint.setAntiAlias(true);
        mTextRect = new Rect();
        mMatrix = new Matrix();

    }

    /**
     * 获取文本宽高
     * @param text
     * @param rect
     */
    private void getTextWidth(String text, Rect rect) {
        mPaint.getTextBounds(text, 0, text.length(), rect);
        rect.width();
        rect.height();
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);


    }

    @Override
    protected void onDraw(Canvas canvas) {

        canvas.drawText(mText, 0, mTextRect.height(), mPaint);

    }

    @Override
    protected void onAttachedToWindow() {
        super.onAttachedToWindow();
//        startAnim();
    }

    @Override
    protected void onDetachedFromWindow() {
        super.onDetachedFromWindow();
        stopAnim();
    }

    public void setText(String text){
        mText = text;
        initGradient();
        startAnim();
    }

    /**
     * 初始化线性渐变
     */
    private void initGradient(){
        getTextWidth(mText, mTextRect);

        mSplashWidth = mTextRect.width() / 3;
        //初始化线性渐变
        mLinearGradient = new LinearGradient(-mSplashWidth, 0, 0, mTextRect.height(),
                //渐变颜色,这里写死颜色值,需要的可暴露接口另行修改
                new int[]{Color.parseColor("#CACACA"), Color.RED, Color.parseColor("#CACACA")},
                new float[]{0.2f, 0.5f, 0.8f},
                Shader.TileMode.CLAMP);

        mLinearGradient.setLocalMatrix(mMatrix);

        mPaint.setShader(mLinearGradient);
    }
    //开始闪烁动画
    private void startAnim() {
        stopAnim();
        mValueAnimator = ValueAnimator.ofInt(0, mSplashWidth + mTextRect.width());
        mValueAnimator.setDuration(DURATION);
        mValueAnimator.setRepeatCount(-1);
        mValueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                int value = (int) animation.getAnimatedValue();
                //改变渐变的位置
                mMatrix.setTranslate(value, 0);
                //注意这里要重新设置Matrix
                mLinearGradient.setLocalMatrix(mMatrix);
//                Log.d(TAG, "onAnimationUpdate: " + value);
                invalidate();
            }
        });
        mValueAnimator.start();
    }

    private void stopAnim() {
        if (mValueAnimator == null){
            return;
        }
        mValueAnimator.cancel();
    }

}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值