Blog post

Persisting the Tabs State on Android

Illustration: Persisting the Tabs State on Android

One of the most requested use cases regarding tabs is the ability to persist the tabs state between activity instances. And recently, PSPDFKit for Android shipped PersistentTabsExample as part of the Catalog app. In this blog post, I’ll walk you through the steps that were necessary to build this example.

Introduction

PSPDFKit supports opening multiple documents inside a single activity. Doing so results in the appearance of a material design tab strip for switching between these opened documents. PSPDFKit provides a full API for manipulating document tabs programmatically but does not complicate things with business-specific use cases such as restoring the state of previously opened tabs when restarting an activity. So, I am going to provide step-by-step instructions on how to build a state manager that does this.

This post is split into multiple parts:

  • The Tabs Primer section explains the basics of how multiple documents can be opened inside PdfActivity.

  • The Model Layer section describes how to persist opened documents in shared preferences.

  • The State Management section describes how to save a list of opened documents in a custom PdfActivity and start the activity with the same documents restored.

Tabs Primer

Let’s first look at how PSPDFKit models opened documents inside PdfActivity and how the tabs UI interacts with this model.

Each document loaded inside PdfActivity is represented by a DocumentDescriptor class, which encapsulates the document sources and the UI state associated with the document.

Document sources make up the list of parcelable DocumentSource instances. A document is loaded by merging all documents opened from these sources in their list order. Parcelable sources are required in order to allow transferring document descriptors when launching an activity via Intent or when saving its state in the saved state Bundle.

The UI state stored inside DocumentDescriptor is a simple Bundle, and the entire handling of the document state is managed by the PdfActivity itself: The state is set when switching to a different document and restored when returning back to the old document that already has the saved state.

Document descriptors are managed by DocumentCoordinator, which is owned by the PdfActivity. To obtain an instance, use PdfActivity#getDocumentCoordinator(). You can use this coordinator to modify a list of opened documents, change which document is visible, and more:

// Retrieve the document coordinator owned by the `PdfActivity`.
val documentCoordinator = activity.documentCoordinator

// Add a new document.
documentCoordinator.addDocument(documentDescriptor)

// Display the newly added document.
documentCoordinator.setVisibleDocument(documentDescriptor)

// Remove some other document.
documentCoordinator.removeDocument(otherDocumentDescriptor)

// Retrieve all documents.
val documents = documentCoordinator.documents

// Retrieve the visible document.
val visibleDocument = documentCoordinator.visibleDocument

Note that PdfActivity’s default tabs UI, PdfTabBar, is decoupled from the DocumentCoordinator. Whenever the user clicks on a tab, the DocumentCoordinator#setVisibleDocument() is called with the document descriptor for that tab. In addition, the tab bar responds to changes to the documents list inside DocumentCoordinator and updates the tabs UI accordingly. You can do the same in your own custom document-switching UI:

documentCoordinator.addOnDocumentVisibleListener { documentDescriptor ->
   // Called when the document is made visible.
}

documentCoordinator.addOnDocumentsChangedListener(object: DocumentCoordinator.OnDocumentsChangedListener {
    override fun onDocumentAdded(documentDescriptor: DocumentDescriptor) {
        // Called after the document has been added to the opened documents list.
    }

    override fun onDocumentRemoved(documentDescriptor: DocumentDescriptor) {
        // Called after the document has been removed from the opened documents list.
    }

    override fun onDocumentMoved(documentDescriptor: DocumentDescriptor, targetIndex: Int) {
        // Called after the document has been moved inside the opened documents list.
    }

    override fun onDocumentReplaced(oldDocument: DocumentDescriptor, newDocument: DocumentDescriptor) {
        // Called when the document has been replaced with a different document inside the opened documents list.
    }

    override fun onDocumentUpdated(documentDescriptor: DocumentDescriptor) {
        // Called after the document has been updated — for example, when its title has changed.
    }
})

Model Layer

This section describes how to build a model layer for persisting DocumentDescriptors that are opened in PdfActivity to shared preferences.

What We Are Dealing With

Shared preferences allow storing just a few simple data types — int, long, float, Boolean, String and an unordered set of String. Let’s look at the data we need to persist before we decide on the data model inside shared preferences.

DocumentDescriptor is Parcelable, but parcelables are not well suited for persistent storage. What we really need to save is just the DocumentSource and a custom title that could have been set on the document descriptor.

ℹ️ Note: We won’t store the UI state, as it does not make sense to restore the UI state for documents that were not recently opened. Similar to persisting parcelables, it’s also unfeasible to store the UI state Bundle in shared preferences. We would need to store only the data that is interesting to us — for example, the currently visible page, the selected annotation, or the text.

