TypedPreferences

Additional

Language
Kotlin
Version
v2.1.1 (Apr 16, 2018)
Created
Jul 18, 2017
Updated
Mar 13, 2023
Owner
Markus Ressel (markusressel)
Contributor
Markus Ressel (markusressel)
1
Activity
Badge
Generate
Download
Source code

TypedPreferences

A simple library to make the use of Android's SharedPreferences easier while keeping it type safe. For base types this library only uses generics to provide type safety. However it is also possible to use more complex classes as types. Those will be serialized to json using the GSON library (Link) before saving them to SharedPrefrences.

This library was designed to be used with a Dependency Injection Framework like Dagger 2 (Link) and Lombok (GitHub, Examples) for boilerplate code generation in mind. If you have never used one of those tools I highly recommend looking into them before you start building your app.

Build Status

Master Beta Dev

How to use

Have a look at the demo app (app module) for a complete sample. The sample app uses Dagger 2 to inject the PreferenceHandler into the activity and fragment. Using DI is the recommended way to use this library.

Gradle

To use this library just include it in your depencencies using

repositories {
    ...
    maven { url "https://jitpack.io" }
}

in your project build.gradle file and

dependencies {
    implementation('com.github.markusressel:TypedPreferences:v2.1.1') {
        exclude module: 'app'
        transitive = true
    }
}

in your desired module build.gradle file.

Create a PreferenceHandler

The first thing you have to do to get started is creating a class which extends the provided PreferencesHandlerBase class. Override the necessary methods like getSharedPreferencesName() to provide the name of your SharedPreferences file and initialize the allPreferenceItems property to return a set of all your PreferenceItems.

A simple example would look something like this:

@Singleton
class PreferenceHandler @Inject
constructor(context: Context) : PreferencesHandlerBase(context) {

    // be sure to override the get() method
    override var sharedPreferencesName: String? = null
        get() = "preferences"

    override val allPreferenceItems: Set<PreferenceItem<*>> = hashSetOf(
            THEME,
            BOOLEAN_SETTING,
            COMPLEX_SETTING
    )

    companion object {
        val THEME = PreferenceItem(R.string.key_theme, 0)
        val BOOLEAN_SETTING = PreferenceItem(R.string.key_boolean_setting, true)
        val COMPLEX_SETTING = PreferenceItem(R.string.key_complex_setting, ComplexClass("Complex ^", 10, listOf(1, 2, 3)))
    }

}

Attention:

Be sure to override the get() method of the sharedPreferencesName variable as seen above! The example below will not work and the PreferenceHandler will be initialized using null instead!

You can also purposefully set it to null if you want to use the default preferences.

// THIS WILL NOT WORK!
// because of the initialization order of java objects
// use the getter variant instead as seen above
override val sharedPreferencesName: String? = "preferences"

Define your PreferenceItems

To make accessing your preferences as easy as possible define them by declaring a PreferenceItem in the companion object of your PreferenceHandler or any other accessible place.

companion object {
    val THEME = PreferenceItem(R.string.key_theme, 0)
    val BOOLEAN_SETTING = PreferenceItem(R.string.key_boolean_setting, true)
    val COMPLEX_SETTING = PreferenceItem(R.string.key_complex_setting, ComplexClass("Complex ^", 10, listOf(1, 2, 3)))
}

Important to note here is that the key is not a String but a StringRes (Int) that you define in your strings.xml (or a similar resource file). This makes it possible to also use this value in a PreferenceFragment like shown in the example app. The generic type will be inferred from the default value, which therefore must not be null. Otherwise you'd have to specify the type manually which would reduce type safety and isn't possible since the kotlin port of this library (v2.0).

Since v1.1 the type of your PreferenceItem is not limited to base types anymore but can be any class extending Object. If needed your custom object will be serialized to json using the GSON library (Link) and then saved to the SharedPreferences as a String. Refer to the GSON User Guide for more info on what types are fully serializable by GSON.

Get a stored value

To retrieve a value use the getValue(preferenceItem: PreferenceItem<T>): T method of your PreferenceHandler:

val value = preferenceHandler.getValue(PreferenceHandler.BOOLEAN_SETTING)

If possible the result type will be detected automatically. If this doesn't work for some reason (f.ex. because you are accessing a generic PreferenceItem) you need to check it's type manually (if necessary). This will break type safety though and should only be used as a last resort.

Example:

val key = "boolean_preference"
val preferenceItem = preferenceHandler.getPreferenceItem(key)
preferenceItem?.let {
    val itemValue = preferenceHandler.getValue(it) as Boolean
}

This will break type safety though and should only be used as a last resort.

You may use any type you want, f.ex.:

// the type (ComplexClass) is automatically inferred
val complexClass = preferenceHandler.getValue(PreferenceHandler.COMPLEX_SETTING)

Keep in mind though that saving base types like Boolean, Int, Long, Float, String is always preferable to using complex classes that need serialization before they can be saved. Base types are saved without json serialization and can therefore be saved and requested much faster.

Set a new value

To set a new value use the setValue(preferenceItem: PreferenceItem<T>, newValue: T) method:

preferenceHandler.setValue(PreferenceHandler.BOOLEAN_SETTING, true);

The target type of your preference will be detected automatically from the default value of your PreferenceItem. If the type of newValue is not the expected one this line will show an error at compile time.

Another example for using a complex class type:

preferenceHandler.setValue(PreferenceHandler.COMPLEX_SETTING, ComplexClass("Test", 0, listOf()))

Listen for value changes

Just like with the default SharedPreferences you can add a listener to get notified of value changes. Note that this listener will only trigger if the the value for this PreferenceItem actually changed (oldValue != newValue).

To preserve type safety you can only add listeners for specific PreferenceItem's:

val booleanSettingListener = preferenceHandler.addOnPreferenceChangedListener(PreferenceHandler.BOOLEAN_SETTING) { 
    preference, old, new ->
        Timber.d { "Preference '${preference.getKey(appContext)}' changed from '$old' to '$new'" }
    }

To make it easier for you to remove this listener when you don't need it anymore the method will return the exact listener you passed in as an argument.

To remove a listener use one of these methods:

// remove a single listener
booleanSettingListener?.let {
    preferenceHandler.removeOnPreferenceChangedListener(it)
}

// remove all listeners of a specific preference
preferenceHandler.removeAllOnPreferenceChangedListeners(PreferenceHandler.BOOLEAN_SETTING)

// remove all listeners of the handler
preferenceHandler.removeAllOnPreferenceChangedListeners()

Check if a PreferenceHandler contains a specific PreferenceItem

If you implement multiple PreferenceHandler's it might be necessary to check if the PreferenceHandler you are trying to use contains the PreferenceItem you want to access. To do this you can use this method:

val hasPreference = preferenceHandler.hasPreference(PreferenceHandler.BOOLEAN_SETTING)

Troubleshooting

If you are using a custom class as the type of your PreferenceItem make sure it can be parsed by the GSON library. Refer to the GSON User Guide for more info on what types are fully serializable by GSON.

Contributing

Github is for social coding: if you want to write code, I encourage contributions through pull requests from forks of this repository. Create Github tickets for bugs and new features and comment on the ones that you are interested in.

License

Copyright (c) 2017 Markus Ressel

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.