Android定时器Chronometer切到后台,无法监听回调onChronometerTick解决方案

关于定时器的使用,我就不过多介绍了。今天就讲一下关于定时器Chronometer的回调onChronometer方法的使用。

在一些特殊场景中,我们使用定时器会有一定的限制;比如:

倒计时

Chronometer只能进行正计时,无法进行倒计时。但是想实现倒计时的需求该怎么做呢?
这时候就用到了onChronometer回调方法
这个回调方法是在定时器在每次计时的时候都会回调的一个方法,具体用法如下:

 chronometer.setOnChronometerTickListener(new Chronometer.OnChronometerTickListener() {
            @Override
            public void onChronometerTick(Chronometer ch) {
            	 if(countDownNum>0){
                        countDownNum--;
                 }
                 chronometer.setText("00:00:"+countDownNum);
            }
        });

我们只需要通过Chronometer的 setText 方法,再去定义一下我们需要倒计时的时间,具体实现逻辑自己随心,然后去设置我们需要显示的倒计时就可以了。

问题

这时候问题就来了。
当APP切到后台时候,再切回来时候,发现时间还是切出去时候的时间,这是为什么呢?
原来:
当APP进入后台时,会调用onWindowVisibilityChanged()方法,Chronometer回调的onChronometer方法是通过dispatchChronometerTick()方法进行回调的,但是在updateRunning()方法中有一个running变量,这个变量进行了一个是否在显示在前台判断,所以在后台时并不会调用dispatchChronometerTick()方法
在这里插入图片描述
在这里插入图片描述

解决方案:
针对于需要在后台用到回调的场景,我的解决方案是这样的:
重写一个自定义View,将Chronometer中的几个必要方法复制过来,在将是否前台判断去掉即可;下面是我的代码:


public class MyChronometer extends android.support.v7.widget.AppCompatTextView {
    private static final String TAG = "Chronometer";

    public interface OnChronometerTickListener {
        void onChronometerTick(MyChronometer myChronometer);
    }

    private long mBase;
    private boolean mVisible;
    private boolean mStarted;
    private boolean mRunning;
    private boolean mLogged;
    private String mFormat;
    private Formatter mFormatter;
    private Locale mFormatterLocale;
    private Object[] mFormatterArgs = new Object[1];
    private StringBuilder mFormatBuilder;
    private OnChronometerTickListener mOnChronometerTickListener;
    private StringBuilder mRecycle = new StringBuilder(8);

    private static final int TICK_WHAT = 2;

    public MyChronometer(Context context) {
        super(context);
    }