PSPDFKit supports a file Uri or a DataProvider as a DocumentSource. We’ll simplify this example by only dealing with file Uri document sources. If you wish to use a source for a custom DataProvider, you’ll need to store additional information in order to be able to restore DataProvider with the data required for opening the document when the list of opened documents is restored.

Another requirement we have is that we need to store an ordered list of opened document descriptors to make sure they are restored in the correct order.

Data Format

We will use a simple JSON as the format for storing a list of opened document descriptors in our preferences file. This list is going to be represented by a JSON array, with each element in a simple format:

{
  "uri": " <document_uri>",
  "title": "<document_title>"
}

We can easily serialize these JSONs as strings in our shared preferences.

Accessing Preferences

To access the shared preferences with our tabs state, we first create a separate class:

class TabsPreferences(context: Context) {

    companion object {
        const val PREFERENCES_NAME = "PSPDFKit.PersistentTabsExample"

        const val JSON_DESCRIPTOR_URI = "uri"
        const val JSON_DESCRIPTOR_TITLE = "title"

        const val PREF_DOCUMENT_DESCRIPTORS_JSON = "document_descriptors"
        const val PREF_VISIBLE_DOCUMENT_INDEX = "visible_document_index"
    }

    private val preferences = context.getSharedPreferences(PREFERENCES_NAME, Context.MODE_PRIVATE)

    /**
     * Stores the list of open document descriptors in preferences.
     */
    fun setDocumentDescriptors(descriptors: List<DocumentDescriptor>)

    /**
     * Returns the list of document descriptors stored in the shared preferences.
     */
    fun getDocumentDescriptors(context: Context): List<DocumentDescriptor>?

    /**
     * Sets the index of the currently visible document in the list of stored document descriptors.
     */
    fun setVisibleDocumentIndex(visibleDocumentIndex: Int)

    /**
     * Returns the index of the currently visible document in the list of stored document descriptors.
     */
    fun getVisibleDocumentIndex(): Int
}

Let’s look at the document descriptors serialization code:

fun setDocumentDescriptors(descriptors: List<DocumentDescriptor>) {
    // Our JSON will consist of a top-level array of document descriptors.
    val descriptorsArray = JSONArray()

    // Iterate through all descriptors to save and add them to the JSON array.
    for (descriptor in descriptors) {
        val descriptorJson = JSONObject()
        // Save the document source Uri. We expect all document sources
        // to be Uri-based for the sake of simplicity, as discussed above.
        descriptorJson.put(JSON_DESCRIPTOR_URI, descriptor.documentSource.fileUri)
        // Save the custom title set on the document descriptor.
        descriptorJson.put(JSON_DESCRIPTOR_TITLE, descriptor.customTitle)

        // Put the document descriptor JSON into the JSON array.
        descriptorsArray.put(descriptorJson)
    }

    // Put the created JSON into the shared preferences as a string.
    preferences.edit().putString(PREF_DOCUMENT_DESCRIPTORS_JSON, descriptorsArray.toString()).apply()
}

Parsing document descriptors is straightforward. However, one complication is that we need to distinguish between image documents and standard documents when creating document descriptors. We could do so by storing the isImageDocument Boolean in the descriptor’s JSON or, as we present here, by using PSPDFKit’s ImageDocumentUtils#isImageUri() utility method, which resolves the document Uri and checks whether it’s an image type:

fun getDocumentDescriptors(context: Context): List<DocumentDescriptor>? {
    // Retrieve the descriptors JSON from preferences. Return immediately if
    // it's not set — this method returns `null` when there is no state saved.
    val descriptorsJson = preferences.getString(PREF_DOCUMENT_DESCRIPTORS_JSON, null)
        ?: return null

    // Parse the JSON string.
    val descriptorsArray = JSONArray(descriptorsJson)

    // Iterate through all elements in the array and extract document descriptors out of it.
    val documentDescriptors = mutableListOf<DocumentDescriptor>()
    for (i in 0 until descriptorsArray.length()) {
        val descriptorJson = descriptorsArray[i] as JSONObject

        // Extract the document file Uri or skip the entry if it's not available.
        val uri = descriptorJson.getString(JSON_DESCRIPTOR_URI) ?: continue
        // Extract optional title if available, we don't store `null` values in our JSON.
        val title = if (descriptorJson.has(JSON_DESCRIPTOR_TITLE)) descriptorJson.getString(JSON_DESCRIPTOR_TITLE) else null

        // Parse the file Uri.
        val fileUri = Uri.parse(uri)
        val documentDescriptor = if (ImageDocumentUtils.isImageUri(context, fileUri)) {
            // Create the image document descriptor for image Uris.
            DocumentDescriptor.imageDocumentFromUri(fileUri)
        } else {
            // Create the standard document descriptor for other Uris.
            DocumentDescriptor.fromUri(fileUri)
        }

        // Set the custom title as stored in preferences.
        if (title != null) {
            documentDescriptor.setTitle(title)
        }

        documentDescriptors.add(documentDescriptor)
    }
    return documentDescriptors
}

