Android 自定义view-倒计时控件

在这里插入图片描述

在这里插入图片描述

直接上代码

自定义TextView

文字展示
public class StrokeTextView extends TextView {

    private TextView borderText = null;///用于描边的TextView
    private Context mContext;

    public StrokeTextView(Context context) {
        super(context);
        mContext = context;
        borderText = new TextView(context);
        init();
    }

    public StrokeTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
        mContext = context;
        borderText = new TextView(context, attrs);
        init();
    }

    public StrokeTextView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        mContext = context;
        borderText = new TextView(context, attrs, defStyle);
        init();
    }

    public void init() {
        TextPaint tp1 = borderText.getPaint();
        tp1.setStrokeWidth(12);                                  //设置描边宽度
        tp1.setStyle(Paint.Style.STROKE);                             //对文字只描边
        //设置自定义字体
        Typeface fromAsset = Typeface.createFromAsset(mContext.getAssets(), "fonts/Alibaba-PuHuiTi-Heavy.ttf");
        borderText.setTypeface(fromAsset, Typeface.ITALIC); //自定义字体 ITALIC斜体
        borderText.setTextColor(Color.parseColor("#F46059"));  //设置描边颜色
        borderText.setShadowLayer(3.0F, 2F, 2F, Color.parseColor("#ffd44042"));  //设置阴影效果(投影)
        borderText.setGravity(getGravity());
    }

    @Override
    public void setLayoutParams(ViewGroup.LayoutParams params) {
        super.setLayoutParams(params);
        borderText.setLayoutParams(params);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        CharSequence tt = borderText.getText();

        //两个TextView上的文字必须一致
        if (tt == null || !tt.equals(this.getText())) {
            borderText.setText(getText());
            this.postInvalidate();
        }
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        borderText.measure(widthMeasureSpec, heightMeasureSpec);
    }

    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        super.onLayout(changed, left, top, right, bottom);
        borderText.layout(left, top, right, bottom);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        borderText.draw(canvas);
        super.onDraw(canvas);
    }

}
xml
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:background="#F3B243"
    android:layout_height="match_parent"
    tools:context=".countdown.TestCountActivity">

    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
        <com.xiao.test.countdown.StrokeTextView
            android:layout_marginTop="100dp"
            android:id="@+id/tv_test"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:background="@drawable/play_advertising_timer_bg"
            android:paddingLeft="15dp"
            android:textColor="#FFFFFF"
            android:textSize="33sp"
            android:layout_centerHorizontal="true"
            android:layout_gravity="center"
            android:gravity="center_vertical"
            android:textStyle="italic"
            android:typeface="monospace"
            tools:ignore="RtlSymmetry"
            android:paddingStart="15dp" />

    </RelativeLayout>

</LinearLayout>
倒计时帮助类
public class CountDownHelper {

    private OnCountDownListener onCountDownListener;
    private Disposable disposable;
    private long remainingTime;

    public CountDownHelper(long remainingTime) {
        this.remainingTime = remainingTime;
    }

    /**
     * 回收倒计时
     */
    public void destory() {
        if (disposable != null && !disposable.isDisposed()) {
            disposable.dispose();
        }
    }

