
The Tic-tac-toe app in the android studio is a beginner-friendly application, where the first screen will ask for both the player’s name, and then the game will start on another page where there will be a grid and two player names.
After, the game ends a dialog box will appear which will show the winner’s name and a start again button. When we click on the start again button then again the game will start from scratch.
ic_xicon.png

ic_oicon.png

ximage.png

oimage.png

pagebkg.png

Step 1: Open Android Studio, Create New Project, Choose Empty Activity and name the project “TicTacToe”.

Step 2: colors.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<color name="purple_200">#FFBB86FC</color>
<color name="purple_500">#FF6200EE</color>
<color name="purple_700">#FF3700B3</color>
<color name="teal_200">#FF03DAC5</color>
<color name="teal_700">#FF018786</color>
<color name="black">#FF000000</color>
<color name="white">#FFFFFFFF</color>
<color name="lavender">#8692f7</color>
</resources>
Step 3: themes.xml
<resources xmlns:tools="http://schemas.android.com/tools">
<!-- Base application theme. -->
<style name="Theme.TicTacToe" parent="Theme.MaterialComponents.DayNight.NoActionBar">
<!-- Primary brand color. -->
<item name="colorPrimary">@color/lavender</item>
<item name="colorPrimaryVariant">@color/lavender</item>
<item name="colorOnPrimary">@color/white</item>
<!-- Secondary brand color. -->
<item name="colorSecondary">@color/teal_200</item>
<item name="colorSecondaryVariant">@color/teal_700</item>
<item name="colorOnSecondary">@color/black</item>
<!-- Status bar color. -->
<item name="android:statusBarColor">?attr/colorPrimaryVariant</item>
<!-- Customize your theme here. -->
</style>
</resources>
Step 3: black_border.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<solid
android:color="@color/white"/>
<corners
android:radius="30dp"/>
<stroke
android:color="@color/black"
android:width="2dp"/>
</shape>
lavender_border.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<stroke
android:width="2dp"
android:color="@color/lavender"/>
<corners
android:radius="30dp"/>
</shape>
white_box.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android"
android:shape="rectangle">
<solid
android:color="@color/white"/>
<corners
android:radius="30dp"/>
</shape>
Download 5 Images from above.
Step 4: gradle: app
buildFeatures {
viewBinding true
}
Step 5: Create two activities: AddPlayer and ResultDialog and make sure to set AddPlayer as the launcher activity in AndroidManifest.xml instead of MainActivity.
Step 6: activity_add_players.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
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:layout_height="match_parent"
android:orientation="vertical"
android:gravity="center"
android:background="@drawable/pagebkg"
tools:context=".AddPlayers">
<androidx.cardview.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="30dp"
app:cardCornerRadius="30dp"
app:cardElevation="20dp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical"
android:layout_gravity="center_horizontal"
android:padding="24dp"
android:background="@drawable/lavender_border">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="ENTER \n PLAYERS NAMES"
android:textAlignment="center"
android:textSize="36sp"
android:textStyle="bold"
android:textColor="@color/lavender"/>
<EditText
android:layout_width="match_parent"
android:layout_height="50dp"
android:id="@+id/playerOne"
android:background="@drawable/lavender_border"
android:layout_marginTop="40dp"
android:padding="8dp"
android:maxLines="1"
android:textStyle="italic"
android:hint="Enter player one name"
android:textColor="@color/lavender"
android:drawableLeft="@drawable/ic_xicon"
android:drawablePadding="8dp"/>
<EditText
android:layout_width="match_parent"
android:layout_height="50dp"
android:id="@+id/playerTwo"
android:background="@drawable/lavender_border"
android:layout_marginTop="20dp"
android:padding="8dp"
android:maxLines="1"
android:textStyle="italic"
android:hint="Enter player two name"
android:textColor="@color/lavender"
android:drawableLeft="@drawable/ic_oicon"
android:drawablePadding="8dp"/>
<Button
android:layout_width="match_parent"
android:layout_height="60dp"
android:layout_marginTop="30dp"
android:id="@+id/startGameButton"
android:text="Start Game"
android:textSize="18sp"
app:cornerRadius = "20dp"/>
</LinearLayout>
</androidx.cardview.widget.CardView>
</RelativeLayout>
Step 7: activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
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:layout_height="match_parent"
android:background="@color/lavender"
tools:context=".MainActivity">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:gravity="center"
android:layout_above="@id/container"
android:layout_alignParentTop="true"
android:orientation="horizontal">
<LinearLayout
android:layout_width="120dp"
android:layout_height="wrap_content"
android:id="@+id/playerOneLayout"
android:orientation="vertical"
android:gravity="center"
android:background="@drawable/white_box">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="20dp"
android:text="Player One"
android:id="@+id/playerOneName"
android:textSize="20sp"
android:gravity="center"
android:textStyle="bold"
android:textColor="@color/lavender"
android:maxLines="1"/>
<ImageView
android:layout_width="40dp"
android:layout_height="40dp"
android:src="@drawable/ximage"
android:layout_marginTop="20dp"
android:layout_marginBottom="20dp"
android:layout_gravity="center"/>
</LinearLayout>
<LinearLayout
android:layout_width="120dp"
android:layout_height="wrap_content"
android:layout_marginStart="10dp"
android:id="@+id/playerTwoLayout"
android:orientation="vertical"
android:gravity="center"
android:background="@drawable/white_box">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="20dp"
android:text="Player Two"
android:id="@+id/playerTwoName"
android:textSize="20sp"
android:gravity="center"
android:textStyle="bold"
android:textColor="@color/lavender"
android:maxLines="1"/>
<ImageView
android:layout_width="40dp"
android:layout_height="40dp"
android:src="@drawable/oimage"
android:layout_marginTop="20dp"
android:layout_marginBottom="20dp"
android:layout_gravity="center"/>
</LinearLayout>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/container"
android:layout_alignParentBottom="true"
android:layout_marginStart="10dp"
android:layout_marginEnd="10dp"
android:layout_marginBottom="50dp"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:weightSum="3">
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image1"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image2"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image3"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:weightSum="3">
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image4"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image5"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image6"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:weightSum="3">
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image7"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:adjustViewBounds="true"
android:padding="20dp"/>
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image8"
android:layout_weight="1"
android:adjustViewBounds="true"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:padding="20dp"/>
<ImageView
android:layout_width="0dp"
android:layout_height="115dp"
android:id="@+id/image9"
android:layout_weight="1"
android:background="@drawable/white_box"
android:layout_marginTop="10dp"
android:layout_marginEnd="5dp"
android:layout_marginStart="10dp"
android:padding="20dp"
android:adjustViewBounds="true"/>
</LinearLayout>
</LinearLayout>
</RelativeLayout>
Step 8: activity_result_dialog.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
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:layout_height="match_parent"
tools:context=".ResultDialog">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/lavender"
android:orientation="vertical">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/messageText"
android:gravity="center"
android:text="Results"
android:textColor="@color/white"
android:layout_marginTop="20dp"
android:layout_marginStart="20dp"
android:layout_marginEnd="20dp"
android:textSize="18sp"
android:textStyle="bold"/>
<Button
android:layout_width="match_parent"
android:layout_height="60dp"
android:id="@+id/startAgainButton"
android:textSize="18sp"
android:textStyle="bold"
android:backgroundTint="@color/white"
android:text="Start Again"
android:layout_margin="20dp"
android:textColor="@color/lavender"
app:cornerRadius = "20dp"/>
</LinearLayout>
</RelativeLayout>
Step 9: AddPlayers.java
package com.example.tictactoe;
import androidx.appcompat.app.AppCompatActivity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;
public class AddPlayers extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_add_players);
EditText playerOne = findViewById(R.id.playerOne);
EditText playerTwo = findViewById(R.id.playerTwo);
Button startGameButton = findViewById(R.id.startGameButton);
startGameButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
String getPlayerOneName = playerOne.getText().toString();
String getPlayerTwoName = playerTwo.getText().toString();
if (getPlayerOneName.isEmpty() || getPlayerTwoName.isEmpty()) {
Toast.makeText(AddPlayers.this, "Please enter player name", Toast.LENGTH_SHORT).show();
} else {
Intent intent = new Intent(AddPlayers.this, MainActivity.class);
intent.putExtra("playerOne", getPlayerOneName);
intent.putExtra("playerTwo", getPlayerTwoName);
startActivity(intent);
}
}
});
}
}
Step 10: MainActivity.java
package com.example.tictactoe;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.ImageView;
import com.example.tictactoe.databinding.ActivityMainBinding;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
ActivityMainBinding binding;
private final List<int[]> combinationList = new ArrayList<>();
private int[] boxPositions = {0,0,0,0,0,0,0,0,0}; //9 zero
private int playerTurn = 1;
private int totalSelectedBoxes = 1;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
binding = ActivityMainBinding.inflate(getLayoutInflater());
setContentView(binding.getRoot());
combinationList.add(new int[] {0,1,2});
combinationList.add(new int[] {3,4,5});
combinationList.add(new int[] {6,7,8});
combinationList.add(new int[] {0,3,6});
combinationList.add(new int[] {1,4,7});
combinationList.add(new int[] {2,5,8});
combinationList.add(new int[] {2,4,6});
combinationList.add(new int[] {0,4,8});
String getPlayerOneName = getIntent().getStringExtra("playerOne");
String getPlayerTwoName = getIntent().getStringExtra("playerTwo");
binding.playerOneName.setText(getPlayerOneName);
binding.playerTwoName.setText(getPlayerTwoName);
binding.image1.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(0)){
performAction((ImageView) view, 0);
}
}
});
binding.image2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(1)){
performAction((ImageView) view, 1);
}
}
});
binding.image3.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(2)){
performAction((ImageView) view, 2);
}
}
});
binding.image4.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(3)){
performAction((ImageView) view, 3);
}
}
});
binding.image5.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(4)){
performAction((ImageView) view, 4);
}
}
});
binding.image6.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(5)){
performAction((ImageView) view, 5);
}
}
});
binding.image7.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(6)){
performAction((ImageView) view, 6);
}
}
});
binding.image8.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(7)){
performAction((ImageView) view, 7);
}
}
});
binding.image9.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (isBoxSelectable(8)){
performAction((ImageView) view, 8);
}
}
});
}
private void performAction(ImageView imageView, int selectedBoxPosition) {
boxPositions[selectedBoxPosition] = playerTurn;
if (playerTurn == 1) {
imageView.setImageResource(R.drawable.ximage);
if (checkResults()) {
ResultDialog resultDialog = new ResultDialog(MainActivity.this, binding.playerOneName.getText().toString()
+ " is a Winner!", MainActivity.this);
resultDialog.setCancelable(false);
resultDialog.show();
} else if(totalSelectedBoxes == 9) {
ResultDialog resultDialog = new ResultDialog(MainActivity.this, "Match Draw", MainActivity.this);
resultDialog.setCancelable(false);
resultDialog.show();
} else {
changePlayerTurn(2);
totalSelectedBoxes++;
}
} else {
imageView.setImageResource(R.drawable.oimage);
if (checkResults()) {
ResultDialog resultDialog = new ResultDialog(MainActivity.this, binding.playerTwoName.getText().toString()
+ " is a Winner!", MainActivity.this);
resultDialog.setCancelable(false);
resultDialog.show();
} else if(totalSelectedBoxes == 9) {
ResultDialog resultDialog = new ResultDialog(MainActivity.this, "Match Draw", MainActivity.this);
resultDialog.setCancelable(false);
resultDialog.show();
} else {
changePlayerTurn(1);
totalSelectedBoxes++;
}
}
}
private void changePlayerTurn(int currentPlayerTurn) {
playerTurn = currentPlayerTurn;
if (playerTurn == 1) {
binding.playerOneLayout.setBackgroundResource(R.drawable.black_border);
binding.playerTwoLayout.setBackgroundResource(R.drawable.white_box);
} else {
binding.playerTwoLayout.setBackgroundResource(R.drawable.black_border);
binding.playerOneLayout.setBackgroundResource(R.drawable.white_box);
}
}
private boolean checkResults(){
boolean response = false;
for (int i = 0; i < combinationList.size(); i++){
final int[] combination = combinationList.get(i);
if (boxPositions[combination[0]] == playerTurn && boxPositions[combination[1]] == playerTurn &&
boxPositions[combination[2]] == playerTurn) {
response = true;
}
}
return response;
}
private boolean isBoxSelectable(int boxPosition) {
boolean response = false;
if (boxPositions[boxPosition] == 0) {
response = true;
}
return response;
}
public void restartMatch(){
boxPositions = new int[] {0,0,0,0,0,0,0,0,0}; //9 zero
playerTurn = 1;
totalSelectedBoxes = 1;
binding.image1.setImageResource(R.drawable.white_box);
binding.image2.setImageResource(R.drawable.white_box);
binding.image3.setImageResource(R.drawable.white_box);
binding.image4.setImageResource(R.drawable.white_box);
binding.image5.setImageResource(R.drawable.white_box);
binding.image6.setImageResource(R.drawable.white_box);
binding.image7.setImageResource(R.drawable.white_box);
binding.image8.setImageResource(R.drawable.white_box);
binding.image9.setImageResource(R.drawable.white_box);
}
}
Step 11: ResultDialog.java
package com.example.tictactoe;
import androidx.annotation.NonNull;
import androidx.appcompat.app.AppCompatActivity;
import android.app.Dialog;
import android.content.Context;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class ResultDialog extends Dialog {
private final String message;
private final MainActivity mainActivity;
public ResultDialog(@NonNull Context context, String message, MainActivity mainActivity) {
super(context);
this.message = message;
this.mainActivity = mainActivity;
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_result_dialog);
TextView messageText = findViewById(R.id.messageText);
Button startAgainButton = findViewById(R.id.startAgainButton);
messageText.setText(message);
startAgainButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
mainActivity.restartMatch();
dismiss();
}
});
}
}

If you have any queries or errors, please feel free to comment below 😉
Please subscribe to my youtube channel ❤️
Check our previous post: Navigation Drawer Menu in Android Studio using Java
Check our Youtube Video: Tic-Tac-Toe Game in Android Studio