The remaining two methods are pretty basic since we are only storing the integer value in the shared preferences:

fun setVisibleDocumentIndex(visibleDocumentIndex: Int) {
    preferences.edit().putInt(PREF_VISIBLE_DOCUMENT_INDEX, visibleDocumentIndex).apply()
}

fun getVisibleDocumentIndex(): Int {
    return preferences.getInt(PREF_VISIBLE_DOCUMENT_INDEX, 0)
}

State Management

Now we have our preferences data model in place, so let’s put it to use.

Saving the Tabs State

We create a new activity that extends PdfActivity in order to make it possible to add functionality to PSPDFKit’s default activity:

class PersistentTabsActivity : PdfActivity() {
}

We want to store the currently opened documents when leaving the activity. The recommended place to do this is in onStop(), which is executed when the activity goes to the background:

override fun onStop() {
    // Save the opened document descriptors and the currently visible document index to preferences.
    val tabsPreferences = TabsPreferences(this)

    // Retrieve the document descriptors for opened documents.
    val documents = documentCoordinator.documents

    // Save them to preferences.
    tabsPreferences.setDocumentDescriptors(documents)

    // Calculate the index of the visible document in opened documents.
    val visibleDocumentIndex = documents.indexOf(documentCoordinator.visibleDocument)
    tabsPreferences.setVisibleDocumentIndex(if (visibleDocumentIndex >= 0) visibleDocumentIndex else 0)

    // Don't forget to call `super.onStop()` to handle `onStop` correctly by the `PdfActivity`.
    super.onStop()
}

Restoring the Tabs State

To restore tabs, we’ll retrieve the saved document descriptors from the preferences in our app’s main activity. If there is no saved state, we’ll proceed with opening the PersistentTabsActivity with the default documents loaded:

val tabsPreferences = TabsPreferences(context)

// Retrieve the document descriptors saved in the shared preferences.
val restoredDocumentDescriptors = tabsPreferences.getDocumentDescriptors(context)

if (restoredDocumentDescriptors == null) {
    // No state is saved. Proceed with opening `PdfActivity` with the default documents.
    ...
} else {
    // Retrieve the visible document index.
    val visibleDocumentIndex = tabsPreferences.getVisibleDocumentIndex()
    launchActivityWithDocuments(context, documentDescriptors, visibleDocumentIndex)
}

If there is a saved state, we’ll build the activity’s intent from these document descriptors and launch it:

fun launchActivityWithDocuments(context: Context, documentDescriptors: List<DocumentDescriptor>, visibleDocumentIndex: Int) {
    val intentBuilder = if (documentDescriptors.isEmpty()) {
        // There is a separate factory method for creating an empty activity. This is
        a special state that displays an empty activity message instead of the document.
        PdfActivityIntentBuilder.emptyActivity(context)
    } else {
        // Passing the restored document descriptors here will result in opening these documents in tabs.
        PdfActivityIntentBuilder.fromDocumentDescriptor(context, *documentDescriptors.toTypedArray())
    }
    // Set the visible document we restored from preferences.
    intentBuilder.visibleDocument(visibleDocumentIndex)
        // You can also provide the required activity configuration here.
        .configuration(configuration.build())
        // Don't forget to set the activity class to our custom activity.
        .activityClass(PersistentTabsActivity::class.java)

    context.startActivity(intentBuilder.build())
}

Conclusion

In this blog post, I guided you through building your own persistent store for document tabs. If you want to learn more about PSPDFKit’s multi-document support, please refer to our guides. You can also check out the full example inside our Catalog — just search for the Persistent Tabs example.

Author
Tomáš Šurín
Tomáš Šurín Server and Services Engineer

Tomáš has a deep interest in building (and breaking) stuff both in the digital and physical world. In his spare time, you’ll find him relaxing off the grid, cooking good food, playing board games, and discussing science and philosophy.

Explore related topics

Free trial Ready to get started?
Free trial