mirror of
				https://github.com/johrpan/musicus_mobile.git
				synced 2025-10-26 18:57:25 +01:00 
			
		
		
		
	Rename top level directories
This commit is contained in:
		
							parent
							
								
									256bda1b72
								
							
						
					
					
						commit
						674ce19414
					
				
					 90 changed files with 0 additions and 0 deletions
				
			
		
							
								
								
									
										162
									
								
								mobile/lib/selectors/files.dart
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								mobile/lib/selectors/files.dart
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,162 @@ | |||
| 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.musicLibraryUri, 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; | ||||
|     } | ||||
|   } | ||||
| } | ||||
							
								
								
									
										108
									
								
								mobile/lib/selectors/instruments.dart
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								mobile/lib/selectors/instruments.dart
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,108 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../database.dart'; | ||||
| import '../editors/instrument.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> { | ||||
|   Set<Instrument> selection = {}; | ||||
| 
 | ||||
|   @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: StreamBuilder( | ||||
|         stream: backend.db.allInstruments().watch(), | ||||
|         builder: (context, snapshot) { | ||||
|           if (snapshot.hasData) { | ||||
|             return ListView.builder( | ||||
|               itemCount: snapshot.data.length, | ||||
|               itemBuilder: (context, index) { | ||||
|                 final instrument = snapshot.data[index]; | ||||
| 
 | ||||
|                 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), | ||||
|                   ); | ||||
|                 } | ||||
|               }, | ||||
|             ); | ||||
|           } else { | ||||
|             return Container(); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|       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); | ||||
|               }); | ||||
|             } else { | ||||
|               Navigator.pop(context, instrument); | ||||
|             } | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										200
									
								
								mobile/lib/selectors/performer.dart
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								mobile/lib/selectors/performer.dart
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,200 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../database.dart'; | ||||
| import '../editors/ensemble.dart'; | ||||
| import '../editors/person.dart'; | ||||
| 
 | ||||
| import 'instruments.dart'; | ||||
| 
 | ||||
| class PerformerSelector extends StatefulWidget { | ||||
|   @override | ||||
|   _PerformerSelectorState createState() => _PerformerSelectorState(); | ||||
| } | ||||
| 
 | ||||
| class _Selection { | ||||
|   final bool isPerson; | ||||
|   final Person person; | ||||
|   final Ensemble ensemble; | ||||
| 
 | ||||
|   _Selection.person(this.person) | ||||
|       : isPerson = true, | ||||
|         ensemble = null; | ||||
| 
 | ||||
|   _Selection.ensemble(this.ensemble) | ||||
|       : isPerson = false, | ||||
|         person = null; | ||||
| } | ||||
| 
 | ||||
