switched to json storage to allow sharing

This commit is contained in:
tom.hempel
2025-10-15 12:20:27 +02:00
parent c15d1d1e49
commit 67dd2de460
7 changed files with 157 additions and 26 deletions

View File

@ -90,6 +90,53 @@ class _HomeScreenState extends State<HomeScreen> {
);
}
Future<void> _showStorageInfo() async {
final filePath = await _storageService.getProjectsFilePath();
if (!mounted) return;
showDialog(
context: context,
builder: (context) => AlertDialog(
title: const Row(
children: [
Icon(Icons.folder_open),
SizedBox(width: 8),
Text('Storage Location'),
],
),
content: Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: [
const Text(
'Your projects are saved in:',
style: TextStyle(fontWeight: FontWeight.bold),
),
const SizedBox(height: 8),
SelectableText(
filePath,
style: const TextStyle(
fontFamily: 'monospace',
fontSize: 12,
),
),
const SizedBox(height: 16),
const Text(
'You can share this file with other users or back it up for safekeeping.',
style: TextStyle(fontSize: 12),
),
],
),
actions: [
TextButton(
onPressed: () => Navigator.pop(context),
child: const Text('Close'),
),
],
),
);
}
void _showWebWarning() {
showDialog(
context: context,
@ -137,6 +184,12 @@ class _HomeScreenState extends State<HomeScreen> {
tooltip: 'Web Platform Limitations',
),
),
if (!kIsWeb)
IconButton(
icon: const Icon(Icons.folder),
onPressed: _showStorageInfo,
tooltip: 'Storage Location',
),
IconButton(
icon: const Icon(Icons.info_outline),
onPressed: _showAboutDialog,

View File

@ -1,22 +1,27 @@
import 'dart:convert';
import 'package:shared_preferences/shared_preferences.dart';
import 'dart:io';
import 'package:path_provider/path_provider.dart';
import '../models/project.dart';
class StorageService {
static const String _projectsKey = 'projects';
static const String _currentPortKey = 'current_port';
static const String _projectsFileName = 'unityudp_projects.json';
Future<String> get _localPath async {
final directory = await getApplicationDocumentsDirectory();
return directory.path;
}
Future<File> get _projectsFile async {
final path = await _localPath;
return File('$path/$_projectsFileName');
}
Future<void> saveProjects(List<Project> projects) async {
try {
final prefs = await SharedPreferences.getInstance();
final file = await _projectsFile;
final jsonList = projects.map((p) => p.toJson()).toList();
final jsonString = json.encode(jsonList);
final success = await prefs.setString(_projectsKey, jsonString);
if (!success) {
throw Exception('Failed to save projects to storage');
}
// Force a commit on web
await prefs.reload();
final jsonString = const JsonEncoder.withIndent(' ').convert(jsonList);
await file.writeAsString(jsonString);
} catch (e) {
rethrow;
}
@ -24,31 +29,60 @@ class StorageService {
Future<List<Project>> loadProjects() async {
try {
final prefs = await SharedPreferences.getInstance();
// Reload to ensure we get the latest data
await prefs.reload();
final jsonString = prefs.getString(_projectsKey);
final file = await _projectsFile;
if (jsonString == null || jsonString.isEmpty) {
// Check if file exists
if (!await file.exists()) {
return [];
}
final List<dynamic> jsonList = json.decode(jsonString);
final contents = await file.readAsString();
if (contents.isEmpty) {
return [];
}
final List<dynamic> jsonList = json.decode(contents);
return jsonList.map((json) => Project.fromJson(json)).toList();
} catch (e) {
return [];
}
}
Future<void> saveCurrentPort(int port) async {
final prefs = await SharedPreferences.getInstance();
await prefs.setInt(_currentPortKey, port);
Future<String> getProjectsFilePath() async {
final file = await _projectsFile;
return file.path;
}
Future<int> loadCurrentPort() async {
final prefs = await SharedPreferences.getInstance();
return prefs.getInt(_currentPortKey) ?? 8888;
Future<bool> importProjectsFromFile(String filePath) async {
try {
final importFile = File(filePath);
if (!await importFile.exists()) {
return false;
}
final contents = await importFile.readAsString();
final List<dynamic> jsonList = json.decode(contents);
final projects = jsonList.map((json) => Project.fromJson(json)).toList();
// Save the imported projects
await saveProjects(projects);
return true;
} catch (e) {
return false;
}
}
Future<bool> exportProjectsToFile(String filePath, List<Project> projects) async {
try {
final exportFile = File(filePath);
final jsonList = projects.map((p) => p.toJson()).toList();
final jsonString = const JsonEncoder.withIndent(' ').convert(jsonList);
await exportFile.writeAsString(jsonString);
return true;
} catch (e) {
return false;
}
}
}

View File

@ -35,7 +35,7 @@ class AppAboutDialog extends StatelessWidget {
const Text('• Configure custom UDP ports'),
const Text('• Store pre-defined packages'),
const Text('• Quick send functionality'),
const Text('Persistent local storage'),
const Text('JSON file storage (easily shareable)'),
const SizedBox(height: 16),
const Divider(),
const SizedBox(height: 16),