I’ve been learning Flutter and learned this to improve usage with Pub.dev package Shared Preferences,
To sum up, it is to create a service that I can consume and work in a better way all around my app with SharedPreferences, it works perfectly on my Code for token
because I have my async function
So, the getMethod works perfectly using final token = await keyValueStorageService.getValue<String>('token')
(also async function above),
but I can’t manage to use this code on my UI code, it gives me this instead (Because of the lack of an await function)
but as I read you can use it and initiate (shown below) on your main.dart, I’ve been trying to use a method getSharedPrefs()
on the main.dart
to implement and use it all around my app for example calling my getValue('user')
wherever I need to but haven’t made it work
void main() async {
WidgetsFlutterBinding.ensureInitialized();
await Enviroment.initEnvirontment();
//! Key Value instance for main async
await KeyValueStorageServiceImpl().getSharedPrefs();
await initializeDateFormatting('es_ES', null);
SystemChrome.setPreferredOrientations([DeviceOrientation.portraitUp])
.then((value) => runApp(ProviderScope(child: MyApp())));
}
My actual code is this
key_value_storage.dart
import 'key_value_storage.dart';
import 'package:shared_preferences/shared_preferences.dart';
class KeyValueStorageServiceImpl extends KeyValueStorageService {
Future<SharedPreferences> getSharedPrefs() async {
return await SharedPreferences.getInstance();
}
@override
Future<T?> getValue<T>(String key) async {
final prefs = await getSharedPrefs();
switch (T) {
case String:
return prefs.getString(key) as T?;
case int:
return prefs.getInt(key) as T?;
case double:
return prefs.getDouble(key) as T?;
case bool:
return prefs.getBool(key) as T?;
case List:
return prefs.getStringList(key) as T?;
default:
throw UnimplementedError(
'No implementation Get for ${T.runtimeType}');
}
}
@override
Future<bool> removeKey(String key) async {
final prefs = await getSharedPrefs();
return await prefs.remove(key);
}
@override
Future<void> setKeyValue<T>(String key, T value) async {
final prefs = await getSharedPrefs();
switch (T) {
case String:
await prefs.setString(key, value as String);
break;
case int:
await prefs.setInt(key, value as int);
break;
case double:
await prefs.setDouble(key, value as double);
break;
case bool:
await prefs.setBool(key, value as bool);
break;
case List:
await prefs.setStringList(key, value as List<String>);
break;
default:
throw UnimplementedError(
'No implementation Set for ${T.runtimeType}');
}
}
}
key_value_class.dart
abstract class KeyValueStorageService {
Future<void> setKeyValue<T>(String key, T value);
Future<T?> getValue<T>(String key);
Future<bool> removeKey(String key);
}
What & why I made the changes:
-
Store an instance of
SharedPreferences
in theKeyValueStorageServiceImpl
class directly and initialize with the constructor. This makes sure that the once themain
function initializesSharedPreferences
once, there is no subsequent need of callinggetInstance
again inside thegetValue
function. -
The
get
functions ofSharedPreferences
return the T? value directly and not Future<T?> value, hence there is no need for thegetValue
function to returnFuture<T?>
, so I changed it’s definition. -
Changed the function signature in
key_value_class.dart
-
Changed the main function to pass a
SharedPreferences
object to the constructor ofKeyValueStorageServiceImpl
. -
Gave an alternative approach by making the
KeyValueStorageServiceImpl
into a Singleton.
You can read my comments in the code if you like.
key_value_storage.dart
import 'key_value_storage.dart';
import 'package:shared_preferences/shared_preferences.dart';
class KeyValueStorageServiceImpl extends KeyValueStorageService {
/// The instance of shared preferences is declared once for every object you create of this type.
final SharedPreferences prefs;
/// Call the `SharedPreferences.getInstance();` function outside of this class in order to not need to return a
/// future from the `getValue` method.
KeyValueStorageServiceImpl(this.prefs);
/// The `get` operation returns the value itself, not a future.
@override
T? getValue<T>(String key) {
switch (T) {
case String:
return prefs.getString(key) as T?;
case int:
return prefs.getInt(key) as T?;
case double:
return prefs.getDouble(key) as T?;
case bool:
return prefs.getBool(key) as T?;
case List:
return prefs.getStringList(key) as T?;
default:
throw UnimplementedError(
'No implementation Get for ${T.runtimeType}');
}
}
@override
Future<bool> removeKey(String key) async {
return await prefs.remove(key);
}
@override
Future<void> setKeyValue<T>(String key, T value) async {
switch (T) {
case String:
await prefs.setString(key, value as String);
break;
case int:
await prefs.setInt(key, value as int);
break;
case double:
await prefs.setDouble(key, value as double);
break;
case bool:
await prefs.setBool(key, value as bool);
break;
case List:
await prefs.setStringList(key, value as List<String>);
break;
default:
throw UnimplementedError(
'No implementation Set for ${T.runtimeType}');
}
}
}
key_value_class.dart
abstract class KeyValueStorageService {
Future<void> setKeyValue<T>(String key, T value);
T? getValue<T>(String key);
Future<bool> removeKey(String key);
}
main()
function
void main() async {
WidgetsFlutterBinding.ensureInitialized();
await Enviroment.initEnvirontment();
//! Key Value instance for main async
var prefs = await SharedPreferences.getInstance();
// use this wherever you need to call `getValue`
var keyValueStorageService = KeyValueStorageServiceImpl(prefs);
await initializeDateFormatting('es_ES', null);
SystemChrome.setPreferredOrientations([DeviceOrientation.portraitUp])
.then((value) => runApp(ProviderScope(child: MyApp())));
}
Alternative Approach using Singleton
key_value_storage.dart
import 'key_value_storage.dart';
import 'package:shared_preferences/shared_preferences.dart';
class KeyValueStorageServiceImpl extends KeyValueStorageService {
/// The instance of shared preferences
final SharedPreferences prefs;
/// The single instance which will remain all through the app lifecycle
static KeyValueStorageServiceImpl? _instance;
/// Getter for the instance
static KeyValueStorageServiceImpl get instance => _instance!;
/// Set the instance value, if not already set, by creating an object
/// of `KeyValueStorageServiceImpl` type and passing the SharedPreferences.
static void setSharedPreferences(SharedPreferences prefs) {
_instance ??= KeyValueStorageServiceImpl._(prefs);
}
/// Private Constructor to prevent multiple instances of this class.
KeyValueStorageServiceImpl._(this.prefs);
/// The `get` operation returns the value itself, not a future.
@override
T? getValue<T>(String key) {
switch (T) {
case String:
return prefs.getString(key) as T?;
case int:
return prefs.getInt(key) as T?;
case double:
return prefs.getDouble(key) as T?;
case bool:
return prefs.getBool(key) as T?;
case List:
return prefs.getStringList(key) as T?;
default:
throw UnimplementedError(
'No implementation Get for ${T.runtimeType}');
}
}
@override
Future<bool> removeKey(String key) async {
return await prefs.remove(key);
}
@override
Future<void> setKeyValue<T>(String key, T value) async {
switch (T) {
case String:
await prefs.setString(key, value as String);
break;
case int:
await prefs.setInt(key, value as int);
break;
case double:
await prefs.setDouble(key, value as double);
break;
case bool:
await prefs.setBool(key, value as bool);
break;
case List:
await prefs.setStringList(key, value as List<String>);
break;
default:
throw UnimplementedError(
'No implementation Set for ${T.runtimeType}');
}
}
}
main()
function
void main() async {
WidgetsFlutterBinding.ensureInitialized();
await Enviroment.initEnvirontment();
//! Key Value instance for main async
var prefs = await SharedPreferences.getInstance();
KeyValueStorageServiceImpl.setSharedPreferences(prefs);
await initializeDateFormatting('es_ES', null);
SystemChrome.setPreferredOrientations([DeviceOrientation.portraitUp])
.then((value) => runApp(ProviderScope(child: MyApp())));
}
Now wherever you need to get a value, you can just do the following
var value = KeyValueStorageServiceImpl.instance.getValue<Type>('key');
Hope this solves your issue! Happy Coding! 🙂
Can you add relevant UI code which leads to this problem? And you could initiate that call to the
getValue
function insideinitState
function of aStatefulWidget
.Hello, I didn’t add any UI code because what I need is to use it as I please, I mean as Text on a Container, as a Title for a Drawer, etc. Not in a specific widget.
Ohh okay got it now. Check my answer please. thank you!