💄 Optimize settings screen

This commit is contained in:
2025-12-19 22:49:50 +08:00
parent e820fb08de
commit ac4ceb3960

View File

@@ -3,6 +3,7 @@ import 'package:groovybox/providers/settings_provider.dart';
import 'package:groovybox/providers/watch_folder_provider.dart'; import 'package:groovybox/providers/watch_folder_provider.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:file_picker/file_picker.dart'; import 'package:file_picker/file_picker.dart';
import 'package:styled_widget/styled_widget.dart';
class SettingsScreen extends ConsumerWidget { class SettingsScreen extends ConsumerWidget {
const SettingsScreen({super.key}); const SettingsScreen({super.key});
@@ -15,152 +16,179 @@ class SettingsScreen extends ConsumerWidget {
return Scaffold( return Scaffold(
appBar: AppBar(title: const Text('Settings')), appBar: AppBar(title: const Text('Settings')),
body: settingsAsync.when( body: settingsAsync.when(
data: (settings) => SingleChildScrollView( data: (settings) => Align(
padding: const EdgeInsets.all(16.0), alignment: Alignment.topCenter,
child: Column( child: ConstrainedBox(
crossAxisAlignment: CrossAxisAlignment.start, constraints: const BoxConstraints(maxWidth: 640),
children: [ child: SingleChildScrollView(
// Auto Scan Section padding: const EdgeInsets.all(16.0),
Card( child: Column(
child: Padding( spacing: 16,
padding: const EdgeInsets.all(16.0), crossAxisAlignment: CrossAxisAlignment.start,
child: Column( children: [
crossAxisAlignment: CrossAxisAlignment.start, // Auto Scan Section
children: [ Card(
const Text( margin: EdgeInsets.zero,
'Auto Scan', child: Column(
style: TextStyle( crossAxisAlignment: CrossAxisAlignment.start,
fontSize: 18, children: [
fontWeight: FontWeight.bold, const Text(
'Auto Scan',
style: TextStyle(
fontSize: 18,
fontWeight: FontWeight.bold,
),
).padding(horizontal: 16, bottom: 8, top: 16),
SwitchListTile(
title: const Text('Auto-scan music libraries'),
subtitle: const Text(
'Automatically scan music libraries for new music files',
),
shape: const RoundedRectangleBorder(
borderRadius: BorderRadius.all(Radius.circular(8)),
),
value: settings.autoScan,
onChanged: (value) {
ref.read(autoScanProvider.notifier).update(value);
},
), ),
), SwitchListTile(
const SizedBox(height: 8), title: const Text('Watch for changes'),
SwitchListTile( subtitle: const Text(
title: const Text('Auto-scan music libraries'), 'Monitor music libraries for file changes',
subtitle: const Text( ),
'Automatically scan music libraries for new music files', shape: const RoundedRectangleBorder(
borderRadius: BorderRadius.all(Radius.circular(8)),
),
value: settings.watchForChanges,
onChanged: (value) {
ref
.read(watchForChangesProvider.notifier)
.update(value);
},
), ),
value: settings.autoScan, const SizedBox(height: 8),
onChanged: (value) { ],
ref.read(autoScanProvider.notifier).update(value); ),
},
),
SwitchListTile(
title: const Text('Watch for changes'),
subtitle: const Text(
'Monitor music libraries for file changes',
),
value: settings.watchForChanges,
onChanged: (value) {
ref
.read(watchForChangesProvider.notifier)
.update(value);
},
),
],
), ),
),
),
const SizedBox(height: 16), // Watch Folders Section
Card(
// Watch Folders Section margin: EdgeInsets.zero,
Card( child: Column(
child: Padding( crossAxisAlignment: CrossAxisAlignment.start,
padding: const EdgeInsets.all(16.0), children: [
child: Column( Column(
crossAxisAlignment: CrossAxisAlignment.start, children: [
children: [ Row(
Row( mainAxisAlignment: MainAxisAlignment.spaceBetween,
mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [
children: [ const Text(
const Text( 'Music Libraries',
'Music Libraries', style: TextStyle(
style: TextStyle( fontSize: 18,
fontSize: 18, fontWeight: FontWeight.bold,
fontWeight: FontWeight.bold, ),
),
),
Row(
children: [
IconButton(
onPressed: () => _scanLibraries(context, ref),
icon: const Icon(Icons.refresh),
tooltip: 'Scan Libraries',
),
IconButton(
onPressed: () => _addMusicLibrary(context, ref),
icon: const Icon(Icons.add),
tooltip: 'Add Music Library',
),
],
),
],
),
const SizedBox(height: 8),
const Text(
'Add folder libraries to index music files. Files will be copied to internal storage for playback.',
style: TextStyle(color: Colors.grey, fontSize: 14),
),
const SizedBox(height: 8),
watchFoldersAsync.when(
data: (folders) => folders.isEmpty
? const Text(
'No music libraries added yet.',
style: TextStyle(
color: Colors.grey,
fontSize: 14,
), ),
) Row(
: Column( children: [
children: folders IconButton(
.map( onPressed: () =>
(folder) => ListTile( _scanLibraries(context, ref),
title: Text(folder.name), icon: const Icon(Icons.refresh),
subtitle: Text(folder.path), tooltip: 'Scan Libraries',
trailing: Row( visualDensity: const VisualDensity(
mainAxisSize: MainAxisSize.min, horizontal: -4,
children: [ vertical: -4,
Switch(
value: folder.isActive,
onChanged: (value) {
ref
.read(
watchFolderServiceProvider,
)
.toggleWatchFolder(
folder.id,
value,
);
},
),
IconButton(
icon: const Icon(Icons.delete),
onPressed: () {
ref
.read(
watchFolderServiceProvider,
)
.removeWatchFolder(
folder.id,
);
},
),
],
),
), ),
) ),
.toList(), IconButton(
onPressed: () =>
_addMusicLibrary(context, ref),
icon: const Icon(Icons.add),
tooltip: 'Add Music Library',
visualDensity: const VisualDensity(
horizontal: -4,
vertical: -4,
),
),
],
),
],
),
const Text(
'Add folder libraries to index music files. Files will be copied to internal storage for playback.',
style: TextStyle(
color: Colors.grey,
fontSize: 14,
), ),
loading: () => const CircularProgressIndicator(), ),
error: (error, _) => ],
Text('Error loading libraries: $error'), ).padding(horizontal: 16, top: 16, bottom: 8),
), watchFoldersAsync.when(
], data: (folders) => folders.isEmpty
? const Text(
'No music libraries added yet.',
style: TextStyle(
color: Colors.grey,
fontSize: 14,
),
).padding(horizontal: 16, vertical: 8)
: Column(
children: folders
.map(
(folder) => ListTile(
title: Text(folder.name),
subtitle: Text(folder.path),
contentPadding: const EdgeInsets.only(
left: 16,
right: 16,
),
trailing: Row(
mainAxisSize: MainAxisSize.min,
children: [
Switch(
value: folder.isActive,
onChanged: (value) {
ref
.read(
watchFolderServiceProvider,
)
.toggleWatchFolder(
folder.id,
value,
);
},
),
IconButton(
icon: const Icon(Icons.delete),
onPressed: () {
ref
.read(
watchFolderServiceProvider,
)
.removeWatchFolder(
folder.id,
);
},
),
],
),
),
)
.toList(),
),
loading: () => const CircularProgressIndicator(),
error: (error, _) =>
Text('Error loading libraries: $error'),
),
const SizedBox(height: 8),
],
),
), ),
), ],
), ),
], ),
), ),
), ),
loading: () => const Center(child: CircularProgressIndicator()), loading: () => const Center(child: CircularProgressIndicator()),