| class _PerformerSelectorState extends State<PerformerSelector> { | ||||
|   Instrument role; | ||||
|   _Selection selection; | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select performer'), | ||||
|         actions: <Widget>[ | ||||
|           FlatButton( | ||||
|             child: Text('DONE'), | ||||
|             onPressed: () => Navigator.pop( | ||||
|               context, | ||||
|               PerformanceModel( | ||||
|                 person: selection?.person, | ||||
|                 ensemble: selection?.ensemble, | ||||
|                 role: role, | ||||
|               ), | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       body: Column( | ||||
|         children: <Widget>[ | ||||
|           Material( | ||||
|             elevation: 2.0, | ||||
|             child: ListTile( | ||||
|               title: Text('Instrument/Role'), | ||||
|               subtitle: | ||||
|                   Text(role != null ? role.name : 'Select instrument/role'), | ||||
|               trailing: IconButton( | ||||
|                 icon: const Icon(Icons.delete), | ||||
|                 onPressed: () { | ||||
|                   setState(() { | ||||
|                     role = null; | ||||
|                   }); | ||||
|                 }, | ||||
|               ), | ||||
|               onTap: () async { | ||||
|                 final Instrument newRole = await Navigator.push( | ||||
|                     context, | ||||
|                     MaterialPageRoute( | ||||
|                       builder: (context) => InstrumentsSelector(), | ||||
|                       fullscreenDialog: true, | ||||
|                     )); | ||||
| 
 | ||||
|                 if (newRole != null) { | ||||
|                   setState(() { | ||||
|                     role = newRole; | ||||
|                   }); | ||||
|                 } | ||||
|               }, | ||||
|             ), | ||||
|           ), | ||||
|           Expanded( | ||||
|             child: ListView( | ||||
|               children: <Widget>[ | ||||
|                 StreamBuilder<List<Person>>( | ||||
|                   stream: backend.db.allPersons().watch(), | ||||
|                   builder: (context, snapshot) { | ||||
|                     if (snapshot.hasData && snapshot.data.isNotEmpty) { | ||||
|                       return ExpansionTile( | ||||
|                         initiallyExpanded: true, | ||||
|                         title: Text('Persons'), | ||||
|                         children: snapshot.data | ||||
|                             .map((person) => RadioListTile<Person>( | ||||
|                                   title: Text( | ||||
|                                       '${person.lastName}, ${person.firstName}'), | ||||
|                                   value: person, | ||||
|                                   groupValue: selection?.person, | ||||
|                                   onChanged: (person) { | ||||
|                                     setState(() { | ||||
|                                       selection = _Selection.person(person); | ||||
|                                     }); | ||||
|                                   }, | ||||
|                                 )) | ||||
|                             .toList(), | ||||
|                       ); | ||||
|                     } else { | ||||
|                       return Container(); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|                 StreamBuilder<List<Ensemble>>( | ||||
|                   stream: backend.db.allEnsembles().watch(), | ||||
|                   builder: (context, snapshot) { | ||||
|                     if (snapshot.hasData && snapshot.data.isNotEmpty) { | ||||
|                       return ExpansionTile( | ||||
|                         initiallyExpanded: true, | ||||
|                         title: Text('Ensembles'), | ||||
|                         children: snapshot.data | ||||
|                             .map((ensemble) => RadioListTile<Ensemble>( | ||||
|                                   title: Text(ensemble.name), | ||||
|                                   value: ensemble, | ||||
|                                   groupValue: selection?.ensemble, | ||||
|                                   onChanged: (ensemble) { | ||||
|                                     setState(() { | ||||
|                                       selection = _Selection.ensemble(ensemble); | ||||
|                                     }); | ||||
|                                   }, | ||||
|                                 )) | ||||
|                             .toList(), | ||||
|                       ); | ||||
|                     } else { | ||||
|                       return Container(); | ||||
|                     } | ||||
|                   }, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|           ), | ||||
|         ], | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           showModalBottomSheet( | ||||
|             context: context, | ||||
|             builder: (context) => Column( | ||||
|               mainAxisSize: MainAxisSize.min, | ||||
|               children: <Widget>[ | ||||
|                 ListTile( | ||||
|                   leading: const Icon(Icons.add), | ||||
|                   title: Text('Add person'), | ||||
|                   onTap: () async { | ||||
|                     final Person person = await Navigator.push( | ||||
|                         context, | ||||
|                         MaterialPageRoute( | ||||
|                           builder: (context) => PersonEditor(), | ||||
|                           fullscreenDialog: true, | ||||
|                         )); | ||||
| 
 | ||||
|                     if (person != null) { | ||||
|                       setState(() { | ||||
|                         selection = _Selection.person(person); | ||||
|                       }); | ||||
|                     } | ||||
| 
 | ||||
|                     Navigator.pop(context); | ||||
|                   }, | ||||
|                 ), | ||||
|                 ListTile( | ||||
|                   leading: const Icon(Icons.add), | ||||
|                   title: Text('Add ensemble'), | ||||
|                   onTap: () async { | ||||
|                     final Ensemble ensemble = await Navigator.push( | ||||
|                         context, | ||||
|                         MaterialPageRoute( | ||||
|                           builder: (context) => EnsembleEditor(), | ||||
|                           fullscreenDialog: true, | ||||
|                         )); | ||||
| 
 | ||||
|                     if (ensemble != null) { | ||||
|                       setState(() { | ||||
|                         selection = _Selection.ensemble(ensemble); | ||||
|                       }); | ||||
|                     } | ||||
| 
 | ||||
|                     Navigator.pop(context); | ||||
|                   }, | ||||
|                 ), | ||||
|               ], | ||||
|             ), | ||||
|           ); | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										53
									
								
								mobile/lib/selectors/person.dart
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								mobile/lib/selectors/person.dart
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,53 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../database.dart'; | ||||
| import '../editors/person.dart'; | ||||
| 
 | ||||
| class PersonsSelector extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select person'), | ||||
|       ), | ||||
|       body: StreamBuilder( | ||||
|         stream: backend.db.allPersons().watch(), | ||||
|         builder: (context, snapshot) { | ||||
|           if (snapshot.hasData) { | ||||
|             return ListView.builder( | ||||
|               itemCount: snapshot.data.length, | ||||
|               itemBuilder: (context, index) { | ||||
|                 final person = snapshot.data[index]; | ||||
| 
 | ||||
|                 return ListTile( | ||||
|                   title: Text('${person.lastName}, ${person.firstName}'), | ||||
|                   onTap: () => Navigator.pop(context, person), | ||||
|                 ); | ||||
|               }, | ||||
|             ); | ||||
|           } else { | ||||
|             return Container(); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|       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); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										211
									
								
								mobile/lib/selectors/recording.dart
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										211
									
								
								mobile/lib/selectors/recording.dart
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,211 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../database.dart'; | ||||
| import '../editors/recording.dart'; | ||||
| import '../widgets/texts.dart'; | ||||
| import '../widgets/works_by_composer.dart'; | ||||
| 
 | ||||
| class PersonList extends StatelessWidget { | ||||
|   final void Function(int personId) onSelect; | ||||
| 
 | ||||
|   PersonList({ | ||||
|     this.onSelect, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Material( | ||||
|           elevation: 2.0, | ||||
|           child: ListTile( | ||||
|             title: Text('Composers'), | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: StreamBuilder<List<Person>>( | ||||
|             stream: backend.db.allPersons().watch(), | ||||
|             builder: (context, snapshot) { | ||||
|               if (snapshot.hasData) { | ||||
|                 return ListView.builder( | ||||
|                   itemCount: snapshot.data.length, | ||||
|                   itemBuilder: (context, index) { | ||||
|                     final person = snapshot.data[index]; | ||||
|                     return ListTile( | ||||
|                       title: Text('${person.lastName}, ${person.firstName}'), | ||||
|                       onTap: () => onSelect(person.id), | ||||
|                     ); | ||||
|                   }, | ||||
|                 ); | ||||
|               } else { | ||||
|                 return Container(); | ||||
|               } | ||||
|             }, | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class WorkList extends StatelessWidget { | ||||
|   final int composerId; | ||||
|   final void Function(int workId) onSelect; | ||||
| 
 | ||||
|   WorkList({ | ||||
|     this.composerId, | ||||
|     this.onSelect, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Material( | ||||
|           elevation: 2.0, | ||||
|           child: ListTile( | ||||
|             leading: IconButton( | ||||
|               icon: const Icon(Icons.arrow_back), | ||||
|               onPressed: () => Navigator.pop(context), | ||||
|             ), | ||||
|             title: PersonText(composerId), | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: WorksByComposer( | ||||
|             personId: composerId, | ||||
|             onTap: (selectedWork) => onSelect(selectedWork.id), | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class RecordingList extends StatelessWidget { | ||||
|   final int workId; | ||||
|   final void Function(Recording recording) onSelect; | ||||
| 
 | ||||
|   RecordingList({ | ||||
|     this.workId, | ||||
|     this.onSelect, | ||||
|   }); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
|     return Column( | ||||
|       children: <Widget>[ | ||||
|         Material( | ||||
|           elevation: 2.0, | ||||
|           child: ListTile( | ||||
|             leading: IconButton( | ||||
|               icon: const Icon(Icons.arrow_back), | ||||
|               onPressed: () => Navigator.pop(context), | ||||
|             ), | ||||
|             title: WorkText(workId), | ||||
|             subtitle: ComposersText(workId), | ||||
|           ), | ||||
|         ), | ||||
|         Expanded( | ||||
|           child: StreamBuilder<List<Recording>>( | ||||
|             stream: backend.db.recordingsByWork(workId).watch(), | ||||
|             builder: (context, snapshot) { | ||||
|               if (snapshot.hasData) { | ||||
|                 return ListView.builder( | ||||
|                   itemCount: snapshot.data.length, | ||||
|                   itemBuilder: (context, index) { | ||||
|                     final recording = snapshot.data[index]; | ||||
|                     return ListTile( | ||||
|                       title: PerformancesText(recording.id), | ||||
|                       onTap: () => onSelect(recording), | ||||
|                     ); | ||||
|                   }, | ||||
|                 ); | ||||
|               } else { | ||||
|                 return Container(); | ||||
|               } | ||||
|             }, | ||||
|           ), | ||||
|         ), | ||||
|       ], | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| class RecordingsSelector extends StatefulWidget { | ||||
|   @override | ||||
|   _RecordingsSelectorState createState() => _RecordingsSelectorState(); | ||||
| } | ||||
| 
 | ||||
| class _RecordingsSelectorState extends State<RecordingsSelector> { | ||||
|   final nestedNavigator = GlobalKey<NavigatorState>(); | ||||
| 
 | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     // This exists to circumvent the nested navigator when selecting a | ||||
|     // recording. | ||||
|     void popUpperNavigator(Recording recording) { | ||||
|       Navigator.pop(context, recording); | ||||
|     } | ||||
| 
 | ||||
|     return WillPopScope( | ||||
|       child: Scaffold( | ||||
|         appBar: AppBar( | ||||
|           leading: IconButton( | ||||
|             icon: const Icon(Icons.close), | ||||
|             onPressed: () => Navigator.pop(context), | ||||
|           ), | ||||
|           title: Text('Select recording'), | ||||
|         ), | ||||
|         body: Navigator( | ||||
|           key: nestedNavigator, | ||||
|           onGenerateRoute: (settings) => settings.name == '/' | ||||
|               ? MaterialPageRoute( | ||||
|                   builder: (context) => PersonList( | ||||
|                     onSelect: (personId) => nestedNavigator.currentState.push( | ||||
|                       MaterialPageRoute( | ||||
|                         builder: (context) => WorkList( | ||||
|                           composerId: personId, | ||||
|                           onSelect: (workId) => | ||||
|                               nestedNavigator.currentState.push( | ||||
|                             MaterialPageRoute( | ||||
|                               builder: (context) => RecordingList( | ||||
|                                 workId: workId, | ||||
|                                 onSelect: (recording) => | ||||
|                                     popUpperNavigator(recording), | ||||
|                               ), | ||||
|                             ), | ||||
|                           ), | ||||
|                         ), | ||||
|                       ), | ||||
|                     ), | ||||
|                   ), | ||||
|                 ) | ||||
|               : null, | ||||
|           initialRoute: '/', | ||||
|         ), | ||||
|         floatingActionButton: FloatingActionButton( | ||||
|           child: const Icon(Icons.add), | ||||
|           onPressed: () async { | ||||
|             final recording = await Navigator.push( | ||||
|               context, | ||||
|               MaterialPageRoute( | ||||
|                 builder: (context) => RecordingEditor(), | ||||
|                 fullscreenDialog: true, | ||||
|               ), | ||||
|             ); | ||||
| 
 | ||||
|             if (recording != null) { | ||||
|               Navigator.pop(context, recording); | ||||
|             } | ||||
|           }, | ||||
|         ), | ||||
|       ), | ||||
|       onWillPop: () async => !(await nestedNavigator.currentState.maybePop()), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
							
								
								
									
										71
									
								
								mobile/lib/selectors/work.dart
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								mobile/lib/selectors/work.dart
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,71 @@ | |||
| import 'package:flutter/material.dart'; | ||||
| 
 | ||||
| import '../backend.dart'; | ||||
| import '../database.dart'; | ||||
| import '../editors/work.dart'; | ||||
| 
 | ||||
| // TODO: Lazy load works and/or optimize queries. | ||||
| class WorkSelector extends StatelessWidget { | ||||
|   @override | ||||
|   Widget build(BuildContext context) { | ||||
|     final backend = Backend.of(context); | ||||
| 
 | ||||
|     return Scaffold( | ||||
|       appBar: AppBar( | ||||
|         title: Text('Select work'), | ||||
|       ), | ||||
|       body: StreamBuilder<List<Person>>( | ||||
|         stream: backend.db.allPersons().watch(), | ||||
|         builder: (context, snapshot) { | ||||
|           if (snapshot.hasData) { | ||||
|             return ListView.builder( | ||||
|               itemCount: snapshot.data.length, | ||||
|               itemBuilder: (context, index) { | ||||
|                 final person = snapshot.data[index]; | ||||
|                 final title = Text('${person.lastName}, ${person.firstName}'); | ||||
|                 return StreamBuilder<List<Work>>( | ||||
|                   stream: backend.db.worksByComposer(person.id).watch(), | ||||
|                   builder: (context, snapshot) { | ||||
|                     if (snapshot.hasData && snapshot.data.isNotEmpty) { | ||||
|                       return ExpansionTile( | ||||
|                         title: title, | ||||
|                         children: <Widget>[ | ||||
|                           for (final work in snapshot.data) | ||||
|                             ListTile( | ||||
|                               title: Text(work.title), | ||||
|                               onTap: () => Navigator.pop(context, work), | ||||
|                             ), | ||||
|                         ], | ||||
|                       ); | ||||
|                     } else { | ||||
|                       return ListTile( | ||||
|                         title: title, | ||||
|                       ); | ||||
|                     } | ||||
|                   }, | ||||
|                 ); | ||||
|               }, | ||||
|             ); | ||||
|           } else { | ||||
|             return Container(); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|       floatingActionButton: FloatingActionButton( | ||||
|         child: const Icon(Icons.add), | ||||
|         onPressed: () async { | ||||
|           final Work work = await Navigator.push( | ||||
|               context, | ||||
|               MaterialPageRoute( | ||||
|                 builder: (context) => WorkEditor(), | ||||
|                 fullscreenDialog: true, | ||||
|               )); | ||||
| 
 | ||||
|           if (work != null) { | ||||
|             Navigator.pop(context, work); | ||||
|           } | ||||
|         }, | ||||
|       ), | ||||
|     ); | ||||
|   } | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Elias Projahn
						Elias Projahn