switched to json storage to allow sharing
This commit is contained in:
@ -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,
|
||||
|
||||
@ -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;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
@ -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),
|
||||
|
||||
Reference in New Issue
Block a user