    public MyChronometer(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public MyChronometer(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    private void init() {
        mBase = SystemClock.elapsedRealtime();
        updateText(mBase);
    }

    public void setBase(long base) {
        mBase = base;
        dispatchChronometerTick();
        updateText(SystemClock.elapsedRealtime());
    }

    public long getBase() {
        return mBase;
    }

    public void setFormat(String format) {
        mFormat = format;
        if (format != null && mFormatBuilder == null) {
            mFormatBuilder = new StringBuilder(format.length() * 2);
        }
    }

    public String getFormat() {
        return mFormat;
    }

    public void setOnChronometerTickListener(OnChronometerTickListener listener) {
        mOnChronometerTickListener = listener;
    }

    public OnChronometerTickListener getOnChronometerTickListener() {
        return mOnChronometerTickListener;
    }

    public void start() {
        mStarted = true;
        updateRunning();
    }

    public void stop() {
        mStarted = false;
        updateRunning();
    }

    public void setStarted(boolean started) {
        mStarted = started;
        updateRunning();
    }

    @Override
    protected void onDetachedFromWindow() {
        super.onDetachedFromWindow();
        mVisible = false;
        updateRunning();
    }

    @Override
    protected void onWindowVisibilityChanged(int visibility) {
        super.onWindowVisibilityChanged(visibility);
        updateRunning();
    }

    private synchronized void updateText(long now) {
        long seconds = now - mBase;
        seconds /= 1000;
        String text = DateUtils.formatElapsedTime(mRecycle, seconds);
        if (mFormat != null) {
            Locale loc = Locale.getDefault();
            if (mFormatter == null || !loc.equals(mFormatterLocale)) {
                mFormatterLocale = loc;
                mFormatter = new Formatter(mFormatBuilder, loc);
            }
            mFormatBuilder.setLength(0);
            mFormatterArgs[0] = text;
            try {
                mFormatter.format(mFormat, mFormatterArgs);
                text = mFormatBuilder.toString();
            } catch (IllegalFormatException ex) {
                if (!mLogged) {
                    Log.w(TAG, "Illegal format string: " + mFormat);
                    mLogged = true;
                }
            }
        }
        setText(text);
    }
    
    //重点在这里,将running变量赋值去掉mVisible和isShown()
    private void updateRunning() {
        //boolean running =mVisible && mStarted && isShown(); 源代码是这样的
        boolean running = mStarted;//改成这样
        if (running != mRunning) {
            if (running) {
                updateText(SystemClock.elapsedRealtime());
                dispatchChronometerTick();
                mHandler.sendMessageDelayed(Message.obtain(mHandler, TICK_WHAT), 1000);
            } else {
                mHandler.removeMessages(TICK_WHAT);
            }
            mRunning = running;
        }
    }

    private Handler mHandler = new Handler() {
        public void handleMessage(Message m) {
            if (mRunning) {
                updateText(SystemClock.elapsedRealtime());
                dispatchChronometerTick();
                sendMessageDelayed(Message.obtain(this, TICK_WHAT), 1000);
            }
        }
    };

    void dispatchChronometerTick() {
        if (mOnChronometerTickListener != null) {
            mOnChronometerTickListener.onChronometerTick(this);
        }
    }

    @SuppressLint("NewApi")
    @Override
    public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
        super.onInitializeAccessibilityEvent(event);
        event.setClassName(MyChronometer.class.getName());
    }

    @SuppressLint("NewApi")
    @Override
    public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) {
        super.onInitializeAccessibilityNodeInfo(info);
        info.setClassName(MyChronometer.class.getName());
    }

}

这样APP在后台的时候也可以收到onChronometerTick的回调了。

  • 5
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 5
    评论
AndroidChronometer控件可以用于实现计时器功能,包括正计时和倒计时。如果想要实现倒计时,需要在代码中设置Chronometer控件的计时时间,并在倒计时结束时触发相关操作。 具体实现方式如下: 1. 在布局文件中添加Chronometer控件: ``` <Chronometer android:id="@+id/chronometer" android:layout_width="wrap_content" android:layout_height="wrap_content" android:format="倒计时:%s" android:layout_centerInParent="true"/> ``` 其中,format属性用于设置Chronometer控件显示的文本格式,%s表示计时器的时间将会被替换为具体的倒计时时间。 2. 在代码中设置倒计时时间: ``` Chronometer chronometer = findViewById(R.id.chronometer); chronometer.setBase(SystemClock.elapsedRealtime() + 10 * 1000); //设置倒计时时间为10秒 chronometer.start(); //开始倒计时 ``` 其中,setBase方法用于设置Chronometer控件的起始时间,这里使用SystemClock.elapsedRealtime()获取当前时间,再加上10秒的时间作为起始时间,即表示倒计时从当前时间开始,倒计时时间为10秒。start方法用于开始倒计时。 3. 在倒计时结束时触发相关操作: ``` chronometer.setOnChronometerTickListener(new Chronometer.OnChronometerTickListener() { @Override public void onChronometerTick(Chronometer chronometer) { if (chronometer.getText().toString().equals("00:00")) { //倒计时结束,执行相关操作 chronometer.stop(); //停止倒计时 } } }); ``` 在倒计时过程中,可以通过设置setOnChronometerTickListener监听器,在每秒钟的时钟周期中判断Chronometer控件显示的时间是否为00:00,如果是则表示倒计时结束,可以在这里触发相关操作。同时,需要在倒计时结束时停止Chronometer的计时器,避免继续计时。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值