Saturday, January 11, 2025
Google search engine
HomeLanguagesJavaAndroid – Create a New Playlist in Your Spotify Account using Spotify...

Android – Create a New Playlist in Your Spotify Account using Spotify Web API

Spotify is one of the world’s most popular music streaming platforms, and it has a vast collection of songs, albums, playlists, and podcasts. With the Spotify Web API, developers can create amazing applications that interact with the Spotify service. Using the Web API’s endpoints, one can get detailed metadata about artists, albums, and tracks directly from the Spotify Data Catalogue. It also provides access to the currently logged-in user’s data, such as the playlists and tracks saved in the Your Music Library of the user. 

In this article, we will create an Android app using which we can create a new playlist in the logged-in user’s Spotify Account. Basics of Java and Android App Development knowledge are prerequisites for this article. We will also use the Volley library to make the networking calls to the Spotify Web API and the GSON library to parse the returned JSON data.

Step By Step Implementation

Step 1: Create a New Project in Android Studio

Please refer to  How to Create/Start a New Project in Android Studio to learn how to create a new project. Name the project “My Spotify App” (or any name of your choice) and select Java as the programming language. A new project will be created with a Main Activity and its XML file as follows:

 

Step 2: Create an app in the Spotify Dashboard

  • Head over to the Spotify Dashboard and log in using your Spotify account. 
  • Click on “Create an app.”
  • Write the app name (My Spotify App), give it a description, accept the T&C, and click on create button.

Once the app is created, the Dashboard will look like this:

 

Step 3: Finalize project settings in the dashboard

We will add a Redirect URI, our package name, and the SHA-1 key of our Android Studio. The Redirect URI is used to redirect back to your application from a web browser (in this case, the web page for authentication). Therefore we create a unique URI for our app.

The package name of an Android app is a unique identifier that distinguishes it from other apps on the device and the Google Play store. Similarly, the SHA-1 key is used to uniquely identify your Android Studio installed on your PC. Refer to this article to learn how to get the SHA-1 key in Android Studio.

These details are needed to restrict our API key to only work with our app and prevent unauthorized use by others. Follow the following steps to add these details:

  • In the project dashboard, click on “Edit Settings.”
  • Under “Redirect URIs”, add your package name as a redirect URI: <package-name>://callback, which in this case is:
com.example.myspotifyapp://callback
  • Also, add your package name under the “Android Packages” section, along with your Android Studio’s unique SHA-1 key. 

Click on Save, and you will be done with the project settings in the Spotify Dashboard. Your app will currently be in development mode and will only support a maximum of 25 users. To make your app accessible to more than 25 users, you will need to Submit a quota extension request to the Spotify team.

Step 4: Add the necessary dependencies and permissions

We will need to authenticate the user to make calls to the Spotify Web API. We will use the Spotify Android auth library, which is responsible for authenticating the user and fetching the authorization code/access token that can subsequently be used to play music or in requests to the API. 

We will use the Volley library to make HTTP calls to the API and the GSON library to parse the results and use them in our app. We’ll check the Maven Central repository to get the latest version of the libraries. The links for each library are mentioned below. Choose the latest version and Gradle from the respective dropdowns, copy the dependency and, paste it into your module-level gradle file, then sync your project. 

The Maven Central repository will look as follows:

 

Since Spotify Auth library version 2.0.0, we must also provide the scheme and host of the redirect URI our app uses for authorizing in our module-level build.gradle file. In this case, we need to paste the following in our defaultConfig scope within android:

manifestPlaceholders = [redirectSchemeName: "com.example.myspotifyapp", redirectHostName: "callback"]

Sync your project. The Gradle file will look like this:

 

We must also add internet permission in our app’s manifest file. Paste the following snippet in your AndroidManifest.xml file above the application tag:

<uses-permission android:name="android.permission.INTERNET" />

Step 5: Add the necessary string values in the strings.xml file

Adding the strings like keys etc. which are used in many places across the android app in the strings.xml file, is considered a best practice. We’ll store our Client ID, Redirect URI, and the shared preferences key in the string.xml file. You can get your Client ID and redirect URI from the Spotify Developer Dashboard. We’ll use “Spotify” as our key to access shared preferences. The code for the strings.xml file will look as follows:

