mirror of
				https://github.com/johrpan/musicus_mobile.git
				synced 2025-10-26 10:47:25 +01:00 
			
		
		
		
	Move reusable code from mobile to common
This will be useful for a future desktop application.
This commit is contained in:
		
							parent
							
								
									6e1255f26e
								
							
						
					
					
						commit
						711b19c998
					
				
					 40 changed files with 813 additions and 581 deletions
				
			
		|  | @ -1,15 +1,18 @@ | |||
| import 'dart:async'; | ||||
| 
 | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| 
 | ||||
| import 'backend.dart'; | ||||
| import 'screens/home.dart'; | ||||
| import 'widgets/player_bar.dart'; | ||||
| 
 | ||||
| class App extends StatelessWidget { | ||||
|   static const _platform = MethodChannel('de.johrpan.musicus/platform'); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     final backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     return MaterialApp( | ||||
|       title: 'Musicus', | ||||
|  | @ -36,11 +39,11 @@ class App extends StatelessWidget { | |||
|       ), | ||||
|       home: Builder( | ||||
|         builder: (context) { | ||||
|           if (backend.status == BackendStatus.loading) { | ||||
|           if (backend.status == MusicusBackendStatus.loading) { | ||||
|             return Material( | ||||
|               color: Theme.of(context).scaffoldBackgroundColor, | ||||
|             ); | ||||
|           } else if (backend.status == BackendStatus.setup) { | ||||
|           } else if (backend.status == MusicusBackendStatus.setup) { | ||||
|             return Material( | ||||
|               color: Theme.of(context).scaffoldBackgroundColor, | ||||
|               child: Column( | ||||
|  | @ -57,8 +60,13 @@ class App extends StatelessWidget { | |||
|                   ListTile( | ||||
|                     leading: const Icon(Icons.folder_open), | ||||
|                     title: Text('Choose path'), | ||||
|                     onTap: () { | ||||
|                       backend.settings.chooseMusicLibraryUri(); | ||||
|                     onTap: () async { | ||||
|                       final uri = | ||||
|                           await _platform.invokeMethod<String>('openTree'); | ||||
| 
 | ||||
|                       if (uri != null) { | ||||
|                         backend.settings.setMusicLibraryPath(uri); | ||||
|                       } | ||||
|                     }, | ||||
|                   ), | ||||
|                 ], | ||||
|  | @ -82,7 +90,7 @@ class _ContentState extends State<Content> with SingleTickerProviderStateMixin { | |||
|   final nestedNavigator = GlobalKey<NavigatorState>(); | ||||
| 
 | ||||
|   AnimationController playerBarAnimation; | ||||
|   BackendState backend; | ||||
|   MusicusBackendState backend; | ||||
|   StreamSubscription<bool> playerActiveSubscription; | ||||
| 
 | ||||
|   @override | ||||
|  | @ -99,14 +107,14 @@ class _ContentState extends State<Content> with SingleTickerProviderStateMixin { | |||
|   void didChangeDependencies() { | ||||
|     super.didChangeDependencies(); | ||||
| 
 | ||||
|     backend = Backend.of(context); | ||||
|     playerBarAnimation.value = backend.player.active.value ? 1.0 : 0.0; | ||||
|     backend = MusicusBackend.of(context); | ||||
|     playerBarAnimation.value = backend.playback.active.value ? 1.0 : 0.0; | ||||
| 
 | ||||
|     if (playerActiveSubscription != null) { | ||||
|       playerActiveSubscription.cancel(); | ||||
|     } | ||||
| 
 | ||||
|     playerActiveSubscription = backend.player.active.listen((active) => | ||||
|     playerActiveSubscription = backend.playback.active.listen((active) => | ||||
|         active ? playerBarAnimation.forward() : playerBarAnimation.reverse()); | ||||
|   } | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,176 +0,0 @@ | |||
| import 'dart:io'; | ||||
| import 'dart:isolate'; | ||||
| import 'dart:ui'; | ||||
| 
 | ||||
| import 'package:flutter/widgets.dart'; | ||||
| import 'package:moor/isolate.dart'; | ||||
| import 'package:moor/moor.dart'; | ||||
| import 'package:moor_ffi/moor_ffi.dart'; | ||||
| import 'package:musicus_client/musicus_client.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| import 'package:path/path.dart' as p; | ||||
| import 'package:path_provider/path_provider.dart' as pp; | ||||
| 
 | ||||
| import 'music_library.dart'; | ||||
| import 'player.dart'; | ||||
| import 'settings.dart'; | ||||
| 
 | ||||
| // The following code was taken from | ||||
| // https://moor.simonbinder.eu/docs/advanced-features/isolates/ and just | ||||
| // slightly modified. | ||||
| 
 | ||||
| Future<MoorIsolate> _createMoorIsolate() async { | ||||
|   // This method is called from the main isolate. Since we can't use | ||||
|   // getApplicationDocumentsDirectory on a background isolate, we calculate | ||||
|   // the database path in the foreground isolate and then inform the | ||||
|   // background isolate about the path. | ||||
|   final dir = await pp.getApplicationDocumentsDirectory(); | ||||
|   final path = p.join(dir.path, 'db.sqlite'); | ||||
|   final receivePort = ReceivePort(); | ||||
| 
 | ||||
|   await Isolate.spawn( | ||||
|     _startBackground, | ||||
|     _IsolateStartRequest(receivePort.sendPort, path), | ||||
|   ); | ||||
| 
 | ||||
|   // _startBackground will send the MoorIsolate to this ReceivePort. | ||||
|   return (await receivePort.first as MoorIsolate); | ||||
| } | ||||
| 
 | ||||
| void _startBackground(_IsolateStartRequest request) { | ||||
|   // This is the entrypoint from the background isolate! Let's create | ||||
|   // the database from the path we received. | ||||
|   final executor = VmDatabase(File(request.targetPath)); | ||||
|   // We're using MoorIsolate.inCurrent here as this method already runs on a | ||||
|   // background isolate. If we used MoorIsolate.spawn, a third isolate would be | ||||
|   // started which is not what we want! | ||||
|   final moorIsolate = MoorIsolate.inCurrent( | ||||
|     () => DatabaseConnection.fromExecutor(executor), | ||||
|   ); | ||||
|   // Inform the starting isolate about this, so that it can call .connect(). | ||||
|   request.sendMoorIsolate.send(moorIsolate); | ||||
| } | ||||
| 
 | ||||
| // Used to bundle the SendPort and the target path, since isolate entrypoint | ||||
| // functions can only take one parameter. | ||||
| class _IsolateStartRequest { | ||||
|   final SendPort sendMoorIsolate; | ||||
|   final String targetPath; | ||||
| 
 | ||||
|   _IsolateStartRequest(this.sendMoorIsolate, this.targetPath); | ||||
| } | ||||
| 
 | ||||
| enum BackendStatus { | ||||
|   loading, | ||||
|   setup, | ||||
|   ready, | ||||
| } | ||||
| 
 | ||||
| class Backend extends StatefulWidget { | ||||
|   final Widget child; | ||||
| 
 | ||||
|   Backend({ | ||||
|     @required this.child, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   BackendState createState() => BackendState(); | ||||
| 
 | ||||
|   static BackendState of(BuildContext context) => | ||||
|       context.dependOnInheritedWidgetOfExactType<_InheritedBackend>().state; | ||||
| } | ||||
| 
 | ||||
| class BackendState extends State<Backend> { | ||||
|   final player = Player(); | ||||
|   final settings = Settings(); | ||||
| 
 | ||||
|   BackendStatus status = BackendStatus.loading; | ||||
|   Database db; | ||||
|   MusicusClient client; | ||||
|   MusicLibrary ml; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
|     _load(); | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return _InheritedBackend( | ||||
|       child: widget.child, | ||||
|       state: this, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   Future<void> _load() async { | ||||
|     MoorIsolate moorIsolate; | ||||
| 
 | ||||
|     final moorPort = IsolateNameServer.lookupPortByName('moorPort'); | ||||
|     if (moorPort != null) { | ||||
|       moorIsolate = MoorIsolate.fromConnectPort(moorPort); | ||||
|     } else { | ||||
|       moorIsolate = await _createMoorIsolate(); | ||||
|       IsolateNameServer.registerPortWithName( | ||||
|           moorIsolate.connectPort, 'moorPort'); | ||||
|     } | ||||
| 
 | ||||
|     final dbConnection = await moorIsolate.connect(); | ||||
|     db = Database.connect(dbConnection); | ||||
| 
 | ||||
|     player.setup(); | ||||
| 
 | ||||
|     await settings.load(); | ||||
| 
 | ||||
|     _updateMusicLibrary(settings.musicLibraryUri.value); | ||||
|     settings.musicLibraryUri.listen((uri) { | ||||
|       _updateMusicLibrary(uri); | ||||
|     }); | ||||
| 
 | ||||
|     _updateClient(settings.server.value); | ||||
|     settings.server.listen((serverSettings) { | ||||
|       _updateClient(serverSettings); | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   Future<void> _updateMusicLibrary(String uri) async { | ||||
|     if (uri == null) { | ||||
|       setState(() { | ||||
|         status = BackendStatus.setup; | ||||
|       }); | ||||
|     } else { | ||||
|       ml = MusicLibrary(uri); | ||||
|       await ml.load(); | ||||
|       setState(() { | ||||
|         status = BackendStatus.ready; | ||||
|       }); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   Future<void> _updateClient(ServerSettings serverSettings) async { | ||||
|     client = MusicusClient( | ||||
|       host: serverSettings.host, | ||||
|       port: serverSettings.port, | ||||
|       basePath: serverSettings.basePath, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   void dispose() { | ||||
|     super.dispose(); | ||||
|     client.dispose(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class _InheritedBackend extends InheritedWidget { | ||||
|   final Widget child; | ||||
|   final BackendState state; | ||||
| 
 | ||||
|   _InheritedBackend({ | ||||
|     @required this.child, | ||||
|     @required this.state, | ||||
|   }) : super(child: child); | ||||
| 
 | ||||
|   @override | ||||
|   bool updateShouldNotify(_InheritedBackend old) => true; | ||||
| } | ||||
|  | @ -1,96 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| 
 | ||||
| class EnsembleEditor extends StatefulWidget { | ||||
|   final Ensemble ensemble; | ||||
| 
 | ||||
|   EnsembleEditor({ | ||||
|     this.ensemble, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _EnsembleEditorState createState() => _EnsembleEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _EnsembleEditorState extends State<EnsembleEditor> { | ||||
|   final nameController = TextEditingController(); | ||||
| 
 | ||||
|   bool uploading = false; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.ensemble != null) { | ||||
|       nameController.text = widget.ensemble.name; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Ensemble'), | ||||
|         actions: <Widget>[ | ||||
|           uploading | ||||
|               ? Padding( | ||||
|                   padding: const EdgeInsets.all(16.0), | ||||
|                   child: Center( | ||||
|                     child: SizedBox( | ||||
|                       width: 24.0, | ||||
|                       height: 24.0, | ||||
|                       child: CircularProgressIndicator( | ||||
|                         strokeWidth: 2.0, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               : FlatButton( | ||||
|                   child: Text('DONE'), | ||||
|                   onPressed: () async { | ||||
|                     setState(() { | ||||
|                       uploading = true; | ||||
|                     }); | ||||
| 
 | ||||
|                     final ensemble = Ensemble( | ||||
|                       id: widget.ensemble?.id ?? generateId(), | ||||
|                       name: nameController.text, | ||||
|                     ); | ||||
| 
 | ||||
|                     final success = await backend.client.putEnsemble(ensemble); | ||||
| 
 | ||||
|                     setState(() { | ||||
|                       uploading = false; | ||||
|                     }); | ||||
| 
 | ||||
|                     if (success) { | ||||
|                       Navigator.pop(context, ensemble); | ||||
|                     } else { | ||||
|                       Scaffold.of(context).showSnackBar(SnackBar( | ||||
|                         content: Text('Failed to upload'), | ||||
|                       )); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ListView( | ||||
|         children: <Widget>[ | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.all(16.0), | ||||
|             child: TextField( | ||||
|               controller: nameController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'Name', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,97 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| 
 | ||||
| class InstrumentEditor extends StatefulWidget { | ||||
|   final Instrument instrument; | ||||
| 
 | ||||
|   InstrumentEditor({ | ||||
|     this.instrument, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _InstrumentEditorState createState() => _InstrumentEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _InstrumentEditorState extends State<InstrumentEditor> { | ||||
|   final nameController = TextEditingController(); | ||||
| 
 | ||||
|   bool uploading = false; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.instrument != null) { | ||||
|       nameController.text = widget.instrument.name; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Instrument/Role'), | ||||
|         actions: <Widget>[ | ||||
|           uploading | ||||
|               ? Padding( | ||||
|                   padding: const EdgeInsets.all(16.0), | ||||
|                   child: Center( | ||||
|                     child: SizedBox( | ||||
|                       width: 24.0, | ||||
|                       height: 24.0, | ||||
|                       child: CircularProgressIndicator( | ||||
|                         strokeWidth: 2.0, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               : FlatButton( | ||||
|                   child: Text('DONE'), | ||||
|                   onPressed: () async { | ||||
|                     setState(() { | ||||
|                       uploading = true; | ||||
|                     }); | ||||
| 
 | ||||
|                     final instrument = Instrument( | ||||
|                       id: widget.instrument?.id ?? generateId(), | ||||
|                       name: nameController.text, | ||||
|                     ); | ||||
| 
 | ||||
|                     final success = | ||||
|                         await backend.client.putInstrument(instrument); | ||||
| 
 | ||||
|                     setState(() { | ||||
|                       uploading = false; | ||||
|                     }); | ||||
| 
 | ||||
|                     if (success) { | ||||
|                       Navigator.pop(context, instrument); | ||||
|                     } else { | ||||
|                       Scaffold.of(context).showSnackBar(SnackBar( | ||||
|                         content: Text('Failed to upload'), | ||||
|                       )); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ListView( | ||||
|         children: <Widget>[ | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.all(16.0), | ||||
|             child: TextField( | ||||
|               controller: nameController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'Name', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,131 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../selectors/ensemble.dart'; | ||||
| import '../selectors/instruments.dart'; | ||||
| import '../selectors/person.dart'; | ||||
| 
 | ||||
| class PerformanceEditor extends StatefulWidget { | ||||
|   final PerformanceInfo performanceInfo; | ||||
| 
 | ||||
|   PerformanceEditor({ | ||||
|     this.performanceInfo, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _PerformanceEditorState createState() => _PerformanceEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _PerformanceEditorState extends State<PerformanceEditor> { | ||||
|   Person person; | ||||
|   Ensemble ensemble; | ||||
|   Instrument role; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.performanceInfo != null) { | ||||
|       person = widget.performanceInfo.person; | ||||
|       ensemble = widget.performanceInfo.ensemble; | ||||
|       role = widget.performanceInfo.role; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Edit performer'), | ||||
|         actions: <Widget>[ | ||||
|           FlatButton( | ||||
|             child: Text('DONE'), | ||||
|             onPressed: () => Navigator.pop( | ||||
|               context, | ||||
|               PerformanceInfo( | ||||
|                 person: person, | ||||
|                 ensemble: ensemble, | ||||
|                 role: role, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ListView( | ||||
|         children: <Widget>[ | ||||
|           ListTile( | ||||
|             title: Text('Person'), | ||||
|             subtitle: Text(person != null | ||||
|                 ? '${person.firstName} ${person.lastName}' | ||||
|                 : 'Select person'), | ||||
|             onTap: () async { | ||||
|               final Person newPerson = await Navigator.push( | ||||
|                 context, | ||||
|                 MaterialPageRoute( | ||||
|                   builder: (context) => PersonsSelector(), | ||||
|                   fullscreenDialog: true, | ||||
|                 ), | ||||
|               ); | ||||
| 
 | ||||
|               if (newPerson != null) { | ||||
|                 setState(() { | ||||
|                   person = newPerson; | ||||
|                   ensemble = null; | ||||
|                 }); | ||||
|               } | ||||
|             }, | ||||
|           ), | ||||
|           ListTile( | ||||
|             title: Text('Ensemble'), | ||||
|             subtitle: Text(ensemble?.name ?? 'Select ensemble'), | ||||
|             onTap: () async { | ||||
|               final Ensemble newEnsemble = await Navigator.push( | ||||
|                 context, | ||||
|                 MaterialPageRoute( | ||||
|                   builder: (context) => EnsembleSelector(), | ||||
|                   fullscreenDialog: true, | ||||
|                 ), | ||||
|               ); | ||||
| 
 | ||||
|               if (newEnsemble != null) { | ||||
|                 setState(() { | ||||
|                   ensemble = newEnsemble; | ||||
|                   person = null; | ||||
|                 }); | ||||
|               } | ||||
|             }, | ||||
|           ), | ||||
|           ListTile( | ||||
|             title: Text('Role'), | ||||
|             subtitle: Text(role?.name ?? 'Select instrument/role'), | ||||
|             trailing: role != null | ||||
|                 ? IconButton( | ||||
|                     icon: const Icon(Icons.delete), | ||||
|                     onPressed: () { | ||||
|                       setState(() { | ||||
|                         role = null; | ||||
|                       }); | ||||
|                     }, | ||||
|                   ) | ||||
|                 : null, | ||||
|             onTap: () async { | ||||
|               final Instrument newRole = await Navigator.push( | ||||
|                 context, | ||||
|                 MaterialPageRoute( | ||||
|                   builder: (context) => InstrumentsSelector(), | ||||
|                   fullscreenDialog: true, | ||||
|                 ), | ||||
|               ); | ||||
| 
 | ||||
|               if (newRole != null) { | ||||
|                 setState(() { | ||||
|                   role = newRole; | ||||
|                 }); | ||||
|               } | ||||
|             }, | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,108 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| 
 | ||||
| class PersonEditor extends StatefulWidget { | ||||
|   final Person person; | ||||
| 
 | ||||
|   PersonEditor({ | ||||
|     this.person, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _PersonEditorState createState() => _PersonEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _PersonEditorState extends State<PersonEditor> { | ||||
|   final firstNameController = TextEditingController(); | ||||
|   final lastNameController = TextEditingController(); | ||||
| 
 | ||||
|   bool uploading = false; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.person != null) { | ||||
|       firstNameController.text = widget.person.firstName; | ||||
|       lastNameController.text = widget.person.lastName; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Person'), | ||||
|         actions: <Widget>[ | ||||
|           uploading | ||||
|               ? Padding( | ||||
|                   padding: const EdgeInsets.all(16.0), | ||||
|                   child: Center( | ||||
|                     child: SizedBox( | ||||
|                       width: 24.0, | ||||
|                       height: 24.0, | ||||
|                       child: CircularProgressIndicator( | ||||
|                         strokeWidth: 2.0, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               : FlatButton( | ||||
|                   child: Text('DONE'), | ||||
|                   onPressed: () async { | ||||
|                     setState(() { | ||||
|                       uploading = true; | ||||
|                     }); | ||||
| 
 | ||||
|                     final person = Person( | ||||
|                       id: widget.person?.id ?? generateId(), | ||||
|                       firstName: firstNameController.text, | ||||
|                       lastName: lastNameController.text, | ||||
|                     ); | ||||
| 
 | ||||
|                     final success = await backend.client.putPerson(person); | ||||
| 
 | ||||
|                     setState(() { | ||||
|                       uploading = false; | ||||
|                     }); | ||||
| 
 | ||||
|                     if (success) { | ||||
|                       Navigator.pop(context, person); | ||||
|                     } else { | ||||
|                       Scaffold.of(context).showSnackBar(SnackBar( | ||||
|                         content: Text('Failed to upload'), | ||||
|                       )); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ListView( | ||||
|         children: <Widget>[ | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.all(16.0), | ||||
|             child: TextField( | ||||
|               controller: firstNameController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'First name', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.all(16.0), | ||||
|             child: TextField( | ||||
|               controller: lastNameController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'Last name', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,215 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../editors/performance.dart'; | ||||
| import '../selectors/recording.dart'; | ||||
| import '../selectors/work.dart'; | ||||
| 
 | ||||
| /// Screen for editing a recording. | ||||
| /// | ||||
| /// If the user has finished editing, the result will be returned using the | ||||
| /// navigator as a [RecordingSelectorResult] object. | ||||
| class RecordingEditor extends StatefulWidget { | ||||
|   /// The recording to edit. | ||||
|   /// | ||||
|   /// If this is null, a new recording will be created. | ||||
|   final RecordingInfo recordingInfo; | ||||
| 
 | ||||
|   RecordingEditor({ | ||||
|     this.recordingInfo, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _RecordingEditorState createState() => _RecordingEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _RecordingEditorState extends State<RecordingEditor> { | ||||
|   final commentController = TextEditingController(); | ||||
| 
 | ||||
|   bool uploading = false; | ||||
|   WorkInfo workInfo; | ||||
|   List<PerformanceInfo> performanceInfos = []; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.recordingInfo != null) { | ||||
|       final backend = Backend.of(context); | ||||
| 
 | ||||
|       () async { | ||||
|         workInfo = await backend.db.getWork(widget.recordingInfo.recording.id); | ||||
|         performanceInfos = List.from(widget.recordingInfo.performances); | ||||
|       }(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     Future<void> selectWork() async { | ||||
|       final WorkInfo newWorkInfo = await Navigator.push( | ||||
|           context, | ||||
|           MaterialPageRoute( | ||||
|             builder: (context) => WorkSelector(), | ||||
|             fullscreenDialog: true, | ||||
|           )); | ||||
| 
 | ||||
|       if (newWorkInfo != null) { | ||||
|         setState(() { | ||||
|           workInfo = newWorkInfo; | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     final List<Widget> performanceTiles = []; | ||||
|     for (var i = 0; i < performanceInfos.length; i++) { | ||||
|       final p = performanceInfos[i]; | ||||
| 
 | ||||
|       performanceTiles.add(ListTile( | ||||
|         title: Text(p.person != null | ||||
|             ? '${p.person.firstName} ${p.person.lastName}' | ||||
|             : p.ensemble.name), | ||||
|         subtitle: p.role != null ? Text(p.role.name) : null, | ||||
|         trailing: IconButton( | ||||
|           icon: const Icon(Icons.delete), | ||||
|           onPressed: () { | ||||
|             setState(() { | ||||
|               performanceInfos.remove(p); | ||||
|             }); | ||||
|           }, | ||||
|         ), | ||||
|         onTap: () async { | ||||
|           final PerformanceInfo performanceInfo = await Navigator.push( | ||||
|               context, | ||||
|               MaterialPageRoute( | ||||
|                 builder: (context) => PerformanceEditor( | ||||
|                   performanceInfo: p, | ||||
|                 ), | ||||
|                 fullscreenDialog: true, | ||||
|               )); | ||||
| 
 | ||||
|           if (performanceInfo != null) { | ||||
|             setState(() { | ||||
|               performanceInfos[i] = performanceInfo; | ||||
|             }); | ||||
|           } | ||||
|         }, | ||||
|       )); | ||||
|     } | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Recording'), | ||||
|         actions: <Widget>[ | ||||
|           uploading | ||||
|               ? Padding( | ||||
|                   padding: const EdgeInsets.all(16.0), | ||||
|                   child: Center( | ||||
|                     child: SizedBox( | ||||
|                       width: 24.0, | ||||
|                       height: 24.0, | ||||
|                       child: CircularProgressIndicator( | ||||
|                         strokeWidth: 2.0, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               : FlatButton( | ||||
|                   child: Text('DONE'), | ||||
|                   onPressed: () async { | ||||
|                     setState(() { | ||||
|                       uploading = true; | ||||
|                     }); | ||||
| 
 | ||||
|                     final recordingInfo = RecordingInfo( | ||||
|                       recording: Recording( | ||||
|                         id: widget?.recordingInfo?.recording?.id ?? | ||||
|                             generateId(), | ||||
|                         work: workInfo.work.id, | ||||
|                         comment: commentController.text, | ||||
|                       ), | ||||
|                       performances: performanceInfos, | ||||
|                     ); | ||||
| 
 | ||||
|                     final success = | ||||
|                         await backend.client.putRecording(recordingInfo); | ||||
| 
 | ||||
|                     setState(() { | ||||
|                       uploading = false; | ||||
|                     }); | ||||
| 
 | ||||
|                     if (success) { | ||||
|                       Navigator.pop( | ||||
|                         context, | ||||
|                         RecordingSelectorResult( | ||||
|                           workInfo: workInfo, | ||||
|                           recordingInfo: recordingInfo, | ||||
|                         ), | ||||
|                       ); | ||||
|                     } else { | ||||
|                       Scaffold.of(context).showSnackBar(SnackBar( | ||||
|                         content: Text('Failed to upload'), | ||||
|                       )); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ListView( | ||||
|         children: <Widget>[ | ||||
|           workInfo != null | ||||
|               ? ListTile( | ||||
|                   title: Text(workInfo.work.title), | ||||
|                   subtitle: Text(workInfo.composers | ||||
|                       .map((p) => '${p.firstName} ${p.lastName}') | ||||
|                       .join(', ')), | ||||
|                   onTap: selectWork, | ||||
|                 ) | ||||
|               : ListTile( | ||||
|                   title: Text('Work'), | ||||
|                   subtitle: Text('Select work'), | ||||
|                   onTap: selectWork, | ||||
|                 ), | ||||
|           Padding( | ||||
|             padding: const EdgeInsets.only( | ||||
|               left: 16.0, | ||||
|               right: 16.0, | ||||
|               top: 0.0, | ||||
|               bottom: 16.0, | ||||
|             ), | ||||
|             child: TextField( | ||||
|               controller: commentController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'Comment', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           ListTile( | ||||
|             title: Text('Performers'), | ||||
|             trailing: IconButton( | ||||
|               icon: const Icon(Icons.add), | ||||
|               onPressed: () async { | ||||
|                 final PerformanceInfo model = await Navigator.push( | ||||
|                     context, | ||||
|                     MaterialPageRoute( | ||||
|                       builder: (context) => PerformanceEditor(), | ||||
|                       fullscreenDialog: true, | ||||
|                     )); | ||||
| 
 | ||||
|                 if (model != null) { | ||||
|                   setState(() { | ||||
|                     performanceInfos.add(model); | ||||
|                   }); | ||||
|                 } | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|           ...performanceTiles, | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,163 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../music_library.dart'; | ||||
| import '../selectors/files.dart'; | ||||
| import '../selectors/recording.dart'; | ||||
| import '../widgets/recording_tile.dart'; | ||||
| 
 | ||||
| class TrackModel { | ||||
|   int workPartIndex; | ||||
|   String workPartTitle; | ||||
|   String fileName; | ||||
| 
 | ||||
|   TrackModel(this.fileName); | ||||
| } | ||||
| 
 | ||||
| class TracksEditor extends StatefulWidget { | ||||
|   @override | ||||
|   _TracksEditorState createState() => _TracksEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _TracksEditorState extends State<TracksEditor> { | ||||
|   BackendState backend; | ||||
|   WorkInfo workInfo; | ||||
|   RecordingInfo recordingInfo; | ||||
|   String parentId; | ||||
|   List<TrackModel> trackModels = []; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Tracks'), | ||||
|         actions: <Widget>[ | ||||
|           FlatButton( | ||||
|             child: Text('DONE'), | ||||
|             onPressed: () async { | ||||
|               final List<Track> tracks = []; | ||||
| 
 | ||||
|               for (var i = 0; i < trackModels.length; i++) { | ||||
|                 final trackModel = trackModels[i]; | ||||
| 
 | ||||
|                 tracks.add(Track( | ||||
|                   fileName: trackModel.fileName, | ||||
|                   recordingId: recordingInfo.recording.id, | ||||
|                   index: i, | ||||
|                   partIds: [trackModel.workPartIndex], | ||||
|                 )); | ||||
|               } | ||||
| 
 | ||||
|               // We need to copy all information associated with this track we | ||||
|               // got by asking the server to our local database. For now, we | ||||
|               // will just override everything that we already had previously. | ||||
|               backend.db.updateWork(workInfo); | ||||
|               backend.db.updateRecording(recordingInfo); | ||||
| 
 | ||||
|               backend.ml.addTracks(parentId, tracks); | ||||
| 
 | ||||
|               Navigator.pop(context); | ||||
|             }, | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ReorderableListView( | ||||
|         header: Column( | ||||
|           children: <Widget>[ | ||||
|             ListTile( | ||||
|               title: recordingInfo != null | ||||
|                   ? RecordingTile( | ||||
|                       workInfo: workInfo, | ||||
|                       recordingInfo: recordingInfo, | ||||
|                     ) | ||||
|                   : Text('Select recording'), | ||||
|               onTap: selectRecording, | ||||
|             ), | ||||
|             ListTile( | ||||
|               title: Text('Files'), | ||||
|               trailing: IconButton( | ||||
|                 icon: const Icon(Icons.edit), | ||||
|                 onPressed: () async { | ||||
|                   final FilesSelectorResult result = await Navigator.push( | ||||
|                     context, | ||||
|                     MaterialPageRoute( | ||||
|                       builder: (context) => FilesSelector(), | ||||
|                     ), | ||||
|                   ); | ||||
| 
 | ||||
|                   if (result != null) { | ||||
|                     final List<TrackModel> newTrackModels = []; | ||||
| 
 | ||||
|                     for (final document in result.selection) { | ||||
|                       newTrackModels.add(TrackModel(document.name)); | ||||
|                     } | ||||
| 
 | ||||
|                     setState(() { | ||||
|                       parentId = result.parentId; | ||||
|                       trackModels = newTrackModels; | ||||
|                     }); | ||||
| 
 | ||||
|                     if (recordingInfo != null) { | ||||
|                       updateAutoParts(); | ||||
|                     } | ||||
|                   } | ||||
|                 }, | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|         children: trackModels | ||||
|             .map((t) => ListTile( | ||||
|                   key: Key(t.hashCode.toString()), | ||||
|                   leading: const Icon(Icons.drag_handle), | ||||
|                   title: Text(t.workPartTitle ?? 'Set work part'), | ||||
|                   subtitle: Text(t.fileName), | ||||
|                 )) | ||||
|             .toList(), | ||||
|         onReorder: (i1, i2) { | ||||
|           setState(() { | ||||
|             final track = trackModels.removeAt(i1); | ||||
|             final newIndex = i2 > i1 ? i2 - 1 : i2; | ||||
|             trackModels.insert(newIndex, track); | ||||
|           }); | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   Future<void> selectRecording() async { | ||||
|     final RecordingSelectorResult result = await Navigator.push( | ||||
|       context, | ||||
|       MaterialPageRoute( | ||||
|         builder: (context) => RecordingSelector(), | ||||
|       ), | ||||
|     ); | ||||
| 
 | ||||
|     if (result != null) { | ||||
|       setState(() { | ||||
|         workInfo = result.workInfo; | ||||
|         recordingInfo = result.recordingInfo; | ||||
|       }); | ||||
| 
 | ||||
|       updateAutoParts(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Automatically associate the tracks with work parts. | ||||
|   Future<void> updateAutoParts() async { | ||||
|     setState(() { | ||||
|       for (var i = 0; i < trackModels.length; i++) { | ||||
|         if (i >= workInfo.parts.length) { | ||||
|           trackModels[i].workPartIndex = null; | ||||
|           trackModels[i].workPartTitle = null; | ||||
|         } else { | ||||
|           trackModels[i].workPartIndex = workInfo.parts[i].work.partIndex; | ||||
|           trackModels[i].workPartTitle = workInfo.parts[i].work.title; | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|  | @ -1,371 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../selectors/instruments.dart'; | ||||
| import '../selectors/person.dart'; | ||||
| 
 | ||||
| class PartData { | ||||
|   final titleController = TextEditingController(); | ||||
| 
 | ||||
|   Person composer; | ||||
|   List<Instrument> instruments; | ||||
| 
 | ||||
|   PartData({ | ||||
|     String title, | ||||
|     this.composer, | ||||
|     this.instruments = const [], | ||||
|   }) { | ||||
|     titleController.text = title ?? ''; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class WorkProperties extends StatelessWidget { | ||||
|   final TextEditingController titleController; | ||||
|   final Person composer; | ||||
|   final List<Instrument> instruments; | ||||
|   final void Function(Person) onComposerChanged; | ||||
|   final void Function(List<Instrument>) onInstrumentsChanged; | ||||
| 
 | ||||
|   WorkProperties({ | ||||
|     @required this.titleController, | ||||
|     @required this.composer, | ||||
|     @required this.instruments, | ||||
|     @required this.onComposerChanged, | ||||
|     @required this.onInstrumentsChanged, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Padding( | ||||
|           padding: const EdgeInsets.all(16.0), | ||||
|           child: TextField( | ||||
|             controller: titleController, | ||||
|             decoration: InputDecoration( | ||||
|               labelText: 'Title', | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('Composer'), | ||||
|           subtitle: Text(composer != null | ||||
|               ? '${composer.firstName} ${composer.lastName}' | ||||
|               : 'Select composer'), | ||||
|           trailing: IconButton( | ||||
|             icon: const Icon(Icons.delete), | ||||
|             onPressed: () { | ||||
|               onComposerChanged(null); | ||||
|             }, | ||||
|           ), | ||||
|           onTap: () async { | ||||
|             final Person person = await Navigator.push( | ||||
|                 context, | ||||
|                 MaterialPageRoute( | ||||
|                   builder: (context) => PersonsSelector(), | ||||
|                   fullscreenDialog: true, | ||||
|                 )); | ||||
| 
 | ||||
|             if (person != null) { | ||||
|               onComposerChanged(person); | ||||
|             } | ||||
|           }, | ||||
|         ), | ||||
|         ListTile( | ||||
|           title: Text('Instruments'), | ||||
|           subtitle: Text(instruments.isNotEmpty | ||||
|               ? instruments.map((i) => i.name).join(', ') | ||||
|               : 'Select instruments'), | ||||
|           trailing: IconButton( | ||||
|               icon: const Icon(Icons.delete), | ||||
|               onPressed: () { | ||||
|                 onInstrumentsChanged([]); | ||||
|               }), | ||||
|           onTap: () async { | ||||
|             final List<Instrument> selection = await Navigator.push( | ||||
|                 context, | ||||
|                 MaterialPageRoute( | ||||
|                   builder: (context) => InstrumentsSelector( | ||||
|                     multiple: true, | ||||
|                     selection: instruments, | ||||
|                   ), | ||||
|                   fullscreenDialog: true, | ||||
|                 )); | ||||
| 
 | ||||
|             if (selection != null) { | ||||
|               onInstrumentsChanged(selection); | ||||
|             } | ||||
|           }, | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class PartTile extends StatefulWidget { | ||||
|   final PartData part; | ||||
|   final void Function() onMore; | ||||
|   final void Function() onDelete; | ||||
| 
 | ||||
|   PartTile({ | ||||
|     Key key, | ||||
|     @required this.part, | ||||
|     @required this.onMore, | ||||
|     @required this.onDelete, | ||||
|   }) : super(key: key); | ||||
| 
 | ||||
|   @override | ||||
|   _PartTileState createState() => _PartTileState(); | ||||
| } | ||||
| 
 | ||||
| class _PartTileState extends State<PartTile> { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Row( | ||||
|       children: <Widget>[ | ||||
|         Padding( | ||||
|           padding: const EdgeInsets.only(left: 16.0, right: 8.0), | ||||
|           child: Icon( | ||||
|             Icons.drag_handle, | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: TextField( | ||||
|             controller: widget.part.titleController, | ||||
|             decoration: InputDecoration( | ||||
|               border: InputBorder.none, | ||||
|               hintText: 'Part title', | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         IconButton( | ||||
|           icon: const Icon(Icons.more_horiz), | ||||
|           onPressed: widget.onMore, | ||||
|         ), | ||||
|         IconButton( | ||||
|           icon: const Icon(Icons.delete), | ||||
|           onPressed: widget.onDelete, | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// Screen for editing a work. | ||||
| /// | ||||
| /// If the user is finished editing, the result will be returned as a [WorkInfo] | ||||
| /// object. | ||||
| class WorkEditor extends StatefulWidget { | ||||
|   /// The work to edit. | ||||
|   /// | ||||
|   /// If this is null, a new work will be created. | ||||
|   final WorkInfo workInfo; | ||||
| 
 | ||||
|   WorkEditor({ | ||||
|     this.workInfo, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _WorkEditorState createState() => _WorkEditorState(); | ||||
| } | ||||
| 
 | ||||
| class _WorkEditorState extends State<WorkEditor> { | ||||
|   final titleController = TextEditingController(); | ||||
| 
 | ||||
|   bool uploading = false; | ||||
|   Person composer; | ||||
|   List<Instrument> instruments = []; | ||||
|   List<PartData> parts = []; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.workInfo != null) { | ||||
|       titleController.text = widget.workInfo.work.title; | ||||
|       // TODO: Theoretically this includes the composers of all parts. | ||||
|       composer = widget.workInfo.composers.first; | ||||
|       instruments = List.from(widget.workInfo.instruments); | ||||
| 
 | ||||
|       for (final partInfo in widget.workInfo.parts) { | ||||
|         parts.add(PartData( | ||||
|           title: partInfo.work.title, | ||||
|           composer: partInfo.composer, | ||||
|           instruments: List.from(partInfo.instruments), | ||||
|         )); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     final List<Widget> partTiles = []; | ||||
|     for (var i = 0; i < parts.length; i++) { | ||||
|       final part = parts[i]; | ||||
| 
 | ||||
|       partTiles.add(PartTile( | ||||
|         key: Key(part.hashCode.toString()), | ||||
|         part: part, | ||||
|         onMore: () { | ||||
|           showDialog( | ||||
|             context: context, | ||||
|             builder: (context) => StatefulBuilder( | ||||
|               builder: (context, setState) => Dialog( | ||||
|                 child: ListView( | ||||
|                   shrinkWrap: true, | ||||
|                   children: <Widget>[ | ||||
|                     WorkProperties( | ||||
|                       titleController: part.titleController, | ||||
|                       composer: part.composer, | ||||
|                       instruments: part.instruments, | ||||
|                       onComposerChanged: (composer) { | ||||
|                         setState(() { | ||||
|                           part.composer = composer; | ||||
|                         }); | ||||
|                       }, | ||||
|                       onInstrumentsChanged: (instruments) { | ||||
|                         setState(() { | ||||
|                           part.instruments = instruments; | ||||
|                         }); | ||||
|                       }, | ||||
|                     ), | ||||
|                   ], | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ); | ||||
|         }, | ||||
|         onDelete: () { | ||||
|           setState(() { | ||||
|             parts.removeAt(i); | ||||
|           }); | ||||
|         }, | ||||
|       )); | ||||
|     } | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Work'), | ||||
|         actions: <Widget>[ | ||||
|           uploading | ||||
|               ? Padding( | ||||
|                   padding: const EdgeInsets.all(16.0), | ||||
|                   child: Center( | ||||
|                     child: SizedBox( | ||||
|                       width: 24.0, | ||||
|                       height: 24.0, | ||||
|                       child: CircularProgressIndicator( | ||||
|                         strokeWidth: 2.0, | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               : FlatButton( | ||||
|                   child: Text('DONE'), | ||||
|                   onPressed: () async { | ||||
|                     setState(() { | ||||
|                       uploading = true; | ||||
|                     }); | ||||
| 
 | ||||
|                     final workId = widget?.workInfo?.work?.id ?? generateId(); | ||||
| 
 | ||||
|                     List<PartInfo> partInfos = []; | ||||
|                     for (var i = 0; i < parts.length; i++) { | ||||
|                       final part = parts[i]; | ||||
|                       partInfos.add(PartInfo( | ||||
|                         work: Work( | ||||
|                           id: generateId(), | ||||
|                           title: part.titleController.text, | ||||
|                           composer: part.composer?.id, | ||||
|                           partOf: workId, | ||||
|                           partIndex: i, | ||||
|                         ), | ||||
|                         instruments: part.instruments, | ||||
|                         composer: part.composer, | ||||
|                       )); | ||||
|                     } | ||||
| 
 | ||||
|                     final workInfo = WorkInfo( | ||||
|                       work: Work( | ||||
|                         id: workId, | ||||
|                         title: titleController.text, | ||||
|                         composer: composer?.id, | ||||
|                       ), | ||||
|                       instruments: instruments, | ||||
|                       // TODO: Theoretically, this should include all composers | ||||
|                       // from the parts. | ||||
|                       composers: [composer], | ||||
|                       parts: partInfos, | ||||
|                     ); | ||||
| 
 | ||||
|                     final success = await backend.client.putWork(workInfo); | ||||
| 
 | ||||
|                     setState(() { | ||||
|                       uploading = false; | ||||
|                     }); | ||||
| 
 | ||||
|                     if (success) { | ||||
|                       Navigator.pop(context, workInfo); | ||||
|                     } else { | ||||
|                       Scaffold.of(context).showSnackBar(SnackBar( | ||||
|                         content: Text('Failed to upload'), | ||||
|                       )); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|         ], | ||||
|       ), | ||||
|       body: ReorderableListView( | ||||
|         header: Column( | ||||
|           crossAxisAlignment: CrossAxisAlignment.start, | ||||
|           children: <Widget>[ | ||||
|             WorkProperties( | ||||
|               titleController: titleController, | ||||
|               composer: composer, | ||||
|               instruments: instruments, | ||||
|               onComposerChanged: (newComposer) { | ||||
|                 setState(() { | ||||
|                   composer = newComposer; | ||||
|                 }); | ||||
|               }, | ||||
|               onInstrumentsChanged: (newInstruments) { | ||||
|                 setState(() { | ||||
|                   instruments = newInstruments; | ||||
|                 }); | ||||
|               }, | ||||
|             ), | ||||
|             if (parts.length > 0) | ||||
|               Padding( | ||||
|                 padding: const EdgeInsets.only(left: 16.0, top: 16.0), | ||||
|                 child: Text( | ||||
|                   'Parts', | ||||
|                   style: Theme.of(context).textTheme.subtitle1, | ||||
|                 ), | ||||
|               ), | ||||
|           ], | ||||
|         ), | ||||
|         children: partTiles, | ||||
|         onReorder: (i1, i2) { | ||||
|           setState(() { | ||||
|             final part = parts.removeAt(i1); | ||||
|             final newIndex = i2 > i1 ? i2 - 1 : i2; | ||||
| 
 | ||||
|             parts.insert(newIndex, part); | ||||
|           }); | ||||
|         }, | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton.extended( | ||||
|         icon: const Icon(Icons.add), | ||||
|         label: Text('Add part'), | ||||
|         onPressed: () { | ||||
|           setState(() { | ||||
|             parts.add(PartData()); | ||||
|           }); | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,12 +1,26 @@ | |||
| import 'package:audio_service/audio_service.dart'; | ||||
| import 'package:flutter/widgets.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:path/path.dart' as p; | ||||
| import 'package:path_provider/path_provider.dart' as pp; | ||||
| 
 | ||||
| import 'app.dart'; | ||||
| import 'backend.dart'; | ||||
| import 'settings.dart'; | ||||
| import 'platform.dart'; | ||||
| import 'playback.dart'; | ||||
| 
 | ||||
| Future<void> main() async { | ||||
|   WidgetsFlutterBinding.ensureInitialized(); | ||||
| 
 | ||||
|   final dir = await pp.getApplicationDocumentsDirectory(); | ||||
|   final dbPath = p.join(dir.path, 'db.sqlite'); | ||||
| 
 | ||||
| void main() { | ||||
|   runApp(AudioServiceWidget( | ||||
|     child: Backend( | ||||
|     child: MusicusBackend( | ||||
|       dbPath: dbPath, | ||||
|       settingsStorage: SettingsStorage(), | ||||
|       platform: MusicusAndroidPlatform(), | ||||
|       playback: Playback(), | ||||
|       child: App(), | ||||
|     ), | ||||
|   )); | ||||
|  |  | |||
|  | @ -1,189 +0,0 @@ | |||
| import 'dart:convert'; | ||||
| 
 | ||||
| import 'package:flutter/services.dart'; | ||||
| 
 | ||||
| import 'platform.dart'; | ||||
| 
 | ||||
| /// Bundles a [Track] with the URI of the audio file it represents. | ||||
| /// | ||||
| /// The uri shouldn't be stored on disk, but will be used at runtime. | ||||
| class InternalTrack { | ||||
|   /// The represented track. | ||||
|   final Track track; | ||||
| 
 | ||||
|   /// The URI of the represented audio file as retrieved from the SAF. | ||||
|   final String uri; | ||||
| 
 | ||||
|   InternalTrack({ | ||||
|     this.track, | ||||
|     this.uri, | ||||
|   }); | ||||
| 
 | ||||
|   factory InternalTrack.fromJson(Map<String, dynamic> json) => InternalTrack( | ||||
|         track: Track.fromJson(json['track']), | ||||
|         uri: json['uri'], | ||||
|       ); | ||||
| 
 | ||||
|   Map<String, dynamic> toJson() => { | ||||
|         'track': track.toJson(), | ||||
|         'uri': uri, | ||||
|       }; | ||||
| } | ||||
| 
 | ||||
| /// Description of a concrete audio file. | ||||
| /// | ||||
| /// This gets stored in the folder of the audio file and links the audio file | ||||
| /// to a recording in the database. | ||||
| class Track { | ||||
|   /// The name of the file that contains the track's audio. | ||||
|   /// | ||||
|   /// This corresponds to a document ID in terms of the Android Storage Access | ||||
|   /// Framework. | ||||
|   final String fileName; | ||||
| 
 | ||||
|   /// Index within the list of tracks for the corresponding recording. | ||||
|   final int index; | ||||
| 
 | ||||
|   /// Of which recording this track is a part of. | ||||
|   final int recordingId; | ||||
| 
 | ||||
|   /// Which work parts of the recorded work are contained in this track. | ||||
|   final List<int> partIds; | ||||
| 
 | ||||
|   Track({ | ||||
|     this.fileName, | ||||
|     this.index, | ||||
|     this.recordingId, | ||||
|     this.partIds, | ||||
|   }); | ||||
| 
 | ||||
|   factory Track.fromJson(Map<String, dynamic> json) => Track( | ||||
|         fileName: json['fileName'], | ||||
|         index: json['index'], | ||||
|         recordingId: json['recording'], | ||||
|         partIds: List.from(json['parts']), | ||||
|       ); | ||||
| 
 | ||||
|   Map<String, dynamic> toJson() => { | ||||
|         'fileName': fileName, | ||||
|         'index': index, | ||||
|         'recording': recordingId, | ||||
|         'parts': partIds, | ||||
|       }; | ||||
| } | ||||
| 
 | ||||
| /// Representation of all tracked audio files in one folder. | ||||
| class MusicusFile { | ||||
|   /// Current version of the Musicus file format. | ||||
|   /// | ||||
|   /// If incompatible changes are made, this will be increased by one. | ||||
|   static const currentVersion = 0; | ||||
| 
 | ||||
|   /// Musicus file format version in use. | ||||
|   /// | ||||
|   /// This will be used in the future, if incompatible changes are made. | ||||
|   final int version; | ||||
| 
 | ||||
|   /// List of [Track] objects. | ||||
|   final List<Track> tracks; | ||||
| 
 | ||||
|   MusicusFile({ | ||||
|     this.version = currentVersion, | ||||
|     List<Track> tracks, | ||||
|   }) : tracks = tracks ?? []; | ||||
| 
 | ||||
|   factory MusicusFile.fromJson(Map<String, dynamic> json) => MusicusFile( | ||||
|         version: json['version'], | ||||
|         tracks: json['tracks'] | ||||
|             .map<Track>((trackJson) => Track.fromJson(trackJson)) | ||||
|             .toList(growable: true), | ||||
|       ); | ||||
| 
 | ||||
|   Map<String, dynamic> toJson() => { | ||||
|         'version': version, | ||||
|         'tracks': tracks.map((t) => t.toJson()).toList(), | ||||
|       }; | ||||
| } | ||||
| 
 | ||||
| /// Manager for all available tracks and their representation on disk. | ||||
| class MusicLibrary { | ||||
|   static const platform = MethodChannel('de.johrpan.musicus/platform'); | ||||
| 
 | ||||
|   /// URI of the music library folder. | ||||
|   /// | ||||
|   /// This is a tree URI in the terms of the Android Storage Access Framework. | ||||
|   final String treeUri; | ||||
| 
 | ||||
|   /// Map of all available tracks by recording ID. | ||||
|   /// | ||||
|   /// These are [InternalTrack] objects to store the URI of the corresponding | ||||
|   /// audio file alongside the real [Track] object. | ||||
|   final Map<int, List<InternalTrack>> tracks = {}; | ||||
| 
 | ||||
|   MusicLibrary(this.treeUri); | ||||
| 
 | ||||
|   /// Load all available tracks. | ||||
|   /// | ||||
|   /// This recursively searches through the whole music library, reads the | ||||
|   /// content of all files called musicus.json and stores all track information | ||||
|   /// that it found. | ||||
|   Future<void> load() async { | ||||
|     // TODO: Consider capping the recursion somewhere. | ||||
|     Future<void> recurse([String parentId]) async { | ||||
|       final children = await Platform.getChildren(treeUri, parentId); | ||||
| 
 | ||||
|       for (final child in children) { | ||||
|         if (child.isDirectory) { | ||||
|           recurse(child.id); | ||||
|         } else if (child.name == 'musicus.json') { | ||||
|           final content = await Platform.readFile(treeUri, child.id); | ||||
|           final musicusFile = MusicusFile.fromJson(jsonDecode(content)); | ||||
|           for (final track in musicusFile.tracks) { | ||||
|             _indexTrack(parentId, track); | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     await recurse(); | ||||
|   } | ||||
| 
 | ||||
|   /// Add a list of new tracks to the music library. | ||||
|   /// | ||||
|   /// They are stored in this instance and on disk in the directory denoted by | ||||
|   /// [parentId]. | ||||
|   Future<void> addTracks(String parentId, List<Track> newTracks) async { | ||||
|     MusicusFile musicusFile; | ||||
| 
 | ||||
|     final oldContent = | ||||
|         await Platform.readFileByName(treeUri, parentId, 'musicus.json'); | ||||
| 
 | ||||
|     if (oldContent != null) { | ||||
|       musicusFile = MusicusFile.fromJson(jsonDecode(oldContent)); | ||||
|     } else { | ||||
|       musicusFile = MusicusFile(); | ||||
|     } | ||||
| 
 | ||||
|     for (final track in newTracks) { | ||||
|       _indexTrack(parentId, track); | ||||
|       musicusFile.tracks.add(track); | ||||
|     } | ||||
| 
 | ||||
|     await Platform.writeFileByName( | ||||
|         treeUri, parentId, 'musicus.json', jsonEncode(musicusFile.toJson())); | ||||
|   } | ||||
| 
 | ||||
|   /// Add a track to the map of available tracks. | ||||
|   Future<void> _indexTrack(String parentId, Track track) async { | ||||
|     final iTrack = InternalTrack( | ||||
|       track: track, | ||||
|       uri: await Platform.getUriByName(treeUri, parentId, track.fileName), | ||||
|     ); | ||||
| 
 | ||||
|     if (tracks.containsKey(track.recordingId)) { | ||||
|       tracks[track.recordingId].add(iTrack); | ||||
|     } else { | ||||
|       tracks[track.recordingId] = [iTrack]; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -1,44 +1,16 @@ | |||
| import 'package:flutter/services.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| 
 | ||||
| /// Object representing a document in Storage Access Framework terms. | ||||
| class Document { | ||||
|   /// Unique document ID given by the SAF. | ||||
|   final String id; | ||||
| 
 | ||||
|   /// Name of the document (i.e. file name). | ||||
|   final String name; | ||||
| 
 | ||||
|   /// Document ID of the parent document. | ||||
|   final String parent; | ||||
| 
 | ||||
|   /// Whether this document represents a directory. | ||||
|   final bool isDirectory; | ||||
| 
 | ||||
|   // Use Map<dynamic, dynamic> here, as we get casting errors otherwise. This | ||||
|   // won't be typesafe anyway. | ||||
|   Document.fromJson(Map<dynamic, dynamic> json) | ||||
|       : id = json['id'], | ||||
|         name = json['name'], | ||||
|         parent = json['parent'], | ||||
|         isDirectory = json['isDirectory']; | ||||
| } | ||||
| 
 | ||||
| /// Collection of methods that are implemented platform dependent. | ||||
| class Platform { | ||||
| class MusicusAndroidPlatform extends MusicusPlatform { | ||||
|   static const _platform = MethodChannel('de.johrpan.musicus/platform'); | ||||
| 
 | ||||
|   /// Get child documents. | ||||
|   /// | ||||
|   /// [treeId] is the base URI as requested from the SAF. | ||||
|   /// [parentId] is the document ID of the parent. If this is null, the children | ||||
|   /// of the tree base will be returned. | ||||
|   static Future<List<Document>> getChildren( | ||||
|       String treeUri, String parentId) async { | ||||
|   @override | ||||
|   Future<List<Document>> getChildren(String parentId) async { | ||||
|     final List<Map<dynamic, dynamic>> childrenJson = | ||||
|         await _platform.invokeListMethod( | ||||
|       'getChildren', | ||||
|       { | ||||
|         'treeUri': treeUri, | ||||
|         'treeUri': basePath, | ||||
|         'parentId': parentId, | ||||
|       }, | ||||
|     ); | ||||
|  | @ -48,65 +20,51 @@ class Platform { | |||
|         .toList(); | ||||
|   } | ||||
| 
 | ||||
|   /// Read contents of file. | ||||
|   /// | ||||
|   /// [treeId] is the base URI from the SAF, [id] is the document ID of the | ||||
|   /// file. | ||||
|   static Future<String> readFile(String treeUri, String id) async { | ||||
|   @override | ||||
|   Future<String> getIdentifier(String parentId, String fileName) async { | ||||
|     return await _platform.invokeMethod( | ||||
|       'getUriByName', | ||||
|       { | ||||
|         'treeUri': basePath, | ||||
|         'parentId': parentId, | ||||
|         'fileName': fileName, | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Future<String> readDocument(String id) async { | ||||
|     return await _platform.invokeMethod( | ||||
|       'readFile', | ||||
|       { | ||||
|         'treeUri': treeUri, | ||||
|         'treeUri': basePath, | ||||
|         'id': id, | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   /// Get document URI by file name | ||||
|   /// | ||||
|   /// [treeId] is the base URI from the SAF, [parentId] is the document ID of | ||||
|   /// the parent directory. | ||||
|   static Future<String> getUriByName( | ||||
|       String treeUri, String parentId, String fileName) async { | ||||
|     return await _platform.invokeMethod( | ||||
|       'getUriByName', | ||||
|       { | ||||
|         'treeUri': treeUri, | ||||
|         'parentId': parentId, | ||||
|         'fileName': fileName, | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   /// Read contents of file by name | ||||
|   /// | ||||
|   /// [treeId] is the base URI from the SAF, [parentId] is the document ID of | ||||
|   /// the parent directory. | ||||
|   static Future<String> readFileByName( | ||||
|       String treeUri, String parentId, String fileName) async { | ||||
|   @override | ||||
|   Future<String> readDocumentByName(String parentId, String fileName) async { | ||||
|     return await _platform.invokeMethod( | ||||
|       'readFileByName', | ||||
|       { | ||||
|         'treeUri': treeUri, | ||||
|         'treeUri': basePath, | ||||
|         'parentId': parentId, | ||||
|         'fileName': fileName, | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   /// Write to file by name | ||||
|   /// | ||||
|   /// [treeId] is the base URI from the SAF, [parentId] is the document ID of | ||||
|   /// the parent directory. | ||||
|   static Future<void> writeFileByName( | ||||
|       String treeUri, String parentId, String fileName, String content) async { | ||||
|   @override | ||||
|   Future<void> writeDocumentByName( | ||||
|       String parentId, String fileName, String contents) async { | ||||
|     await _platform.invokeMethod( | ||||
|       'writeFileByName', | ||||
|       { | ||||
|         'treeUri': treeUri, | ||||
|         'treeUri': basePath, | ||||
|         'parentId': parentId, | ||||
|         'fileName': fileName, | ||||
|         'content': content, | ||||
|         'content': contents, | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
|  |  | |||
|  | @ -6,10 +6,8 @@ import 'dart:ui'; | |||
| import 'package:audio_service/audio_service.dart'; | ||||
| import 'package:moor/isolate.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:musicus_player/musicus_player.dart'; | ||||
| import 'package:rxdart/rxdart.dart'; | ||||
| 
 | ||||
| import 'music_library.dart'; | ||||
| 
 | ||||
| const _portName = 'playbackService'; | ||||
| 
 | ||||
|  | @ -18,61 +16,11 @@ void _playbackServiceEntrypoint() { | |||
|   AudioServiceBackground.run(() => _PlaybackService()); | ||||
| } | ||||
| 
 | ||||
| class Player { | ||||
|   /// Whether the player is active. | ||||
|   /// | ||||
|   /// This means, that there is at least one item in the queue and the playback | ||||
|   /// service is ready to play. | ||||
|   final active = BehaviorSubject.seeded(false); | ||||
| 
 | ||||
|   /// The current playlist. | ||||
|   /// | ||||
|   /// If the player is not active, this will be an empty list. | ||||
|   final playlist = BehaviorSubject.seeded(<InternalTrack>[]); | ||||
| 
 | ||||
|   /// Index of the currently played (or paused) track within the playlist. | ||||
|   /// | ||||
|   /// This will be zero, if the player is not active! | ||||
|   final currentIndex = BehaviorSubject.seeded(0); | ||||
| 
 | ||||
|   /// The currently played track. | ||||
|   /// | ||||
|   /// This will be null, if there is no  current track. | ||||
|   final currentTrack = BehaviorSubject<InternalTrack>.seeded(null); | ||||
| 
 | ||||
|   /// Whether we are currently playing or not. | ||||
|   /// | ||||
|   /// This will be false, if the player is not active. | ||||
|   final playing = BehaviorSubject.seeded(false); | ||||
| 
 | ||||
|   /// Current playback position. | ||||
|   /// | ||||
|   /// If the player is not active, this will default to zero. | ||||
|   final position = BehaviorSubject.seeded(const Duration()); | ||||
| 
 | ||||
|   /// Duration of the current track. | ||||
|   /// | ||||
|   /// If the player is not active, the duration will default to 1 s. | ||||
|   final duration = BehaviorSubject.seeded(const Duration(seconds: 1)); | ||||
| 
 | ||||
|   /// Playback position normalized to the range from zero to one. | ||||
|   final normalizedPosition = BehaviorSubject.seeded(0.0); | ||||
| 
 | ||||
| class Playback extends MusicusPlayback { | ||||
|   StreamSubscription _playbackServiceStateSubscription; | ||||
| 
 | ||||
|   /// Set everything to its default because the playback service was stopped. | ||||
|   void _stop() { | ||||
|     active.add(false); | ||||
|     playlist.add([]); | ||||
|     currentIndex.add(0); | ||||
|     playing.add(false); | ||||
|     position.add(const Duration()); | ||||
|     duration.add(const Duration(seconds: 1)); | ||||
|     normalizedPosition.add(0.0); | ||||
|   } | ||||
| 
 | ||||
|   /// Start playback service. | ||||
|   Future<void> start() async { | ||||
|   Future<void> _start() async { | ||||
|     if (!AudioService.running) { | ||||
|       await AudioService.start( | ||||
|         backgroundTaskEntrypoint: _playbackServiceEntrypoint, | ||||
|  | @ -109,8 +57,8 @@ class Player { | |||
|     currentTrack.add(playlist.value[index]); | ||||
|   } | ||||
| 
 | ||||
|   /// Connect listeners and initialize streams. | ||||
|   void setup() { | ||||
|   @override | ||||
|   Future<void> setup() async { | ||||
|     if (_playbackServiceStateSubscription != null) { | ||||
|       _playbackServiceStateSubscription.cancel(); | ||||
|     } | ||||
|  | @ -125,8 +73,12 @@ class Player { | |||
|     ).listen((msg) { | ||||
|       // If state is null, the background audio service has stopped. | ||||
|       if (msg == null) { | ||||
|         _stop(); | ||||
|         dispose(); | ||||
|       } else { | ||||
|         if (!active.value) { | ||||
|           active.add(true); | ||||
|         } | ||||
| 
 | ||||
|         if (msg is _StatusMessage) { | ||||
|           playing.add(msg.playing); | ||||
|         } else if (msg is _PositionMessage) { | ||||
|  | @ -154,9 +106,23 @@ class Player { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Toggle whether the player is playing or paused. | ||||
|   /// | ||||
|   /// If the player is not active, this will do nothing. | ||||
|   @override | ||||
|   Future<void> addTracks(List<InternalTrack> tracks) async { | ||||
|     if (!AudioService.running) { | ||||
|       await _start(); | ||||
|     } | ||||
| 
 | ||||
|     await AudioService.customAction('addTracks', jsonEncode(tracks)); | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Future<void> removeTrack(int index) async { | ||||
|     if (AudioService.running) { | ||||
|       await AudioService.customAction('removeTrack', index); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Future<void> playPause() async { | ||||
|     if (active.value) { | ||||
|       if (playing.value) { | ||||
|  | @ -167,29 +133,7 @@ class Player { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Add a list of tracks to the players playlist. | ||||
|   Future<void> addTracks(List<InternalTrack> tracks) async { | ||||
|     if (!AudioService.running) { | ||||
|       await start(); | ||||
|     } | ||||
| 
 | ||||
|     await AudioService.customAction('addTracks', jsonEncode(tracks)); | ||||
|   } | ||||
| 
 | ||||
|   /// Remove the track at [index] from the playlist. | ||||
|   /// | ||||
|   /// If the player is not active or an invalid value is provided, this will do | ||||
|   /// nothing. | ||||
|   Future<void> removeTrack(int index) async { | ||||
|     if (AudioService.running) { | ||||
|       await AudioService.customAction('removeTrack', index); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Seek to [pos], which is a value between (and including) zero and one. | ||||
|   /// | ||||
|   /// If the player is not active or an invalid value is provided, this will do | ||||
|   /// nothing. | ||||
|   @override | ||||
|   Future<void> seekTo(double pos) async { | ||||
|     if (active.value && pos >= 0.0 && pos <= 1.0) { | ||||
|       final durationMs = duration.value.inMilliseconds; | ||||
|  | @ -197,45 +141,31 @@ class Player { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Play the previous track in the playlist. | ||||
|   /// | ||||
|   /// If the player is not active or there is no previous track, this will do | ||||
|   /// nothing. | ||||
|   Future<void> skipToNext() async { | ||||
|     if (AudioService.running) { | ||||
|       await AudioService.skipToNext(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Skip to the next track in the playlist. | ||||
|   /// | ||||
|   /// If the player is not active or there is no next track, this will do | ||||
|   /// nothing. If more than five seconds of the current track have been played, | ||||
|   /// this will go back to its beginning instead. | ||||
|   @override | ||||
|   Future<void> skipToPrevious() async { | ||||
|     if (AudioService.running) { | ||||
|       await AudioService.skipToPrevious(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Switch to the track with the index [index] in the playlist. | ||||
|   @override | ||||
|   Future<void> skipToNext() async { | ||||
|     if (AudioService.running) { | ||||
|       await AudioService.skipToNext(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Future<void> skipTo(int index) async { | ||||
|     if (AudioService.running) { | ||||
|       await AudioService.customAction('skipTo', index); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /// Tidy up. | ||||
|   @override | ||||
|   void dispose() { | ||||
|     super.dispose(); | ||||
|     _playbackServiceStateSubscription.cancel(); | ||||
|     active.close(); | ||||
|     playlist.close(); | ||||
|     currentIndex.close(); | ||||
|     currentTrack.close(); | ||||
|     playing.close(); | ||||
|     position.close(); | ||||
|     duration.close(); | ||||
|     normalizedPosition.close(); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|  | @ -363,7 +293,7 @@ class _PlaybackService extends BackgroundAudioTask { | |||
| 
 | ||||
|   /// Initialize database. | ||||
|   Future<void> _load() async { | ||||
|     final moorPort = IsolateNameServer.lookupPortByName('moorPort'); | ||||
|     final moorPort = IsolateNameServer.lookupPortByName('moor'); | ||||
|     final moorIsolate = MoorIsolate.fromConnectPort(moorPort); | ||||
|     db = Database.connect(await moorIsolate.connect()); | ||||
|     _loading.complete(); | ||||
|  | @ -397,7 +327,7 @@ class _PlaybackService extends BackgroundAudioTask { | |||
|       final title = workInfo.work.title; | ||||
| 
 | ||||
|       AudioServiceBackground.setMediaItem(MediaItem( | ||||
|         id: track.uri, | ||||
|         id: track.identifier, | ||||
|         album: composers, | ||||
|         title: title, | ||||
|       )); | ||||
|  | @ -456,7 +386,7 @@ class _PlaybackService extends BackgroundAudioTask { | |||
|   /// Set the current track, update the player and notify the system. | ||||
|   Future<void> _setCurrentTrack(int index) async { | ||||
|     _currentTrack = index; | ||||
|     _durationMs = await _player.setUri(_playlist[_currentTrack].uri); | ||||
|     _durationMs = await _player.setUri(_playlist[_currentTrack].identifier); | ||||
|     _setState(); | ||||
|   } | ||||
| 
 | ||||
|  | @ -508,7 +438,7 @@ class _PlaybackService extends BackgroundAudioTask { | |||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   void onCustomAction(String name, dynamic arguments) { | ||||
|   Future<void> onCustomAction(String name, dynamic arguments) async { | ||||
|     super.onCustomAction(name, arguments); | ||||
| 
 | ||||
|     // addTracks expects a List<Map<String, dynamic>> as its argument. | ||||
|  | @ -1,10 +1,8 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../editors/tracks.dart'; | ||||
| import '../icons.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| import 'person.dart'; | ||||
| import 'settings.dart'; | ||||
|  | @ -19,7 +17,7 @@ class _HomeScreenState extends State<HomeScreen> { | |||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     final backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|  |  | |||
|  | @ -1,10 +1,7 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../editors/person.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| import 'work.dart'; | ||||
| 
 | ||||
| class PersonScreen extends StatefulWidget { | ||||
|  | @ -23,7 +20,7 @@ class _PersonScreenState extends State<PersonScreen> { | |||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     final backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|  |  | |||
|  | @ -1,12 +1,10 @@ | |||
| import 'dart:async'; | ||||
| 
 | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../music_library.dart'; | ||||
| import '../widgets/play_pause_button.dart'; | ||||
| import '../widgets/recording_tile.dart'; | ||||
| 
 | ||||
| class ProgramScreen extends StatefulWidget { | ||||
|   @override | ||||
|  | @ -14,7 +12,7 @@ class ProgramScreen extends StatefulWidget { | |||
| } | ||||
| 
 | ||||
| class _ProgramScreenState extends State<ProgramScreen> { | ||||
|   BackendState backend; | ||||
|   MusicusBackendState backend; | ||||
| 
 | ||||
|   StreamSubscription<bool> playerActiveSubscription; | ||||
| 
 | ||||
|  | @ -29,14 +27,14 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|   void didChangeDependencies() { | ||||
|     super.didChangeDependencies(); | ||||
| 
 | ||||
|     backend = Backend.of(context); | ||||
|     backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     if (playerActiveSubscription != null) { | ||||
|       playerActiveSubscription.cancel(); | ||||
|     } | ||||
| 
 | ||||
|     // Close the program screen, if the player is no longer active. | ||||
|     playerActiveSubscription = backend.player.active.listen((active) { | ||||
|     playerActiveSubscription = backend.playback.active.listen((active) { | ||||
|       if (!active) { | ||||
|         Navigator.pop(context); | ||||
|       } | ||||
|  | @ -46,7 +44,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|       playlistSubscription.cancel(); | ||||
|     } | ||||
| 
 | ||||
|     playlistSubscription = backend.player.playlist.listen((playlist) { | ||||
|     playlistSubscription = backend.playback.playlist.listen((playlist) { | ||||
|       updateProgram(playlist); | ||||
|     }); | ||||
| 
 | ||||
|  | @ -54,7 +52,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|       positionSubscription.cancel(); | ||||
|     } | ||||
| 
 | ||||
|     positionSubscription = backend.player.normalizedPosition.listen((pos) { | ||||
|     positionSubscription = backend.playback.normalizedPosition.listen((pos) { | ||||
|       if (!seeking) { | ||||
|         setState(() { | ||||
|           position = pos; | ||||
|  | @ -154,7 +152,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|         title: Text('Program'), | ||||
|       ), | ||||
|       body: StreamBuilder<int>( | ||||
|         stream: backend.player.currentIndex, | ||||
|         stream: backend.playback.currentIndex, | ||||
|         builder: (context, snapshot) { | ||||
|           if (snapshot.hasData) { | ||||
|             return ListView.builder( | ||||
|  | @ -181,7 +179,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|                     ], | ||||
|                   ), | ||||
|                   onTap: () { | ||||
|                     backend.player.skipTo(index); | ||||
|                     backend.playback.skipTo(index); | ||||
|                   }, | ||||
|                   onLongPress: () { | ||||
|                     showDialog( | ||||
|  | @ -192,7 +190,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|                             ListTile( | ||||
|                               title: Text('Remove from playlist'), | ||||
|                               onTap: () { | ||||
|                                 backend.player.removeTrack(index); | ||||
|                                 backend.playback.removeTrack(index); | ||||
|                                 Navigator.pop(context); | ||||
|                               }, | ||||
|                             ), | ||||
|  | @ -220,7 +218,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|               }, | ||||
|               onChangeEnd: (pos) { | ||||
|                 seeking = false; | ||||
|                 backend.player.seekTo(pos); | ||||
|                 backend.playback.seekTo(pos); | ||||
|               }, | ||||
|               onChanged: (pos) { | ||||
|                 setState(() { | ||||
|  | @ -233,7 +231,7 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|                 Padding( | ||||
|                   padding: const EdgeInsets.only(left: 24.0), | ||||
|                   child: StreamBuilder<Duration>( | ||||
|                     stream: backend.player.position, | ||||
|                     stream: backend.playback.position, | ||||
|                     builder: (context, snapshot) { | ||||
|                       if (snapshot.hasData) { | ||||
|                         return DurationText(snapshot.data); | ||||
|  | @ -247,21 +245,21 @@ class _ProgramScreenState extends State<ProgramScreen> { | |||
|                 IconButton( | ||||
|                   icon: const Icon(Icons.skip_previous), | ||||
|                   onPressed: () { | ||||
|                     backend.player.skipToPrevious(); | ||||
|                     backend.playback.skipToPrevious(); | ||||
|                   }, | ||||
|                 ), | ||||
|                 PlayPauseButton(), | ||||
|                 IconButton( | ||||
|                   icon: const Icon(Icons.skip_next), | ||||
|                   onPressed: () { | ||||
|                     backend.player.skipToNext(); | ||||
|                     backend.playback.skipToNext(); | ||||
|                   }, | ||||
|                 ), | ||||
|                 Spacer(), | ||||
|                 Padding( | ||||
|                   padding: const EdgeInsets.only(right: 20.0), | ||||
|                   child: StreamBuilder<Duration>( | ||||
|                     stream: backend.player.duration, | ||||
|                     stream: backend.playback.duration, | ||||
|                     builder: (context, snapshot) { | ||||
|                       if (snapshot.hasData) { | ||||
|                         return DurationText(snapshot.data); | ||||
|  |  | |||
|  | @ -1,9 +1,7 @@ | |||
| import 'dart:async'; | ||||
| 
 | ||||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../settings.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| 
 | ||||
| class ServerSettingsScreen extends StatefulWidget { | ||||
|   @override | ||||
|  | @ -13,16 +11,16 @@ class ServerSettingsScreen extends StatefulWidget { | |||
| class _ServerSettingsScreenState extends State<ServerSettingsScreen> { | ||||
|   final hostController = TextEditingController(); | ||||
|   final portController = TextEditingController(); | ||||
|   final basePathController = TextEditingController(); | ||||
|   final apiPathController = TextEditingController(); | ||||
| 
 | ||||
|   BackendState backend; | ||||
|   StreamSubscription<ServerSettings> serverSubscription; | ||||
|   MusicusBackendState backend; | ||||
|   StreamSubscription<MusicusServerSettings> serverSubscription; | ||||
| 
 | ||||
|   @override | ||||
|   void didChangeDependencies() { | ||||
|     super.didChangeDependencies(); | ||||
| 
 | ||||
|     backend = Backend.of(context); | ||||
|     backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     if (serverSubscription != null) { | ||||
|       serverSubscription.cancel(); | ||||
|  | @ -34,10 +32,10 @@ class _ServerSettingsScreenState extends State<ServerSettingsScreen> { | |||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   void _settingsChanged(ServerSettings settings) { | ||||
|   void _settingsChanged(MusicusServerSettings settings) { | ||||
|     hostController.text = settings.host; | ||||
|     portController.text = settings.port.toString(); | ||||
|     basePathController.text = settings.basePath; | ||||
|     apiPathController.text = settings.apiPath; | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|  | @ -50,15 +48,15 @@ class _ServerSettingsScreenState extends State<ServerSettingsScreen> { | |||
|             icon: const Icon(Icons.restore), | ||||
|             tooltip: 'Reset to default', | ||||
|             onPressed: () { | ||||
|               backend.settings.resetServerSettings(); | ||||
|               backend.settings.resetServer(); | ||||
|             }, | ||||
|           ), | ||||
|           FlatButton( | ||||
|             onPressed: () async { | ||||
|               await backend.settings.setServerSettings(ServerSettings( | ||||
|               await backend.settings.setServer(MusicusServerSettings( | ||||
|                 host: hostController.text, | ||||
|                 port: int.parse(portController.text), | ||||
|                 basePath: basePathController.text, | ||||
|                 apiPath: apiPathController.text, | ||||
|               )); | ||||
| 
 | ||||
|               Navigator.pop(context); | ||||
|  | @ -91,7 +89,7 @@ class _ServerSettingsScreenState extends State<ServerSettingsScreen> { | |||
|           Padding( | ||||
|             padding: const EdgeInsets.all(16.0), | ||||
|             child: TextField( | ||||
|               controller: basePathController, | ||||
|               controller: apiPathController, | ||||
|               decoration: InputDecoration( | ||||
|                 labelText: 'API path', | ||||
|               ), | ||||
|  |  | |||
|  | @ -1,14 +1,15 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../settings.dart'; | ||||
| import 'package:flutter/services.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| 
 | ||||
| import 'server_settings.dart'; | ||||
| 
 | ||||
| class SettingsScreen extends StatelessWidget { | ||||
|   static const _platform = MethodChannel('de.johrpan.musicus/platform'); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     final backend = MusicusBackend.of(context); | ||||
|     final settings = backend.settings; | ||||
| 
 | ||||
|     return Scaffold( | ||||
|  | @ -18,18 +19,23 @@ class SettingsScreen extends StatelessWidget { | |||
|       body: ListView( | ||||
|         children: <Widget>[ | ||||
|           StreamBuilder<String>( | ||||
|               stream: settings.musicLibraryUri, | ||||
|               stream: settings.musicLibraryPath, | ||||
|               builder: (context, snapshot) { | ||||
|                 return ListTile( | ||||
|                   title: Text('Music library path'), | ||||
|                   subtitle: Text(snapshot.data ?? 'Choose folder'), | ||||
|                   isThreeLine: snapshot.hasData, | ||||
|                   onTap: () { | ||||
|                     settings.chooseMusicLibraryUri(); | ||||
|                   onTap: () async { | ||||
|                     final uri = | ||||
|                         await _platform.invokeMethod<String>('openTree'); | ||||
| 
 | ||||
|                     if (uri != null) { | ||||
|                       settings.setMusicLibraryPath(uri); | ||||
|                     } | ||||
|                   }, | ||||
|                 ); | ||||
|               }), | ||||
|           StreamBuilder<ServerSettings>( | ||||
|           StreamBuilder<MusicusServerSettings>( | ||||
|               stream: settings.server, | ||||
|               builder: (context, snapshot) { | ||||
|                 final s = snapshot.data; | ||||
|  | @ -37,10 +43,10 @@ class SettingsScreen extends StatelessWidget { | |||
|                 return ListTile( | ||||
|                   title: Text('Musicus server'), | ||||
|                   subtitle: Text( | ||||
|                       s != null ? '${s.host}:${s.port}${s.basePath}' : '...'), | ||||
|                       s != null ? '${s.host}:${s.port}${s.apiPath}' : '...'), | ||||
|                   trailing: const Icon(Icons.chevron_right), | ||||
|                   onTap: () async { | ||||
|                     final ServerSettings result = await Navigator.push( | ||||
|                     final MusicusServerSettings result = await Navigator.push( | ||||
|                       context, | ||||
|                       MaterialPageRoute( | ||||
|                         builder: (context) => ServerSettingsScreen(), | ||||
|  | @ -48,7 +54,7 @@ class SettingsScreen extends StatelessWidget { | |||
|                     ); | ||||
| 
 | ||||
|                     if (result != null) { | ||||
|                       settings.setServerSettings(result); | ||||
|                       settings.setServer(result); | ||||
|                     } | ||||
|                   }, | ||||
|                 ); | ||||
|  |  | |||
|  | @ -1,11 +1,7 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../editors/work.dart'; | ||||
| import '../widgets/texts.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| class WorkScreen extends StatelessWidget { | ||||
|   final WorkInfo workInfo; | ||||
| 
 | ||||
|  | @ -15,7 +11,7 @@ class WorkScreen extends StatelessWidget { | |||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     final backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|  | @ -46,11 +42,11 @@ class WorkScreen extends StatelessWidget { | |||
|             performanceInfos: recordingInfo.performances, | ||||
|           ), | ||||
|           onTap: () { | ||||
|             final tracks = backend.ml.tracks[recordingInfo.recording.id]; | ||||
|             final tracks = backend.library.tracks[recordingInfo.recording.id]; | ||||
|             tracks.sort((t1, t2) => t1.track.index.compareTo(t2.track.index)); | ||||
| 
 | ||||
|             backend.player | ||||
|                 .addTracks(backend.ml.tracks[recordingInfo.recording.id]); | ||||
|             backend.playback | ||||
|                 .addTracks(backend.library.tracks[recordingInfo.recording.id]); | ||||
|           }, | ||||
|         ), | ||||
|       ), | ||||
|  |  | |||
|  | @ -1,41 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../editors/ensemble.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| /// A screen to select an ensemble. | ||||
| /// | ||||
| /// If the user has selected one, it will be returned as an [Ensemble] object | ||||
| /// using the navigator. | ||||
| class EnsembleSelector extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select ensemble'), | ||||
|       ), | ||||
|       body: EnsemblesList( | ||||
|         onSelected: (ensemble) { | ||||
|           Navigator.pop(context, ensemble); | ||||
|         }, | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           final Ensemble ensemble = await Navigator.push( | ||||
|             context, | ||||
|             MaterialPageRoute( | ||||
|               builder: (context) => EnsembleEditor(), | ||||
|               fullscreenDialog: true, | ||||
|             ), | ||||
|           ); | ||||
| 
 | ||||
|           if (ensemble != null) { | ||||
|             Navigator.pop(context, ensemble); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,163 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../platform.dart'; | ||||
| 
 | ||||
| /// Result of the user's interaction with the files selector. | ||||
| /// | ||||
| /// This will be given back when popping the navigator. | ||||
| class FilesSelectorResult { | ||||
|   /// Document ID of the parent directory of the selected files. | ||||
|   /// | ||||
|   /// This will be null, if they are in the toplevel directory. | ||||
|   final String parentId; | ||||
| 
 | ||||
|   /// Selected files. | ||||
|   final Set<Document> selection; | ||||
| 
 | ||||
|   FilesSelectorResult(this.parentId, this.selection); | ||||
| } | ||||
| 
 | ||||
| class FilesSelector extends StatefulWidget { | ||||
|   @override | ||||
|   _FilesSelectorState createState() => _FilesSelectorState(); | ||||
| } | ||||
| 
 | ||||
| class _FilesSelectorState extends State<FilesSelector> { | ||||
|   BackendState backend; | ||||
|   List<Document> history = []; | ||||
|   List<Document> children = []; | ||||
|   Set<Document> selection = {}; | ||||
| 
 | ||||
|   @override | ||||
|   void didChangeDependencies() { | ||||
|     super.didChangeDependencies(); | ||||
| 
 | ||||
|     backend = Backend.of(context); | ||||
|     loadChildren(); | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return WillPopScope( | ||||
|       child: Scaffold( | ||||
|         appBar: AppBar( | ||||
|           title: Text('Choose files'), | ||||
|           leading: IconButton( | ||||
|             icon: const Icon(Icons.close), | ||||
|             onPressed: () { | ||||
|               Navigator.pop(context); | ||||
|             }, | ||||
|           ), | ||||
|           actions: <Widget>[ | ||||
|             FlatButton( | ||||
|               child: Text('DONE'), | ||||
|               onPressed: () { | ||||
|                 Navigator.pop( | ||||
|                   context, | ||||
|                   FilesSelectorResult( | ||||
|                     history.isNotEmpty ? history.last.id : null, | ||||
|                     selection, | ||||
|                   ), | ||||
|                 ); | ||||
|               }, | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|         body: Column( | ||||
|           children: <Widget>[ | ||||
|             Material( | ||||
|               elevation: 2.0, | ||||
|               child: ListTile( | ||||
|                 leading: IconButton( | ||||
|                   icon: const Icon(Icons.arrow_upward), | ||||
|                   onPressed: history.isNotEmpty ? up : null, | ||||
|                 ), | ||||
|                 title: Text( | ||||
|                     history.isNotEmpty ? history.last.name : 'Music library'), | ||||
|               ), | ||||
|             ), | ||||
|             Expanded( | ||||
|               child: ListView.builder( | ||||
|                 itemCount: children.length, | ||||
|                 itemBuilder: (context, index) { | ||||
|                   final document = children[index]; | ||||
| 
 | ||||
|                   if (document.isDirectory) { | ||||
|                     return ListTile( | ||||
|                       leading: const Icon(Icons.folder), | ||||
|                       title: Text(document.name), | ||||
|                       onTap: () { | ||||
|                         setState(() { | ||||
|                           history.add(document); | ||||
|                         }); | ||||
|                         loadChildren(); | ||||
|                       }, | ||||
|                     ); | ||||
|                   } else { | ||||
|                     return CheckboxListTile( | ||||
|                       controlAffinity: ListTileControlAffinity.trailing, | ||||
|                       secondary: const Icon(Icons.insert_drive_file), | ||||
|                       title: Text(document.name), | ||||
|                       value: selection.contains(document), | ||||
|                       onChanged: (selected) { | ||||
|                         setState(() { | ||||
|                           if (selected) { | ||||
|                             selection.add(document); | ||||
|                           } else { | ||||
|                             selection.remove(document); | ||||
|                           } | ||||
|                         }); | ||||
|                       }, | ||||
|                     ); | ||||
|                   } | ||||
|                 }, | ||||
|               ), | ||||
|             ), | ||||
|           ], | ||||
|         ), | ||||
|       ), | ||||
|       onWillPop: () => Future.value(up()), | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|   Future<void> loadChildren() async { | ||||
|     setState(() { | ||||
|       children = []; | ||||
| 
 | ||||
|       // We reset the selection here, because the user should not be able to | ||||
|       // select files from multiple directories for now. | ||||
|       selection = {}; | ||||
|     }); | ||||
| 
 | ||||
|     final newChildren = await Platform.getChildren( | ||||
|         backend.settings.musicLibraryUri.value, | ||||
|         history.isNotEmpty ? history.last.id : null); | ||||
| 
 | ||||
|     newChildren.sort((d1, d2) { | ||||
|       if (d1.isDirectory != d2.isDirectory) { | ||||
|         return d1.isDirectory ? -1 : 1; | ||||
|       } else { | ||||
|         return d1.name.compareTo(d2.name); | ||||
|       } | ||||
|     }); | ||||
| 
 | ||||
|     setState(() { | ||||
|       children = newChildren; | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   bool up() { | ||||
|     if (history.isNotEmpty) { | ||||
|       setState(() { | ||||
|         history.removeLast(); | ||||
|       }); | ||||
| 
 | ||||
|       loadChildren(); | ||||
| 
 | ||||
|       return false; | ||||
|     } else { | ||||
|       return true; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | @ -1,136 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../editors/instrument.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| class InstrumentsSelector extends StatefulWidget { | ||||
|   final bool multiple; | ||||
|   final List<Instrument> selection; | ||||
| 
 | ||||
|   InstrumentsSelector({ | ||||
|     this.multiple = false, | ||||
|     this.selection, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _InstrumentsSelectorState createState() => _InstrumentsSelectorState(); | ||||
| } | ||||
| 
 | ||||
| class _InstrumentsSelectorState extends State<InstrumentsSelector> { | ||||
|   final _list = GlobalKey<PagedListViewState<Instrument>>(); | ||||
|    | ||||
|   Set<Instrument> selection = {}; | ||||
|   String _search; | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     if (widget.selection != null) { | ||||
|       selection = widget.selection.toSet(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text(widget.multiple | ||||
|             ? 'Select instruments/roles' | ||||
|             : 'Select instrument/role'), | ||||
|         actions: widget.multiple | ||||
|             ? <Widget>[ | ||||
|                 FlatButton( | ||||
|                   child: Text('DONE'), | ||||
|                   onPressed: () => Navigator.pop(context, selection.toList()), | ||||
|                 ), | ||||
|               ] | ||||
|             : null, | ||||
|       ), | ||||
|       body: Column( | ||||
|         children: <Widget>[ | ||||
|           Material( | ||||
|             elevation: 2.0, | ||||
|             child: Padding( | ||||
|               padding: const EdgeInsets.symmetric( | ||||
|                 horizontal: 16.0, | ||||
|                 vertical: 4.0, | ||||
|               ), | ||||
|               child: TextField( | ||||
|                 autofocus: true, | ||||
|                 onChanged: (text) { | ||||
|                   setState(() { | ||||
|                     _search = text; | ||||
|                   }); | ||||
|                 }, | ||||
|                 decoration: InputDecoration.collapsed( | ||||
|                   hintText: 'Search by name...', | ||||
|                 ), | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|           Expanded( | ||||
|             child: PagedListView<Instrument>( | ||||
|               key: _list, | ||||
|               search: _search, | ||||
|               fetch: (page, search) async { | ||||
|                 return await backend.client.getInstruments(page, search); | ||||
|               }, | ||||
|               builder: (context, instrument) { | ||||
|                 if (widget.multiple) { | ||||
|                   return CheckboxListTile( | ||||
|                     title: Text(instrument.name), | ||||
|                     value: selection.contains(instrument), | ||||
|                     checkColor: Colors.black, | ||||
|                     onChanged: (selected) { | ||||
|                       setState(() { | ||||
|                         if (selected) { | ||||
|                           selection.add(instrument); | ||||
|                         } else { | ||||
|                           selection.remove(instrument); | ||||
|                         } | ||||
|                       }); | ||||
|                     }, | ||||
|                   ); | ||||
|                 } else { | ||||
|                   return ListTile( | ||||
|                     title: Text(instrument.name), | ||||
|                     onTap: () => Navigator.pop(context, instrument), | ||||
|                   ); | ||||
|                 } | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           final Instrument instrument = await Navigator.push( | ||||
|               context, | ||||
|               MaterialPageRoute( | ||||
|                 builder: (context) => InstrumentEditor(), | ||||
|                 fullscreenDialog: true, | ||||
|               )); | ||||
| 
 | ||||
|           if (instrument != null) { | ||||
|             if (widget.multiple) { | ||||
|               setState(() { | ||||
|                 selection.add(instrument); | ||||
|               }); | ||||
| 
 | ||||
|               // We need to rebuild the list view, because we added an item. | ||||
|               _list.currentState.update(); | ||||
|             } else { | ||||
|               Navigator.pop(context, instrument); | ||||
|             } | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,41 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../editors/person.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| /// A screen to select a person. | ||||
| /// | ||||
| /// If the user has selected a person, it will be returned as a [Person] object | ||||
| /// using the navigator. | ||||
| class PersonsSelector extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select person'), | ||||
|       ), | ||||
|       body: PersonsList( | ||||
|         onSelected: (person) { | ||||
|           Navigator.pop(context, person); | ||||
|         }, | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           final Person person = await Navigator.push( | ||||
|             context, | ||||
|             MaterialPageRoute( | ||||
|               builder: (context) => PersonEditor(), | ||||
|               fullscreenDialog: true, | ||||
|             ), | ||||
|           ); | ||||
| 
 | ||||
|           if (person != null) { | ||||
|             Navigator.pop(context, person); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,90 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../editors/recording.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| class RecordingSelectorResult { | ||||
|   final WorkInfo workInfo; | ||||
|   final RecordingInfo recordingInfo; | ||||
| 
 | ||||
|   RecordingSelectorResult({ | ||||
|     this.workInfo, | ||||
|     this.recordingInfo, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| /// A screen to select a recording. | ||||
| /// | ||||
| /// If the user has selected a recording, a [RecordingSelectorResult] containing | ||||
| /// the selected recording and the recorded work will be returned using the | ||||
| /// navigator. | ||||
| class RecordingSelector extends StatefulWidget { | ||||
|   @override | ||||
|   _RecordingSelectorState createState() => _RecordingSelectorState(); | ||||
| } | ||||
| 
 | ||||
| class _RecordingSelectorState extends State<RecordingSelector> { | ||||
|   Person person; | ||||
|   WorkInfo workInfo; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     Widget body; | ||||
| 
 | ||||
|     if (person == null) { | ||||
|       body = PersonsList( | ||||
|         onSelected: (newPerson) { | ||||
|           setState(() { | ||||
|             person = newPerson; | ||||
|           }); | ||||
|         }, | ||||
|       ); | ||||
|     } else if (workInfo == null) { | ||||
|       body = WorksList( | ||||
|         personId: person.id, | ||||
|         onSelected: (newWorkInfo) { | ||||
|           setState(() { | ||||
|             workInfo = newWorkInfo; | ||||
|           }); | ||||
|         }, | ||||
|       ); | ||||
|     } else { | ||||
|       body = RecordingsList( | ||||
|         workId: workInfo.work.id, | ||||
|         onSelected: (recordingInfo) { | ||||
|           Navigator.pop( | ||||
|             context, | ||||
|             RecordingSelectorResult( | ||||
|               workInfo: workInfo, | ||||
|               recordingInfo: recordingInfo, | ||||
|             ), | ||||
|           ); | ||||
|         }, | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select recording'), | ||||
|       ), | ||||
|       body: body, | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           final RecordingSelectorResult result = await Navigator.push( | ||||
|             context, | ||||
|             MaterialPageRoute( | ||||
|               builder: (context) => RecordingEditor(), | ||||
|               fullscreenDialog: true, | ||||
|             ), | ||||
|           ); | ||||
| 
 | ||||
|           if (result != null) { | ||||
|             Navigator.pop(context, result); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,65 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../editors/work.dart'; | ||||
| import '../widgets/lists.dart'; | ||||
| 
 | ||||
| /// A screen to select a work. | ||||
| /// | ||||
| /// If the user has selected a work, a [WorkInfo] will be returned | ||||
| /// using the navigator. | ||||
| class WorkSelector extends StatefulWidget { | ||||
|   @override | ||||
|   _WorkSelectorState createState() => _WorkSelectorState(); | ||||
| } | ||||
| 
 | ||||
| class _WorkSelectorState extends State<WorkSelector> { | ||||
|   Person person; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     Widget body; | ||||
| 
 | ||||
|     if (person == null) { | ||||
|       body = PersonsList( | ||||
|         onSelected: (newPerson) { | ||||
|           setState(() { | ||||
|             person = newPerson; | ||||
|           }); | ||||
|         }, | ||||
|       ); | ||||
|     } else { | ||||
|       body = WorksList( | ||||
|         personId: person.id, | ||||
|         onSelected: (workInfo) { | ||||
|           setState(() { | ||||
|             Navigator.pop(context, workInfo); | ||||
|           }); | ||||
|         }, | ||||
|       ); | ||||
|     } | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select work'), | ||||
|       ), | ||||
|       body: body, | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           final WorkInfo workInfo = await Navigator.push( | ||||
|             context, | ||||
|             MaterialPageRoute( | ||||
|               builder: (context) => WorkEditor(), | ||||
|               fullscreenDialog: true, | ||||
|             ), | ||||
|           ); | ||||
| 
 | ||||
|           if (workInfo != null) { | ||||
|             Navigator.pop(context, workInfo); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,102 +1,30 @@ | |||
| import 'package:flutter/services.dart'; | ||||
| import 'package:meta/meta.dart'; | ||||
| import 'package:rxdart/rxdart.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:shared_preferences/shared_preferences.dart'; | ||||
| 
 | ||||
| /// Settings concerning the Musicus server to connect to. | ||||
| /// | ||||
| /// We don't support setting a scheme here, because there may be password being | ||||
| /// submitted in the future, so we default to HTTPS. | ||||
| class ServerSettings { | ||||
|   static const defaultHost = 'musicus.johrpan.de'; | ||||
|   static const defaultPort = 1833; | ||||
|   static const defaultBasePath = '/api'; | ||||
| class SettingsStorage extends MusicusSettingsStorage { | ||||
|   SharedPreferences _pref; | ||||
| 
 | ||||
|   /// Host to connect to, e.g. 'musicus.johrpan.de'; | ||||
|   final String host; | ||||
| 
 | ||||
|   /// Port to connect to. | ||||
|   final int port; | ||||
| 
 | ||||
|   /// Path to the API. | ||||
|   /// | ||||
|   /// This should be null, if the API is at the root of the host. | ||||
|   final String basePath; | ||||
| 
 | ||||
|   ServerSettings({ | ||||
|     @required this.host, | ||||
|     @required this.port, | ||||
|     @required this.basePath, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| /// Manager for all settings that are persisted. | ||||
| class Settings { | ||||
|   static const defaultHost = 'musicus.johrpan.de'; | ||||
|   static const defaultPort = 443; | ||||
|   static const defaultBasePath = '/api'; | ||||
| 
 | ||||
|   static const _platform = MethodChannel('de.johrpan.musicus/platform'); | ||||
| 
 | ||||
|   /// The tree storage access framework tree URI of the music library. | ||||
|   final musicLibraryUri = BehaviorSubject<String>(); | ||||
| 
 | ||||
|   /// Musicus server to connect to. | ||||
|   final server = BehaviorSubject<ServerSettings>(); | ||||
| 
 | ||||
|   SharedPreferences _shPref; | ||||
| 
 | ||||
|   /// Initialize the settings. | ||||
|   Future<void> load() async { | ||||
|     _shPref = await SharedPreferences.getInstance(); | ||||
| 
 | ||||
|     final uri = _shPref.getString('musicLibraryUri'); | ||||
|     if (uri != null) { | ||||
|       musicLibraryUri.add(uri); | ||||
|     } | ||||
| 
 | ||||
|     final host = _shPref.getString('serverHost') ?? defaultHost; | ||||
|     final port = _shPref.getInt('serverPort') ?? defaultPort; | ||||
|     final basePath = _shPref.getString('serverBasePath') ?? defaultBasePath; | ||||
| 
 | ||||
|     server.add(ServerSettings( | ||||
|       host: host, | ||||
|       port: port, | ||||
|       basePath: basePath, | ||||
|     )); | ||||
|     _pref = await SharedPreferences.getInstance(); | ||||
|   } | ||||
| 
 | ||||
|   /// Open the system picker to select a new music library URI. | ||||
|   Future<void> chooseMusicLibraryUri() async { | ||||
|     final uri = await _platform.invokeMethod<String>('openTree'); | ||||
| 
 | ||||
|     if (uri != null) { | ||||
|       musicLibraryUri.add(uri); | ||||
|       await _shPref.setString('musicLibraryUri', uri); | ||||
|     } | ||||
|   @override | ||||
|   Future<int> getInt(String key) { | ||||
|     return Future.value(_pref.getInt(key)); | ||||
|   } | ||||
| 
 | ||||
|   /// Change the Musicus server settings. | ||||
|   Future<void> setServerSettings(ServerSettings settings) async { | ||||
|     await _shPref.setString('serverHost', settings.host); | ||||
|     await _shPref.setInt('serverPort', settings.port); | ||||
|     await _shPref.setString('serverBasePath', settings.basePath); | ||||
| 
 | ||||
|     server.add(settings); | ||||
|   @override | ||||
|   Future<String> getString(String key) { | ||||
|     return Future.value(_pref.getString(key)); | ||||
|   } | ||||
| 
 | ||||
|   /// Reset the server settings to their defaults. | ||||
|   Future<void> resetServerSettings() async { | ||||
|     await setServerSettings(ServerSettings( | ||||
|       host: defaultHost, | ||||
|       port: defaultPort, | ||||
|       basePath: defaultBasePath, | ||||
|     )); | ||||
|   @override | ||||
|   Future<void> setInt(String key, int value) async { | ||||
|     await _pref.setInt(key, value); | ||||
|   } | ||||
| 
 | ||||
|   /// Tidy up. | ||||
|   void dispose() { | ||||
|     musicLibraryUri.close(); | ||||
|     server.close(); | ||||
|   @override | ||||
|   Future<void> setString(String key, String value) async { | ||||
|     await _pref.setString(key, value); | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -1,367 +0,0 @@ | |||
| import 'dart:async'; | ||||
| 
 | ||||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../widgets/texts.dart'; | ||||
| 
 | ||||
| /// A list view supporting pagination and searching. | ||||
| /// | ||||
| /// The [fetch] function will be called, when the user has scrolled to the end | ||||
| /// of the list. If you recreate this widget with a new [search] parameter, it | ||||
| /// will update, but it will NOT correctly react to a changed [fetch] method. | ||||
| /// You can call update() on the corresponding state object to manually refresh | ||||
| /// the contents. | ||||
| class PagedListView<T> extends StatefulWidget { | ||||
|   /// A search string. | ||||
|   /// | ||||
|   /// This will be provided when calling [fetch]. | ||||
|   final String search; | ||||
| 
 | ||||
|   /// Callback for fetching a page of entities. | ||||
|   /// | ||||
|   /// This has to tolerate abitrary high page numbers. | ||||
|   final Future<List<T>> Function(int page, String search) fetch; | ||||
| 
 | ||||
|   /// Build function to be called for each entity. | ||||
|   final Widget Function(BuildContext context, T entity) builder; | ||||
| 
 | ||||
|   PagedListView({ | ||||
|     Key key, | ||||
|     this.search, | ||||
|     @required this.fetch, | ||||
|     @required this.builder, | ||||
|   }) : super(key: key); | ||||
| 
 | ||||
|   @override | ||||
|   PagedListViewState<T> createState() => PagedListViewState<T>(); | ||||
| } | ||||
| 
 | ||||
| class PagedListViewState<T> extends State<PagedListView<T>> { | ||||
|   final _scrollController = ScrollController(); | ||||
|   final _entities = <T>[]; | ||||
| 
 | ||||
|   bool loading = true; | ||||
| 
 | ||||
|   /// The last parameters of _fetch(). | ||||
|   int _page; | ||||
|   String _search; | ||||
| 
 | ||||
|   /// Whether the last fetch() call returned no results. | ||||
|   bool _end = false; | ||||
| 
 | ||||
|   /// Fetch new entities. | ||||
|   /// | ||||
|   /// If the function was called again with other parameters, while it was | ||||
|   /// running, it will discard the result. | ||||
|   Future<void> _fetch(int page, String search) async { | ||||
|     if (page != _page || search != _search) { | ||||
|       _page = page; | ||||
|       _search = search; | ||||
| 
 | ||||
|       setState(() { | ||||
|         loading = true; | ||||
|       }); | ||||
| 
 | ||||
|       final newEntities = await widget.fetch(page, search); | ||||
| 
 | ||||
|       if (mounted && search == _search) { | ||||
|         setState(() { | ||||
|           if (newEntities.isNotEmpty) { | ||||
|             _entities.addAll(newEntities); | ||||
|           } else { | ||||
|             _end = true; | ||||
|           } | ||||
| 
 | ||||
|           loading = false; | ||||
|         }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   void initState() { | ||||
|     super.initState(); | ||||
| 
 | ||||
|     _scrollController.addListener(() { | ||||
|       if (_scrollController.position.pixels > | ||||
|               _scrollController.position.maxScrollExtent - 64.0 && | ||||
|           !loading && | ||||
|           !_end) { | ||||
|         _fetch(_page + 1, widget.search); | ||||
|       } | ||||
|     }); | ||||
| 
 | ||||
|     _fetch(0, widget.search); | ||||
|   } | ||||
| 
 | ||||
|   /// Update the content manually. | ||||
|   ///  | ||||
|   /// This will reset the current page to zero and call the provided fetch() | ||||
|   /// method. | ||||
|   void update() { | ||||
|     setState(() { | ||||
|       _entities.clear(); | ||||
|     }); | ||||
| 
 | ||||
|     _page = null; | ||||
|     _fetch(0, widget.search); | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   void didUpdateWidget(PagedListView<T> oldWidget) { | ||||
|     super.didUpdateWidget(oldWidget); | ||||
| 
 | ||||
|     if (oldWidget.search != widget.search) { | ||||
|       // We don't nedd to call setState() because the framework will always call | ||||
|       // build() after this. | ||||
|       _entities.clear(); | ||||
|       _page = null; | ||||
|       _fetch(0, widget.search); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return ListView.builder( | ||||
|       controller: _scrollController, | ||||
|       itemCount: _entities.length + 1, | ||||
|       itemBuilder: (context, index) { | ||||
|         if (index < _entities.length) { | ||||
|           return widget.builder(context, _entities[index]); | ||||
|         } else { | ||||
|           return SizedBox( | ||||
|             height: 64.0, | ||||
|             child: Center( | ||||
|               child: loading ? CircularProgressIndicator() : Container(), | ||||
|             ), | ||||
|           ); | ||||
|         } | ||||
|       }, | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// A list of persons. | ||||
| class PersonsList extends StatefulWidget { | ||||
|   /// Called, when the user has selected a person. | ||||
|   final void Function(Person person) onSelected; | ||||
| 
 | ||||
|   PersonsList({ | ||||
|     @required this.onSelected, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _PersonsListState createState() => _PersonsListState(); | ||||
| } | ||||
| 
 | ||||
| class _PersonsListState extends State<PersonsList> { | ||||
|   String _search; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Material( | ||||
|           elevation: 2.0, | ||||
|           child: Padding( | ||||
|             padding: const EdgeInsets.symmetric( | ||||
|               horizontal: 16.0, | ||||
|               vertical: 4.0, | ||||
|             ), | ||||
|             child: TextField( | ||||
|               autofocus: true, | ||||
|               onChanged: (text) { | ||||
|                 setState(() { | ||||
|                   _search = text; | ||||
|                 }); | ||||
|               }, | ||||
|               decoration: InputDecoration.collapsed( | ||||
|                 hintText: 'Search by last name...', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: PagedListView<Person>( | ||||
|             search: _search, | ||||
|             fetch: (page, search) async { | ||||
|               return await backend.client.getPersons(page, search); | ||||
|             }, | ||||
|             builder: (context, person) => ListTile( | ||||
|               title: Text('${person.lastName}, ${person.firstName}'), | ||||
|               onTap: () { | ||||
|                 widget.onSelected(person); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// A list of ensembles. | ||||
| class EnsemblesList extends StatefulWidget { | ||||
|   /// Called, when the user has selected an ensemble. | ||||
|   final void Function(Ensemble ensemble) onSelected; | ||||
| 
 | ||||
|   EnsemblesList({ | ||||
|     @required this.onSelected, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _EnsemblesListState createState() => _EnsemblesListState(); | ||||
| } | ||||
| 
 | ||||
| class _EnsemblesListState extends State<EnsemblesList> { | ||||
|   String _search; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Material( | ||||
|           elevation: 2.0, | ||||
|           child: Padding( | ||||
|             padding: const EdgeInsets.symmetric( | ||||
|               horizontal: 16.0, | ||||
|               vertical: 4.0, | ||||
|             ), | ||||
|             child: TextField( | ||||
|               autofocus: true, | ||||
|               onChanged: (text) { | ||||
|                 setState(() { | ||||
|                   _search = text; | ||||
|                 }); | ||||
|               }, | ||||
|               decoration: InputDecoration.collapsed( | ||||
|                 hintText: 'Search by name...', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: PagedListView<Ensemble>( | ||||
|             search: _search, | ||||
|             fetch: (page, search) async { | ||||
|               return await backend.client.getEnsembles(page, search); | ||||
|             }, | ||||
|             builder: (context, ensemble) => ListTile( | ||||
|               title: Text(ensemble.name), | ||||
|               onTap: () { | ||||
|                 widget.onSelected(ensemble); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// A list of works by one composer. | ||||
| class WorksList extends StatefulWidget { | ||||
|   /// The ID of the composer. | ||||
|   final int personId; | ||||
| 
 | ||||
|   /// Called, when the user has selected a work. | ||||
|   final void Function(WorkInfo workInfo) onSelected; | ||||
| 
 | ||||
|   WorksList({ | ||||
|     this.personId, | ||||
|     this.onSelected, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   _WorksListState createState() => _WorksListState(); | ||||
| } | ||||
| 
 | ||||
| class _WorksListState extends State<WorksList> { | ||||
|   String _search; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Material( | ||||
|           elevation: 2.0, | ||||
|           child: Padding( | ||||
|             padding: const EdgeInsets.symmetric( | ||||
|               horizontal: 16.0, | ||||
|               vertical: 4.0, | ||||
|             ), | ||||
|             child: TextField( | ||||
|               autofocus: true, | ||||
|               onChanged: (text) { | ||||
|                 setState(() { | ||||
|                   _search = text; | ||||
|                 }); | ||||
|               }, | ||||
|               decoration: InputDecoration.collapsed( | ||||
|                 hintText: 'Search by title...', | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: PagedListView<WorkInfo>( | ||||
|             search: _search, | ||||
|             fetch: (page, search) async { | ||||
|               return await backend.client | ||||
|                   .getWorks(widget.personId, page, search); | ||||
|             }, | ||||
|             builder: (context, workInfo) => ListTile( | ||||
|               title: Text(workInfo.work.title), | ||||
|               onTap: () { | ||||
|                 widget.onSelected(workInfo); | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /// A list of recordings of a work. | ||||
| class RecordingsList extends StatelessWidget { | ||||
|   /// The ID of the work. | ||||
|   final int workId; | ||||
| 
 | ||||
|   /// Called, when the user has selected a recording. | ||||
|   final void Function(RecordingInfo recordingInfo) onSelected; | ||||
| 
 | ||||
|   RecordingsList({ | ||||
|     this.workId, | ||||
|     this.onSelected, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return PagedListView<RecordingInfo>( | ||||
|       fetch: (page, _) async { | ||||
|         return await backend.client.getRecordings(workId, page); | ||||
|       }, | ||||
|       builder: (context, recordingInfo) => ListTile( | ||||
|         title: PerformancesText( | ||||
|           performanceInfos: recordingInfo.performances, | ||||
|         ), | ||||
|         onTap: () { | ||||
|           if (onSelected != null) { | ||||
|             onSelected(recordingInfo); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,8 +1,7 @@ | |||
| import 'dart:async'; | ||||
| 
 | ||||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| 
 | ||||
| class PlayPauseButton extends StatefulWidget { | ||||
|   @override | ||||
|  | @ -12,7 +11,7 @@ class PlayPauseButton extends StatefulWidget { | |||
| class _PlayPauseButtonState extends State<PlayPauseButton> | ||||
|     with SingleTickerProviderStateMixin { | ||||
|   AnimationController playPauseAnimation; | ||||
|   BackendState backend; | ||||
|   MusicusBackendState backend; | ||||
|   StreamSubscription<bool> playingSubscription; | ||||
| 
 | ||||
|   @override | ||||
|  | @ -29,14 +28,14 @@ class _PlayPauseButtonState extends State<PlayPauseButton> | |||
|   void didChangeDependencies() { | ||||
|     super.didChangeDependencies(); | ||||
| 
 | ||||
|     backend = Backend.of(context); | ||||
|     playPauseAnimation.value = backend.player.playing.value ? 1.0 : 0.0; | ||||
|     backend = MusicusBackend.of(context); | ||||
|     playPauseAnimation.value = backend.playback.playing.value ? 1.0 : 0.0; | ||||
| 
 | ||||
|     if (playingSubscription != null) { | ||||
|       playingSubscription.cancel(); | ||||
|     } | ||||
| 
 | ||||
|     playingSubscription = backend.player.playing.listen((playing) => | ||||
|     playingSubscription = backend.playback.playing.listen((playing) => | ||||
|         playing ? playPauseAnimation.forward() : playPauseAnimation.reverse()); | ||||
|   } | ||||
| 
 | ||||
|  | @ -47,7 +46,7 @@ class _PlayPauseButtonState extends State<PlayPauseButton> | |||
|         icon: AnimatedIcons.play_pause, | ||||
|         progress: playPauseAnimation, | ||||
|       ), | ||||
|       onPressed: backend.player.playPause, | ||||
|       onPressed: backend.playback.playPause, | ||||
|     ); | ||||
|   } | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,17 +1,15 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_common/musicus_common.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../music_library.dart'; | ||||
| import '../screens/program.dart'; | ||||
| 
 | ||||
| import 'play_pause_button.dart'; | ||||
| import 'texts.dart'; | ||||
| 
 | ||||
| class PlayerBar extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     final backend = MusicusBackend.of(context); | ||||
| 
 | ||||
|     return BottomAppBar( | ||||
|       child: InkWell( | ||||
|  | @ -19,7 +17,7 @@ class PlayerBar extends StatelessWidget { | |||
|           mainAxisSize: MainAxisSize.min, | ||||
|           children: <Widget>[ | ||||
|             StreamBuilder( | ||||
|               stream: backend.player.normalizedPosition, | ||||
|               stream: backend.playback.normalizedPosition, | ||||
|               builder: (context, snapshot) => LinearProgressIndicator( | ||||
|                 value: snapshot.data, | ||||
|               ), | ||||
|  | @ -32,7 +30,7 @@ class PlayerBar extends StatelessWidget { | |||
|                 ), | ||||
|                 Expanded( | ||||
|                   child: StreamBuilder<InternalTrack>( | ||||
|                     stream: backend.player.currentTrack, | ||||
|                     stream: backend.playback.currentTrack, | ||||
|                     builder: (context, snapshot) { | ||||
|                       if (snapshot.data != null) { | ||||
|                         final recordingId = snapshot.data.track.recordingId; | ||||
|  |  | |||
|  | @ -1,49 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import 'texts.dart'; | ||||
| 
 | ||||
| class RecordingTile extends StatelessWidget { | ||||
|   final WorkInfo workInfo; | ||||
|   final RecordingInfo recordingInfo; | ||||
| 
 | ||||
|   RecordingTile({ | ||||
|     this.workInfo, | ||||
|     this.recordingInfo, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final textTheme = Theme.of(context).textTheme; | ||||
| 
 | ||||
|     return Padding( | ||||
|       padding: const EdgeInsets.symmetric( | ||||
|         vertical: 8.0, | ||||
|       ), | ||||
|       child: Column( | ||||
|         crossAxisAlignment: CrossAxisAlignment.start, | ||||
|         children: <Widget>[ | ||||
|           DefaultTextStyle( | ||||
|             style: textTheme.subtitle1, | ||||
|             child: Text(workInfo.composers | ||||
|                 .map((p) => '${p.firstName} ${p.lastName}') | ||||
|                 .join(', ')), | ||||
|           ), | ||||
|           DefaultTextStyle( | ||||
|             style: textTheme.headline6, | ||||
|             child: Text(workInfo.work.title), | ||||
|           ), | ||||
|           const SizedBox( | ||||
|             height: 4.0, | ||||
|           ), | ||||
|           DefaultTextStyle( | ||||
|             style: textTheme.bodyText1, | ||||
|             child: PerformancesText( | ||||
|               performanceInfos: recordingInfo.performances, | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | @ -1,73 +0,0 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| import 'package:musicus_database/musicus_database.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| 
 | ||||
| /// A widget showing information on a list of performances. | ||||
| class PerformancesText extends StatelessWidget { | ||||
|   /// The information to show. | ||||
|   final List<PerformanceInfo> performanceInfos; | ||||
| 
 | ||||
|   PerformancesText({ | ||||
|     this.performanceInfos, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final List<String> performanceTexts = []; | ||||
| 
 | ||||
|     for (final p in performanceInfos) { | ||||
|       final buffer = StringBuffer(); | ||||
| 
 | ||||
|       if (p.person != null) { | ||||
|         buffer.write('${p.person.firstName} ${p.person.lastName}'); | ||||
|       } else if (p.ensemble != null) { | ||||
|         buffer.write(p.ensemble.name); | ||||
|       } else { | ||||
|         buffer.write('Unknown'); | ||||
|       } | ||||
| 
 | ||||
|       if (p.role != null) { | ||||
|         buffer.write(' (${p.role.name})'); | ||||
|       } | ||||
| 
 | ||||
|       performanceTexts.add(buffer.toString()); | ||||
|     } | ||||
| 
 | ||||
|     return Text(performanceTexts.join(', ')); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class WorkText extends StatelessWidget { | ||||
|   final int workId; | ||||
| 
 | ||||
|   WorkText(this.workId); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return StreamBuilder<Work>( | ||||
|       stream: backend.db.workById(workId).watchSingle(), | ||||
|       builder: (context, snapshot) => Text(snapshot.data?.title ?? '...'), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class ComposersText extends StatelessWidget { | ||||
|   final int workId; | ||||
| 
 | ||||
|   ComposersText(this.workId); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return StreamBuilder<List<Person>>( | ||||
|       stream: backend.db.composersByWork(workId).watch(), | ||||
|       builder: (context, snapshot) => Text(snapshot.hasData | ||||
|           ? snapshot.data.map((p) => '${p.firstName} ${p.lastName}').join(', ') | ||||
|           : '...'), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Elias Projahn
						Elias Projahn