    /**
     * 开始倒计时
     */
    public void startCompute() {
        Observable.interval(1, TimeUnit.SECONDS)
                .subscribeOn(Schedulers.newThread())
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(new Observer<Long>() {
                    @Override
                    public void onSubscribe(Disposable d) {
                        disposable = d;
                    }

                    @Override
                    public void onNext(Long aLong) {
                        if (onCountDownListener == null) {
                            return;
                        }
                        remainingTime -= 1000;
                        if (remainingTime > 0) {
                            int day = (int) (remainingTime / (1000 * 60 * 60 * 24));
                            int hour = (int) ((remainingTime % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
                            int minute = (int) ((remainingTime % (1000 * 60 * 60)) / (1000 * 60));
                            int second = (int) ((remainingTime % (1000 * 60)) / 1000);
                            String dayStr = day >= 10 ? String.valueOf(day) : "0" + day;
                            String hourStr = hour >= 10 ? String.valueOf(hour) : "0" + hour;
                            String minuteStr = minute >= 10 ? String.valueOf(minute) : "0" + minute;
                            String secondStr = second >= 10 ? String.valueOf(second) : "0" + second;
                            onCountDownListener.countDown(dayStr, hourStr, minuteStr, secondStr);
                            if (remainingTime <= 0) {
                                onCountDownListener.countDownFinish();
                                if (disposable != null && !disposable.isDisposed()) {
                                    disposable.dispose();
                                }
                            }
                        } else {
                            onCountDownListener.countDownFinish();
                            if (disposable != null && !disposable.isDisposed()) {
                                disposable.dispose();
                            }
                        }
                    }

                    @Override
                    public void onError(Throwable e) {

                    }

                    @Override
                    public void onComplete() {

                    }
                });
    }


    /**
     * 设置倒计时回调监听
     *
     * @param onCountDownListener 倒计时监听
     */
    public void setOnCountDownListener(OnCountDownListener onCountDownListener) {
        this.onCountDownListener = onCountDownListener;
    }

    public interface OnCountDownListener {

        /**
         * 倒计时
         *
         * @param day    天
         * @param hour   小时
         * @param minute 分钟
         * @param second 秒
         */
        void countDown(String day, String hour, String minute, String second);

        /**
         * 倒计时完成
         */
        void countDownFinish();
    }
}
TestCountActivity.java
public class TestCountActivity extends AppCompatActivity {
    private CountDownHelper mCountDownHelper;
    private StrokeTextView mTvTest;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_test_count);

        mTvTest = findViewById(R.id.tv_test);

//设置自定义字体
        Typeface fromAsset = Typeface.createFromAsset(getAssets(), "fonts/Alibaba-PuHuiTi-Heavy.ttf");
        mTvTest.setTypeface(fromAsset, Typeface.ITALIC);  //自定义字体 ITALIC斜体

        long aLong = 1787;
        mCountDownHelper = new CountDownHelper(aLong * 1000);
        mCountDownHelper.startCompute();
        mCountDownHelper.setOnCountDownListener(new CountDownHelper.OnCountDownListener() {
            @SuppressLint("SetTextI18n")
            @Override
            public void countDown(String day, String hour, String minute, String second) {
                mTvTest.setText(hour + ":" + minute + ":" + second);
            }

            @Override
            public void countDownFinish() {
                Log.d("", "结束倒计时");
                mCountDownHelper.destory();
                //延时跳转
                new Handler(new Handler.Callback() {
                    @Override
                    public boolean handleMessage(Message msg) {

                        Toast.makeText(TestCountActivity.this, "时间到了", Toast.LENGTH_SHORT).show();

                        return false;
                    }
                }).sendEmptyMessageDelayed(0, 10000);//表示延迟10秒发送任务

            }
        });


    }
}
引入依赖

implementation ‘io.reactivex.rxjava2:rxjava:2.0.1’
implementation ‘io.reactivex.rxjava2:rxandroid:2.0.1’

欢迎小伙伴们评论
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
要实现计时圆形进度,你可以创建一个自定义的 View,使用 Canvas 和 Paint 来绘制圆形和进度条。以下是一个示例代码: ```java public class CountdownCircleView extends View { private int maxProgress = 100; // 总进度 private int progress = 100; // 当前进度 private int circleColor = Color.GRAY; // 圆形颜色 private int progressColor = Color.BLUE; // 进度条颜色 private Paint circlePaint; private Paint progressPaint; public CountdownCircleView(Context context) { super(context); init(); } public CountdownCircleView(Context context, @Nullable AttributeSet attrs) { super(context, attrs); init(); } public CountdownCircleView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } private void init() { circlePaint = new Paint(); circlePaint.setAntiAlias(true); circlePaint.setColor(circleColor); progressPaint = new Paint(); progressPaint.setAntiAlias(true); progressPaint.setColor(progressColor); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); int viewWidth = getWidth(); int viewHeight = getHeight(); // 绘制圆形 int diameter = Math.min(viewWidth, viewHeight); int radius = diameter / 2; int centerX = viewWidth / 2; int centerY = viewHeight / 2; canvas.drawCircle(centerX, centerY, radius, circlePaint); // 绘制进度条 RectF rectF = new RectF(centerX - radius, centerY - radius, centerX + radius, centerY + radius); float sweepAngle = 360f * progress / maxProgress; canvas.drawArc(rectF, -90, sweepAngle, true, progressPaint); } public void setMaxProgress(int maxProgress) { this.maxProgress = maxProgress; } public void setProgress(int progress) { this.progress = progress; invalidate(); // 重新绘制 } } ``` 你可以将上述代码放入你的 Android 项目中,并在布局文件中使用 `CountdownCircleView`。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值