XML




<resources>
    <string name="app_name">My Spotify App</string>
    <string name="CLIENT_ID">YOUR_CLIENT_ID</string>
    <string name="REDIRECT_URI">YOUR_REDIRECT_URI</string>
    <string name="shared_pref_key">Spotify</string>
</resources>


Step 6: Add the code for authentication

Create a new Activity, name it LoginActivity in Android Studio, and select it as the launcher activity. We will write the code for authentication in this activity and launch the MainActivity once authentication is successful. Once authentication is done, we receive a token, which is later used for authorization in making requests to the API. We’ll store the token in android’s persistent storage Shared Preferences for later use. We also need to define the scope to access the specific information we’ll be accessing from our user’s account. For example, the user-read-email scope is used to read the linked email address of the user. You can read more about scopes here. The code for LoginActivity.java is as shown below:

Java




package com.example.myspotifyapp;
  
import android.content.Intent;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.util.Log;
import android.widget.Toast;
import androidx.appcompat.app.AppCompatActivity;
import com.android.volley.RequestQueue;
import com.android.volley.toolbox.Volley;
import com.spotify.sdk.android.auth.AuthorizationClient;
import com.spotify.sdk.android.auth.AuthorizationRequest;
import com.spotify.sdk.android.auth.AuthorizationResponse;
  
public class LoginActivity extends AppCompatActivity {
  
    // SharedPreferences to save the returned token
    private SharedPreferences.Editor editor;
    private SharedPreferences sharedPreferences;
  
    // Volley request queue for API calls
    private RequestQueue requestQueue;
  
    // Request code that we'll use to judge if the result of
    // our call comes from the correct activity
    private static final int REQUEST_CODE = 1234;
  
    // The scopes needed for our app
    private static final String SCOPES
        = "user-read-email,user-read-private,playlist-modify-private,playlist-modify-public";
  
    @Override
    protected void onCreate(Bundle savedInstanceState)
    {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_login);
  
        // Initialize shared preferences and volley request queue
        sharedPreferences = this.getSharedPreferences(
            getString(R.string.shared_pref_key),
            MODE_PRIVATE);
        requestQueue = Volley.newRequestQueue(this);
  
        // Code to authenticate our Spotify app
        AuthorizationRequest.Builder builder
            = new AuthorizationRequest.Builder(
                getString(R.string.CLIENT_ID),
                AuthorizationResponse.Type.TOKEN,
                getString(R.string.REDIRECT_URI));
  
        builder.setScopes(new String[] { SCOPES });
        AuthorizationRequest request = builder.build();
  
        // Opens the Login Activity Web page for spotify for auth
        AuthorizationClient.openLoginActivity(
            this, REQUEST_CODE, request);
    }
  
    // This method is used to get the result from any
    // launched activity that is expected to return something
    protected void onActivityResult(int requestCode,
                                    int resultCode,
                                    Intent intent)
    {
        super.onActivityResult(requestCode, resultCode,
                               intent);
  
        // Check if result comes from the our Spotify login activity
        if (requestCode == REQUEST_CODE) {
            AuthorizationResponse response
                = AuthorizationClient.getResponse(
                    resultCode, intent);
  
            switch (response.getType()) {
            // If the response was successful and contains our auth token
            case TOKEN:
                // we'll store the auth token in shared preferences
                editor = getSharedPreferences(
                             getString(
                                 R.string.shared_pref_key),
                             MODE_PRIVATE)
                             .edit();
  
                editor.putString("token",
                                 response.getAccessToken());
                editor.apply();
  
                // We'll make a toast in our app to confirm successful auth
                Toast.makeText(this, "Auth successful",Toast.LENGTH_SHORT)
                    .show();
  
                // We'll start the main activity after auth is successful
                startActivity(new Intent(LoginActivity.this,
                               MainActivity.class));
  
                break;
  
            // Auth returned an error
            case ERROR:
                Log.d("LoginActivity", response.getError());
                break;
  
            default:
                Log.d("LoginActivity", response.toString());
            }
        }
    }
}


