In this article, an Android app is created to display a basic Stopwatch.
The layout for Stopwatch includes:
- A TextView: showing how much time has passed
- Three Buttons:
- Start: To start the stopwatch
- Stop: To stop the stopwatch
- Reset: To reset the stopwatch to 00:00:00
Steps to create the Stopwatch:
- Create a new project for Stopwatch App
- Add String resources
- Update the Stopwatch layout code
- Update the code for activity
Below are the steps one by one in detail:
- Create a new project for Stopwatch App
- Create a new Android project for an application named “Stopwatch” with a company domain of “geeksforgeeks.org”, making the package name org.neveropen.stopwatch.
- The minimum SDK should be API 14 so it can run on almost all devices.
- An empty activity called “StopwatchActivity” and a layout called “activity_stopwatch” will be created.
- Add String resources
We are going to use three String values in our stopwatch layout, one for the text value of each button. These values are String resources, so they need to be added to strings.xml. Add the String values below to your version of strings.xml:Strings.xml
<
resources
>
<
string
name
=
"app_name"
>GFG|Stopwatch</
string
>
<
string
name
=
"start"
>Start</
string
>
<
string
name
=
"stop"
>Stop</
string
>
<
string
name
=
"reset"
>Reset</
string
>
</
resources
>
- Update the Stopwatch layout code
Here is the XML for the layout. It describes a single text view that’s used to display the timer, and three buttons to control the stopwatch. Replace the XML currently in activity_stopwatch.xml with the XML shown here:activity_stopwatch.xml
<?
xml
version
=
"1.0"
encoding
=
"utf-8"
?>
<
LinearLayout
android:layout_width
=
"match_parent"
android:layout_height
=
"match_parent"
android:orientation
=
"vertical"
<!-- Add some background color -->
android:background="#0F9D58"
android:padding="16dp"
tools:context="org.neveropen.stopwatch.StopwatchActivity">
<
TextView
<!-- We will use text view "time_view" -->
<!-- to display the number of seconds. -->
android:id="@+id/time_view"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
<!-- These below attributes make the stopwatch -->
<!-- timer nice and big. -->
android:textAppearance="@android:style/TextAppearance.Large"
android:textSize="56sp" />
<
Button
<!-- This code is for the Start button.
android:id="@+id/start_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:layout_marginTop="20dp"
<!-- When it gets clicked, the Start button -->
<!-- calls the onClickStart() method. -->
android:onClick="onClickStart"
android:text="@string/start" />
<
Button
<!-- This code is for the Stop button. -->
android:id="@+id/stop_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:layout_marginTop="8dp"
<!-- When it gets clicked, the Stop button -->
<!-- calls the onClickStop() method. -->
android:onClick="onClickStop"
android:text="@string/stop" />
<
Button
<!-- This code is for Reset button. -->
android:id="@+id/reset_button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal"
android:layout_marginTop="8dp"
<!-- When it gets clicked, the Reset button -->
<!-- calls the onClickReset() method. -->
android:onClick="onClickReset"
android:text="@string/reset" />
</
LinearLayout
>
- How the activity code will work
The layout defines three buttons that we will use to control the stopwatch. Each button uses its onClick attribute to specify which method in the activity should run when the button is clicked. When the Start button is clicked, the onClickStart() method gets called, when the Stop button is clicked the onClickStop() method gets called, and when the Reset button is clicked the onClickReset() method gets called. We will use these methods to start, stop and reset the stopwatch.We will update the stopwatch using a method we will create called runTimer(). The runTimer() method will run code every second to check whether the stopwatch is running, and, if it is, increment the number of seconds and display the number of seconds in the text view.
To help us with this, we will use two private variables to record the state of the stopwatch. We will use an int called seconds to track how many seconds have passed since the stopwatch started running, and a boolean called running to record whether the stopwatch is currently running.
We will start by writing the code for the buttons, and then we will look at the runTimer() method.
- Add code for the buttons When the user clicks on the Start button, we will set the running variable to true so that the stopwatch will start. When the user clicks on the Stop button, we will set running to false so that the stopwatch stops running. If the user clicks on the Reset button, we will set running to false and seconds to 0 so that the stopwatch is reset and stops running.
- The runTimer() method The next thing we need to do is to create the runTimer() method. This method will get a reference to the text view in the layout; format the contents of the seconds variable into hours, minutes, and seconds; and then display the results in the text view. If the running variable is set to true, it will increment the seconds variable.
- Handlers allow you to schedule code A Handler is an Android class you can use to schedule code that should be run at some point in the future. You can also use it to post code that needs to run on a different thread than the main Android thread. In our case, we are going to use a Handler to schedule the stopwatch code to run every second.
To use the Handler, you wrap the code you wish to schedule in a Runnable object, and then use the Handle post() and postDelayed() methods to specify when you want the code to run. - The post() method The post() method posts code that needs to be run as soon as possible(which is usually immediately). This method takes one parameter, an object of type Runnable. A Runnable object in Androidville is just like a Runnable in plain old Java: a job you want to run. You put the code you want to run in the Runnable’s run() method, and the Handler will make sure the code is run as soon as possible.
- The postDelayed() method The postDelayed() method works in a similar way to the post() method except that you use it to post code that should be run in the future. The postDelayed() method takes two parameters: a Runnable and a long. The Runnable contains the code you want to run in its run() method, and the long specifies the number of milliseconds you wish to delay the code by. The code will run as soon as possible after the delay.
Below is the following code to StopwatchActivity.java:
StopwatchActivity.java
package
org.neveropen.stopwatch;
import
android.app.Activity;
import
android.os.Handler;
import
android.view.View;
import
android.os.Bundle;
import
java.util.Locale;
import
android.widget.TextView;
public
class
StopwatchActivity
extends
Activity {
// Use seconds, running and wasRunning respectively
// to record the number of seconds passed,
// whether the stopwatch is running and
// whether the stopwatch was running
// before the activity was paused.
// Number of seconds displayed
// on the stopwatch.
private
int
seconds =
0
;
// Is the stopwatch running?
private
boolean
running;
private
boolean
wasRunning;
@Override
protected
void
onCreate(Bundle savedInstanceState)
{
super
.onCreate(savedInstanceState);
setContentView(R.layout.activity_stopwatch);
if
(savedInstanceState !=
null
) {
// Get the previous state of the stopwatch
// if the activity has been
// destroyed and recreated.
seconds
= savedInstanceState
.getInt(
"seconds"
);
running
= savedInstanceState
.getBoolean(
"running"
);
wasRunning
= savedInstanceState
.getBoolean(
"wasRunning"
);
}
runTimer();
}
// Save the state of the stopwatch
// if it's about to be destroyed.
@Override
public
void
onSaveInstanceState(
Bundle savedInstanceState)
{
savedInstanceState
.putInt(
"seconds"
, seconds);
savedInstanceState
.putBoolean(
"running"
, running);
savedInstanceState
.putBoolean(
"wasRunning"
, wasRunning);
}
// If the activity is paused,
// stop the stopwatch.
@Override
protected
void
onPause()
{
super
.onPause();
wasRunning = running;
running =
false
;
}
// If the activity is resumed,
// start the stopwatch
// again if it was running previously.
@Override
protected
void
onResume()
{
super
.onResume();
if
(wasRunning) {
running =
true
;
}
}
// Start the stopwatch running
// when the Start button is clicked.
// Below method gets called
// when the Start button is clicked.
public
void
onClickStart(View view)
{
running =
true
;
}
// Stop the stopwatch running
// when the Stop button is clicked.
// Below method gets called
// when the Stop button is clicked.
public
void
onClickStop(View view)
{
running =
false
;
}
// Reset the stopwatch when
// the Reset button is clicked.
// Below method gets called
// when the Reset button is clicked.
public
void
onClickReset(View view)
{
running =
false
;
seconds =
0
;
}
// Sets the NUmber of seconds on the timer.
// The runTimer() method uses a Handler
// to increment the seconds and
// update the text view.
private
void
runTimer()
{
// Get the text view.
final
TextView timeView
= (TextView)findViewById(
R.id.time_view);
// Creates a new Handler
final
Handler handler
=
new
Handler();
// Call the post() method,
// passing in a new Runnable.
// The post() method processes
// code without a delay,
// so the code in the Runnable
// will run almost immediately.
handler.post(
new
Runnable() {
@Override
public
void
run()
{
int
hours = seconds /
3600
;
int
minutes = (seconds %
3600
) /
60
;
int
secs = seconds %
60
;
// Format the seconds into hours, minutes,
// and seconds.
String time
= String
.format(Locale.getDefault(),
"%d:%02d:%02d"
, hours,
minutes, secs);
// Set the text view text.
timeView.setText(time);
// If running is true, increment the
// seconds variable.
if
(running) {
seconds++;
}
// Post the code again
// with a delay of 1 second.
handler.postDelayed(
this
,
1000
);
}
});
}
}
Output: