Replace SharedPreferences by Gson

Replace all SharedPreferences use by gson
This commit is contained in:
Gabriel 2023-12-22 01:05:08 -04:00
parent b2c653ed1b
commit 56398acdc9
6 changed files with 189 additions and 78 deletions

View file

@ -0,0 +1,37 @@
package com.panda3ds.pandroid.data;
import com.google.gson.Gson;
import com.panda3ds.pandroid.lang.Task;
import com.panda3ds.pandroid.utils.FileUtils;
public class GsonConfigParser {
private final Gson gson = new Gson();
private final String name;
public GsonConfigParser(String name){
this.name = name;
}
private String getPath(){
return FileUtils.getConfigPath()+"/"+name+".json";
}
public void save(Object data){
synchronized (this) {
new Task(() -> {
String json = gson.toJson(data);
FileUtils.writeTextFile(FileUtils.getConfigPath(), name + ".json", json);
}).runSync();
}
}
public <T> T load(Class<T> clazz){
String[] content = new String[]{"{}"};
new Task(()->{
if (FileUtils.exists(getPath())){
content[0] = FileUtils.readTextFile(getPath());
}
}).runSync();
return gson.fromJson(content[0], clazz);
}
}

View file

@ -1,58 +1,56 @@
package com.panda3ds.pandroid.data.config;
import android.content.Context;
import android.content.SharedPreferences;
import com.panda3ds.pandroid.app.PandroidApplication;
import com.panda3ds.pandroid.data.GsonConfigParser;
import com.panda3ds.pandroid.utils.Constants;
import java.io.Serializable;
import java.util.HashMap;
public class GlobalConfig {
private static SharedPreferences data;
private static final GsonConfigParser parser = new GsonConfigParser(Constants.PREF_GLOBAL_CONFIG);
public static final int VALUE_THEME_ANDROID = 0;
public static final int VALUE_THEME_LIGHT = 1;
public static final int VALUE_THEME_DARK = 2;
public static final int VALUE_THEME_BLACK = 3;
public static DataModel data;
public static final Key<Integer> KEY_APP_THEME = new Key<>("app.theme", VALUE_THEME_ANDROID);
public static void initialize() {
data = PandroidApplication.getAppContext()
.getSharedPreferences(Constants.PREF_GLOBAL_CONFIG, Context.MODE_PRIVATE);
data = parser.load(DataModel.class);
}
public static <T extends Serializable> T get(Key<T> key) {
Serializable value;
if (!data.configs.containsKey(key.name)){
return key.defaultValue;
}
if (key.defaultValue instanceof String) {
value = data.getString(key.name, (String) key.defaultValue);
value = (String) data.configs.get(key.name);
} else if (key.defaultValue instanceof Integer) {
value = data.getInt(key.name, (int) key.defaultValue);
value = ((Number) data.get(key.name)).intValue();
} else if (key.defaultValue instanceof Boolean) {
value = data.getBoolean(key.name, (boolean) key.defaultValue);
value = (boolean) data.get(key.name);
} else if (key.defaultValue instanceof Long) {
value = data.getLong(key.name, (long) key.defaultValue);
value = ((Number) data.get(key.name)).longValue();
} else {
value = data.getFloat(key.name, (float) key.defaultValue);
value = ((Number) data.get(key.name)).floatValue();
}
return (T) value;
}
public static synchronized <T extends Serializable> void set(Key<T> key, T value) {
if (value instanceof String) {
data.edit().putString(key.name, (String) value).apply();
} else if (value instanceof Integer) {
data.edit().putInt(key.name, (int) value).apply();
} else if (value instanceof Boolean) {
data.edit().putBoolean(key.name, (boolean) value).apply();
} else if (value instanceof Long) {
data.edit().putLong(key.name, (long) value).apply();
} else if (value instanceof Float) {
data.edit().putFloat(key.name, (float) value).apply();
} else {
throw new IllegalArgumentException("Invalid global config value instance");
}
data.configs.put(key.name, value);
writeChanges();
}
private static void writeChanges(){
parser.save(data);
}
private static class Key<T extends Serializable> {
@ -64,4 +62,12 @@ public class GlobalConfig {
this.defaultValue = defaultValue;
}
}
private static class DataModel {
private final HashMap<String, Object> configs = new HashMap<>();
public Object get(String key){
return configs.get(key);
}
}
}

View file

@ -1,30 +1,27 @@
package com.panda3ds.pandroid.input;
import android.content.Context;
import android.content.SharedPreferences;
import com.panda3ds.pandroid.app.PandroidApplication;
import com.panda3ds.pandroid.data.GsonConfigParser;
import com.panda3ds.pandroid.utils.Constants;
public class InputMap {
private static SharedPreferences data;
private static final String KEY_DEAD_ZONE = "deadZone";
public static final GsonConfigParser parser = new GsonConfigParser(Constants.PREF_INPUT_MAP);
private static DataModel data;
public static void initialize() {
data = PandroidApplication.getAppContext().getSharedPreferences(Constants.PREF_INPUT_MAP, Context.MODE_PRIVATE);
data = parser.load(DataModel.class);
}
public static float getDeadZone() {
return data.getFloat(KEY_DEAD_ZONE, 0.2f);
return data.deadZone;
}
public static void set(KeyName key, String name) {
data.edit().putString(key.name(), name).apply();
data.keys[key.ordinal()] = name;
writeConfig();
}
public static String relative(KeyName key) {
return data.getString(key.name(), "-");
return data.keys[key.ordinal()] == null ? "-" : data.keys[key.ordinal()];
}
public static KeyName relative(String name) {
@ -36,7 +33,16 @@ public class InputMap {
}
public static void setDeadZone(float value) {
data.edit().putFloat(KEY_DEAD_ZONE, Math.max(0.0f, Math.min(1.0f, value))).apply();
data.deadZone = Math.max(0.0f, Math.min(1.0f, value));
writeConfig();
}
private static void writeConfig() {
parser.save(data);
}
private static class DataModel {
public float deadZone = 0.2f;
public final String[] keys = new String[32];
}
}

View file

@ -0,0 +1,20 @@
package com.panda3ds.pandroid.lang;
public class Task extends Thread {
public Task(Runnable runnable){
super(runnable);
}
public void runSync(){
start();
waitFinish();
}
public void waitFinish(){
try {
join();
} catch (InterruptedException e) {
throw new RuntimeException(e);
}
}
}

View file

@ -3,20 +3,24 @@ package com.panda3ds.pandroid.utils;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.util.Log;
import androidx.documentfile.provider.DocumentFile;
import com.panda3ds.pandroid.app.PandroidApplication;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
public class FileUtils {
public static final String MODE_READ = "r";
private static DocumentFile parseFile(String path){
if (path.startsWith("/")){
private static DocumentFile parseFile(String path) {
if (path.startsWith("/")) {
return DocumentFile.fromFile(new File(path));
}
Uri uri = Uri.parse(path);
@ -31,32 +35,81 @@ public class FileUtils {
return parseFile(path).getName();
}
public static String getPrivatePath(){
public static String getPrivatePath() {
File file = getContext().getFilesDir();
if (!file.exists()){
if (!file.exists()) {
file.mkdirs();
}
return file.getAbsolutePath();
}
public static boolean exists(String path){
public static String getConfigPath() {
File file = new File(getPrivatePath(), "config");
if (!file.exists()) {
file.mkdirs();
}
return file.getAbsolutePath();
}
public static boolean exists(String path) {
return parseFile(path).exists();
}
public static boolean createDir(String path, String name){
public static boolean createDir(String path, String name) {
DocumentFile folder = parseFile(path);
if (folder.findFile(name) != null)
return true;
return folder.createDirectory(name) != null;
}
public static boolean createFile(String path, String name){
public static boolean createFile(String path, String name) {
DocumentFile folder = parseFile(path);
if (folder.findFile(name) != null)
return true;
return folder.createFile("", name) != null;
}
public static boolean writeTextFile(String path, String name, String content) {
try {
if (!exists(path + "/" + name)) {
createFile(path, name);
}
OutputStream stream = getOutputStream(path + "/" + name);
stream.write(content.getBytes(StandardCharsets.UTF_8));
stream.flush();
stream.close();
} catch (Exception e) {
Log.e(Constants.LOG_TAG, "Error on write text file: ", e);
return false;
}
return true;
}
public static String readTextFile(String path) {
if (!exists(path))
return null;
try {
InputStream stream = getInputStream(path);
ByteArrayOutputStream output = new ByteArrayOutputStream();
int len;
byte[] buffer = new byte[1024 * 8];
while ((len = stream.read(buffer)) != -1)
output.write(buffer, 0, len);
stream.close();
output.flush();
output.close();
byte[] data = output.toByteArray();
return new String(data, 0, data.length);
} catch (Exception e) {
return null;
}
}
public static InputStream getInputStream(String path) throws FileNotFoundException {
return getContext().getContentResolver().openInputStream(parseFile(path).getUri());
}
@ -66,7 +119,6 @@ public class FileUtils {
}
public static void makeUriPermanent(String uri, String mode) {
int flags = Intent.FLAG_GRANT_READ_URI_PERMISSION;
if (mode.toLowerCase().contains("w"))
flags &= Intent.FLAG_GRANT_WRITE_URI_PERMISSION;

View file

@ -2,46 +2,34 @@ package com.panda3ds.pandroid.utils;
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.util.Log;
import com.google.gson.Gson;
import com.panda3ds.pandroid.app.GameActivity;
import com.panda3ds.pandroid.app.PandroidApplication;
import com.panda3ds.pandroid.data.GsonConfigParser;
import com.panda3ds.pandroid.data.game.GameMetadata;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Objects;
public class GameUtils {
private static final Bitmap DEFAULT_ICON = Bitmap.createBitmap(48,48, Bitmap.Config.ARGB_8888);
private static final String KEY_GAME_LIST = "gameList";
private static final ArrayList<GameMetadata> games = new ArrayList<>();
private static SharedPreferences data;
private static final Gson gson = new Gson();
private static final Bitmap DEFAULT_ICON = Bitmap.createBitmap(48, 48, Bitmap.Config.ARGB_8888);
public static GsonConfigParser parser = new GsonConfigParser(Constants.PREF_GAME_UTILS);
private static DataModel data;
private static GameMetadata currentGame;
public static void initialize() {
data = PandroidApplication.getAppContext().getSharedPreferences(Constants.PREF_GAME_UTILS, Context.MODE_PRIVATE);
GameMetadata[] list = gson.fromJson(data.getString(KEY_GAME_LIST, "[]"), GameMetadata[].class);
for (GameMetadata game: list)
game.getIcon();
games.clear();
games.addAll(Arrays.asList(list));
data = parser.load(DataModel.class);
}
public static GameMetadata findByRomPath(String romPath) {
for (GameMetadata game : games) {
for (GameMetadata game : data.games) {
if (Objects.equals(romPath, game.getRomPath())) {
return game;
}
@ -60,51 +48,53 @@ public class GameUtils {
}
public static void removeGame(GameMetadata game) {
games.remove(game);
saveAll();
data.games.remove(game);
writeChanges();
}
public static void addGame(GameMetadata game) {
games.add(0,game);
saveAll();
data.games.add(0, game);
writeChanges();
}
public static ArrayList<GameMetadata> getGames() {
return new ArrayList<>(games);
return new ArrayList<>(data.games);
}
private static synchronized void saveAll() {
data.edit()
.putString(KEY_GAME_LIST, gson.toJson(games.toArray(new GameMetadata[0])))
.apply();
private static void writeChanges() {
parser.save(data);
}
public static void setGameIcon(String id, Bitmap icon) {
try {
String appPath = FileUtils.getPrivatePath();
FileUtils.createDir(appPath, "cache_icons");
FileUtils.createFile(appPath+"/cache_icons/", id+".png");
FileUtils.createFile(appPath + "/cache_icons/", id + ".png");
OutputStream output = FileUtils.getOutputStream(appPath+"/cache_icons/"+id+".png");
OutputStream output = FileUtils.getOutputStream(appPath + "/cache_icons/" + id + ".png");
icon.compress(Bitmap.CompressFormat.PNG, 100, output);
output.close();
} catch (Exception e){
} catch (Exception e) {
Log.e(Constants.LOG_TAG, "Error on save game icon: ", e);
}
}
public static Bitmap loadGameIcon(String id) {
try {
String path = FileUtils.getPrivatePath()+"/cache_icons/"+id+".png";
String path = FileUtils.getPrivatePath() + "/cache_icons/" + id + ".png";
if (FileUtils.exists(path)) {
InputStream stream = FileUtils.getInputStream(path);
Bitmap image = BitmapFactory.decodeStream(stream);
stream.close();
return image;
}
} catch (Exception e){
} catch (Exception e) {
Log.e(Constants.LOG_TAG, "Error on load game icon: ", e);
}
return DEFAULT_ICON;
}
private static class DataModel {
public final ArrayList<GameMetadata> games = new ArrayList<>();
}
}