The XML file of LoginActivity, MainActivity, and its Java file will be the same as the default for now. You can now run the app and check if auth is running successfully. You should see a toast saying “Auth successful”, and the MainActivity should pop up right after. 

Step 7: Create helper classes to get logged-in user’s details

Now that the authentication is done, we will try to get some user details and view them in the MainActivity. We need to get the user details so that we can get hold of the user’s Spotify ID, which will be needed to create a playlist in that user’s account and much more stuff. You can read more about the endpoint for getting user details here. We will first need to create a model User class to serialize the response from the endpoint. We will create variables for name, email, and user ID because that is the only information we currently need.

Java




public class User {
     // the variable names of these strings must be
      // exactly same as in the API response     
    public String display_name;
    public String email;
    public String id;
  
    public String getDisplayName() {
        return display_name;
    }
  
    public String getEmail() {
        return email;
    }
  
  
    public String getId() {
        return id;
    }
}


We will now create a UserInfo class to fetch the response from the API’s endpoint. We will make the call to the API using the Volley library and parse it into the User class using GSON.

Java




import android.content.SharedPreferences;
import com.android.volley.AuthFailureError;
import com.android.volley.RequestQueue;
import com.android.volley.toolbox.JsonObjectRequest;
import com.google.gson.Gson;
import java.util.HashMap;
import java.util.Map;
  
public class UserInfo
{
    // the endpoint for getting user info
    private static final String URL = "https://api.spotify.com/v1/me";
    private SharedPreferences sharedPreferences;
    private RequestQueue queue;
    private User user;
  
    // the constructor requires the 
    // Volley request queue and shared preferences
    // from the calling activity
    public UserInfo(RequestQueue queue, SharedPreferences sharedPreferences) {
        this.queue = queue;
        this.sharedPreferences = sharedPreferences;
    }
  
    public User getUser() {
        return user;
    }
  
    // method to get the user's data
    public void get(final VolleyCallBack callBack) {
          
        // We use JsonObjectRequest method of volley library to
        // retrieve a JSONObject response body at a given URL
        JsonObjectRequest jsonObjectRequest = new JsonObjectRequest(URL,
                null, response -> 
        {
            // initialize the GSON library
            Gson gson = new Gson(); 
  
            // serialize the response into user object
            user = gson.fromJson(response.toString(), User.class);
  
              // indicate successful call
            callBack.onSuccess(); 
        }, error -> get(() -> {
  
        })) {
            // We need to add headers to the call for authorization
            @Override
            public Map<String, String> getHeaders() throws AuthFailureError {
                Map<String, String> headers = new HashMap<>();
                  
                // get the auth token from shared preferences
                String token = sharedPreferences.getString("token", "");
                String auth = "Bearer " + token;
                  
                // add it in headers
                headers.put("Authorization", auth);
                return headers;
            }
        };
        // add the JSON object request call to 
          // the queue of volley library to make the call
        queue.add(jsonObjectRequest);
    }
}


Step 8: Display the details in the Main Activity

Add a TextView on which we will display the user’s details. The XML code for the Main Activity class is given below:

XML




<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">
  
    <TextView
        android:id="@+id/userDetails"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="30dp"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
    
</androidx.constraintlayout.widget.ConstraintLayout>


Now we will write the code in the MainActivity.java file to display the user’s data. 

Java




import androidx.appcompat.app.AppCompatActivity;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.widget.TextView;
  
public class MainActivity extends AppCompatActivity {
    
    TextView userDetails;
    SharedPreferences sharedPreferences;
  
    private static final String TAG = "MainActivity";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
  
        // get the user details TextView
        userDetails = findViewById(R.id.userDetails);
          
        // get the shared preferences
        sharedPreferences = this.getSharedPreferences(getString(R.string.shared_pref_key),
                MODE_PRIVATE);
  
        // retrieve the name and email from the shared preferences
        String name = sharedPreferences.getString("display_name", "Couldn't retrieve name");
        String email = sharedPreferences.getString("email", "Couldn't get token");
  
        // display the details in the text view
        userDetails.setText("Name: " +name);
        userDetails.append("\nEmail ID: " +email);
    }
  
}


You can now run the app. You will be able to see your name and email on the Main Activity. A sample screenshot is attached below: 

 

Step 9: Add code to create a new playlist in your Spotify account

We will now programmatically create a playlist in our Spotify account. The details of the API for creating a new playlist can be found here. First, we will create a button in the Main Activity for creating a new playlist. The XML code is as follows:

XML




<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">
  
    <Button
        android:id="@+id/createPlaylistBtn"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_margin="30dp"
        android:text="Create New Playlist"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent" />
  
    <TextView
        android:id="@+id/userDetails"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="30dp"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
    
</androidx.constraintlayout.widget.ConstraintLayout>


Now we will make the request to the endpoint on the button click in our Main Activity. The Java code is as follows:

Java




import androidx.appcompat.app.AppCompatActivity;
import android.content.SharedPreferences;
import android.os.Bundle;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
  
import com.android.volley.AuthFailureError;
import com.android.volley.Request;
import com.android.volley.RequestQueue;
import com.android.volley.toolbox.JsonObjectRequest;
import com.android.volley.toolbox.Volley;
  
import org.json.JSONException;
import org.json.JSONObject;
  
import java.util.HashMap;
import java.util.Map;
  
public class MainActivity extends AppCompatActivity {
    TextView userDetails;
    Button createPlaylistBtn;
    SharedPreferences sharedPreferences;
    private RequestQueue queue;
  
    private static final String TAG = "MainActivity";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
  
        // get the user details TextView and 
          // the create playlist button
        userDetails = findViewById(R.id.userDetails);
        createPlaylistBtn = findViewById(R.id.createPlaylistBtn);
  
        // get the shared preferences
        sharedPreferences = this.getSharedPreferences(getString(R.string.shared_pref_key),
                MODE_PRIVATE);
  
        // retrieve the name and email from the shared preferences
        String name = sharedPreferences.getString("display_name", "Couldn't retrieve name");
        String email = sharedPreferences.getString("email", "Couldn't get token");
  
        // display the details in the text view
        userDetails.setText("Name: " +name);
        userDetails.append("\nEmail ID: " +email);
  
        // initiate the Volley request queue
        queue = Volley.newRequestQueue(this);
  
        // add a click listener to button
        createPlaylistBtn.setOnClickListener(view -> {
            createTestPlaylist();
        });
    }
  
    // Method to create a new playlist
    private void createTestPlaylist()
    {
        // Prepare put payload for the playlist, 
          // that will contain
        // the name and description of our playlist
        JSONObject data = new JSONObject();
  
        try{
            data.put("name", "The Empty Canvas");
            data.put("description", "A playlist created by me using Spotify's Web API :)");
        } catch (JSONException e) {
            throw new RuntimeException(e);
        }
  
        // the API's endpoint
        String endpoint = "https://api.spotify.com/v1/users/"
                + sharedPreferences.getString("userId", "Couldn't get userid") + "/playlists";
  
        // We use JsonObjectRequest method of volley library to
        // retrieve a JSONObject response body at a given URL
        JsonObjectRequest jsonObjectRequest = new JsonObjectRequest(Request.Method.POST, endpoint,
                data, response -> {
            // upon successful creation of the playlist, we make a toast
            Toast.makeText(this, "\"The Empty Canvas\" playlist created", Toast.LENGTH_SHORT).show();
        },
                error -> {
                    // if error occurs, we create a toast
                    Toast.makeText(this, error.toString(), Toast.LENGTH_SHORT).show();
                }) {
            // We add the headers to the call for authorization
            @Override
            public Map<String, String> getHeaders() throws AuthFailureError {
                Map<String, String> headers = new HashMap<>();
  
                // get the auth token from shared preferences
                String token = sharedPreferences.getString("token", "");
                String auth = "Bearer " + token;
  
                // add it in headers
                headers.put("Authorization", auth);
                headers.put("Content-Type", "application/json");
                return headers;
            }
        };
  
        // add the JSON object request call to the queue 
          // of volley library to make the call
        queue.add(jsonObjectRequest);
    }
}


You can now run the app, and after clicking the button, you’ll find a playlist has been created in your Spotify account. A sample app and Spotify screenshots have been attached below. 

 

 

RELATED ARTICLES

Most Popular

Recent Comments