Implementing a Count down timer using Service in the background

Taslim A. Khan picture Taslim A. Khan · May 5, 2013 · Viewed 11.4k times · Source

What I want to do in my app: 1. When the user opens the app, he will be able to see an already running countdown timer. So in that case I want to show the countdown numbers in a textview that will always update each second. 2. The user will be able to stop it. 3. The user can leave the application, but the countdown should still go on and show updated time when he comes back to the application UI.

So based on the above points, I understand I need to implement Service that does work in the background. I have read this link but my problem is with the implementation. I am quite at a lost about how to implement that way. That being said, I also cannot think of a way as to showing the time in the UI once the user comes back to the application. I have also seen this link as well, but I am not sure if only implementing CountDownTimer makes it a running Service. So how should I proceed and implement it? Any links to specific tutorials/codes are appreciated. Thanks.

UPDATES: I have so far been able to do the following: MainActivity.java

public class MainActivity extends Activity {

Button btnStart,btnStop;
Intent serviceIntent;
TextView tv;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    btnStart = (Button) findViewById(R.id.btnStart);
    btnStop = (Button) findViewById(R.id.btnStop);
    tv = (TextView) findViewById(R.id.timeView);

    //final MyCounter timer = new MyCounter(100000,1000);
    //tv.setText("100"); 
    //timer.start();
    serviceIntent = new Intent(MainActivity.this,MyService.class);

    btnStart.setOnClickListener(new View.OnClickListener() {

        @Override
        public void onClick(View v) {
            // TODO Auto-generated method stub
            startService(serviceIntent);
        }
    });

    btnStop.setOnClickListener(new View.OnClickListener() {

        @Override
        public void onClick(View v) {
            // TODO Auto-generated method stub
            stopService(serviceIntent);
        }
    });
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
    // Inflate the menu; this adds items to the action bar if it is present.
    getMenuInflater().inflate(R.menu.main, menu);
    return true;
}

MyService.java:

public class MyService extends Service {

MyCounter timer;
@Override
public void onCreate() {
    // TODO Auto-generated method stub
    timer = new MyCounter(100000,1000);
    super.onCreate();
}

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    // TODO Auto-generated method stub
    timer.start();
    return super.onStartCommand(intent, flags, startId);
}
private class MyCounter extends CountDownTimer{

    public MyCounter(long millisInFuture, long countDownInterval) {
        super(millisInFuture, countDownInterval);
    }

    @Override
    public void onFinish() {
        Toast.makeText(getApplicationContext(), "death", Toast.LENGTH_LONG).show();
        stopSelf();
    }

    @Override
    public void onTick(long millisUntilFinished) {

        Toast.makeText(getApplicationContext(), (millisUntilFinished/1000)+"", Toast.LENGTH_SHORT).show();
    }
}

@Override
public void onDestroy() {
    // TODO Auto-generated method stub
    timer.cancel();
    //stopSelf();
    super.onDestroy();

}

@Override
public IBinder onBind(Intent arg0) {
    // TODO Auto-generated method stub
    return null;
}

    }

The problem is: 1. I don't know how to show the countdown on the MainActivity UI apart from this Toast. 2. It doesn't stop counting when I press Stop Button.

Answer

3h3 picture 3h3 · May 5, 2013

use bindService(Intent service, ServiceConnection conn, int flag) to bind your countdown service, then in your service, return your binder that contain your count, and in your activity, a instantiated ServiceConnection object can handle the binder. AIDL can do this as well, suggest have a look at how to bind service firstly, wish can help you.

demo

public class BindService extends Service{
private int count;
private boolean quit;
private MyBinder binder = new MyBinder();
// My Binder
public class MyBinder extends Binder
{
    public int getCount()
    {
        // get the counting status:count
        return count;
    }
}

@Override
public IBinder onBind(Intent intent)
{
    System.out.println("Service is Binded");
    // return the binder instance
    return binder;
}

@Override
public void onCreate()
{
    super.onCreate();
    System.out.println("Service is Created");
    // counting work
    new Thread()
    {
        @Override
        public void run()
        {
            while (!quit)
            {
                try
                {
                    Thread.sleep(1000);
                }
                catch (InterruptedException e)
                {
                }
                count++;
            }
        }
    }.start();      
}
// invoke when the service unbind
@Override
public boolean onUnbind(Intent intent)
{
    System.out.println("Service is Unbinded");
    return true;
}

@Override
public void onDestroy()
{
    super.onDestroy();
    this.quit = true;
    System.out.println("Service is Destroyed");
}

@Override
public void onRebind(Intent intent) 
{
    super.onRebind(intent);
    this.quit = true;
    System.out.println("Service is ReBinded");
}   

}

and then the activity

public class MainActivity extends Activity{
Button bind , unbind , getServiceStatus;
BindService.MyBinder binder;
// define a ServiceConnection object
private ServiceConnection conn = new ServiceConnection()
{
    // then the Activity connected with the Service, this will be called
    @Override
    public void onServiceConnected(ComponentName name
        , IBinder service)
    {
        System.out.println("--Service Connected--");
        // achieve MyBinder instance
        binder = (BindService.MyBinder) service;
    }
    // then the connection break off
    @Override
    public void onServiceDisconnected(ComponentName name)
    {
        System.out.println("--Service Disconnected--");         
    }
};
@Override
public void onCreate(Bundle savedInstanceState)
{
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    bind = (Button) findViewById(R.id.bind);
    unbind = (Button) findViewById(R.id.unbind);
    getServiceStatus = (Button) findViewById(R.id.getServiceStatus);

    final Intent intent = new Intent();

    intent.setAction("org.crazyit.service.BIND_SERVICE");       
    bind.setOnClickListener(new OnClickListener()
    {
        @Override
        public void onClick(View source)
        {
            //bind Serivce
            bindService(intent , conn , Service.BIND_AUTO_CREATE);  
        }
    });
    unbind.setOnClickListener(new OnClickListener()
    {
        @Override
        public void onClick(View source)
        {
            //unbind Serivce
            unbindService(conn);
        }
    }); 
    getServiceStatus.setOnClickListener(new OnClickListener()
    {
        @Override
        public void onClick(View source)
        {
            // Toast to show the conut value
            Toast.makeText(MainActivity.this
                , "Serivce's count value is:" + binder.getCount()
                , 4000)
                .show();
        }
    });
}}