| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  | import 'dart:async'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 21:05:49 +01:00
										 |  |  | import 'package:flutter/material.dart'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  | import '../backend.dart'; | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  | import '../database.dart'; | 
					
						
							| 
									
										
										
										
											2020-04-22 10:01:50 +02:00
										 |  |  | import '../music_library.dart'; | 
					
						
							| 
									
										
										
										
											2019-12-02 22:05:07 +01:00
										 |  |  | import '../widgets/play_pause_button.dart'; | 
					
						
							| 
									
										
										
										
											2020-04-22 10:32:36 +02:00
										 |  |  | import '../widgets/recording_tile.dart'; | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  | /// Data class to bundle information from the database on one track.
 | 
					
						
							|  |  |  | class ProgramItem { | 
					
						
							|  |  |  |   /// ID of the recording.
 | 
					
						
							|  |  |  |   ///
 | 
					
						
							|  |  |  |   /// We don't need the real recording, as the [RecordingTile] widget handles
 | 
					
						
							|  |  |  |   /// that for us. If the recording is the same one, as the one from the
 | 
					
						
							|  |  |  |   /// previous track, this will be null.
 | 
					
						
							|  |  |  |   final int recordingId; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /// List of work parts contained in this track.
 | 
					
						
							|  |  |  |   ///
 | 
					
						
							|  |  |  |   /// This will include the parts linked in the track as well as all parents of
 | 
					
						
							|  |  |  |   /// them, if there are gaps between them (i.e. some parts are missing).
 | 
					
						
							|  |  |  |   final List<Work> workParts; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ProgramItem({ | 
					
						
							|  |  |  |     this.recordingId, | 
					
						
							|  |  |  |     this.workParts, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /// Widget displaying a [ProgramItem].
 | 
					
						
							|  |  |  | class ProgramTile extends StatelessWidget { | 
					
						
							|  |  |  |   final ProgramItem item; | 
					
						
							|  |  |  |   final bool isPlaying; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ProgramTile({ | 
					
						
							|  |  |  |     this.item, | 
					
						
							|  |  |  |     this.isPlaying, | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							|  |  |  |     return Row( | 
					
						
							|  |  |  |       children: <Widget>[ | 
					
						
							|  |  |  |         Padding( | 
					
						
							|  |  |  |           padding: const EdgeInsets.all(4.0), | 
					
						
							|  |  |  |           child: isPlaying | 
					
						
							|  |  |  |               ? const Icon(Icons.play_arrow) | 
					
						
							|  |  |  |               : SizedBox( | 
					
						
							|  |  |  |                   width: 24.0, | 
					
						
							|  |  |  |                   height: 24.0, | 
					
						
							|  |  |  |                 ), | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |         Expanded( | 
					
						
							|  |  |  |           child: Padding( | 
					
						
							|  |  |  |             padding: const EdgeInsets.all(8.0), | 
					
						
							|  |  |  |             child: Column( | 
					
						
							|  |  |  |               crossAxisAlignment: CrossAxisAlignment.stretch, | 
					
						
							|  |  |  |               children: <Widget>[ | 
					
						
							|  |  |  |                 if (item.recordingId != null) ...[ | 
					
						
							|  |  |  |                   RecordingTile( | 
					
						
							|  |  |  |                     recordingId: item.recordingId, | 
					
						
							|  |  |  |                   ), | 
					
						
							|  |  |  |                   SizedBox( | 
					
						
							|  |  |  |                     height: 8.0, | 
					
						
							|  |  |  |                   ), | 
					
						
							|  |  |  |                 ], | 
					
						
							|  |  |  |                 Column( | 
					
						
							|  |  |  |                   crossAxisAlignment: CrossAxisAlignment.start, | 
					
						
							|  |  |  |                   children: <Widget>[ | 
					
						
							|  |  |  |                     for (final part in item.workParts) | 
					
						
							|  |  |  |                       Padding( | 
					
						
							| 
									
										
										
										
											2020-04-24 21:18:24 +02:00
										 |  |  |                         padding: const EdgeInsets.only( | 
					
						
							|  |  |  |                           left: 8.0, | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |                         ), | 
					
						
							|  |  |  |                         child: Text( | 
					
						
							|  |  |  |                           part.title, | 
					
						
							|  |  |  |                           style: TextStyle( | 
					
						
							|  |  |  |                             fontStyle: FontStyle.italic, | 
					
						
							|  |  |  |                           ), | 
					
						
							|  |  |  |                         ), | 
					
						
							|  |  |  |                       ), | 
					
						
							|  |  |  |                   ], | 
					
						
							|  |  |  |                 ), | 
					
						
							|  |  |  |               ], | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |           ), | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |       ], | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  | class ProgramScreen extends StatefulWidget { | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   _ProgramScreenState createState() => _ProgramScreenState(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _ProgramScreenState extends State<ProgramScreen> { | 
					
						
							| 
									
										
										
										
											2020-03-28 08:51:45 +01:00
										 |  |  |   BackendState backend; | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 19:56:14 +02:00
										 |  |  |   StreamSubscription<bool> playerActiveSubscription; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |   StreamSubscription<List<InternalTrack>> playlistSubscription; | 
					
						
							|  |  |  |   List<ProgramItem> items = []; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |   StreamSubscription<double> positionSubscription; | 
					
						
							|  |  |  |   double position = 0.0; | 
					
						
							|  |  |  |   bool seeking = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   void didChangeDependencies() { | 
					
						
							|  |  |  |     super.didChangeDependencies(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     backend = Backend.of(context); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 19:56:14 +02:00
										 |  |  |     if (playerActiveSubscription != null) { | 
					
						
							|  |  |  |       playerActiveSubscription.cancel(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Close the program screen, if the player is no longer active.
 | 
					
						
							|  |  |  |     playerActiveSubscription = backend.player.active.listen((active) { | 
					
						
							|  |  |  |       if (!active) { | 
					
						
							|  |  |  |         Navigator.pop(context); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |     if (playlistSubscription != null) { | 
					
						
							|  |  |  |       playlistSubscription.cancel(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     playlistSubscription = backend.player.playlist.listen((playlist) { | 
					
						
							|  |  |  |       updateProgram(playlist); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |     if (positionSubscription != null) { | 
					
						
							|  |  |  |       positionSubscription.cancel(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-18 13:50:38 +02:00
										 |  |  |     positionSubscription = backend.player.normalizedPosition.listen((pos) { | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |       if (!seeking) { | 
					
						
							|  |  |  |         setState(() { | 
					
						
							|  |  |  |           position = pos; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |   /// Go through the tracks of [playlist] and preprocess them for displaying.
 | 
					
						
							|  |  |  |   Future<void> updateProgram(List<InternalTrack> playlist) async { | 
					
						
							|  |  |  |     List<ProgramItem> newItems = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // The following variables exist to adapt the resulting ProgramItem to its
 | 
					
						
							|  |  |  |     // predecessor.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the previous recording was the same, we won't need to include the
 | 
					
						
							|  |  |  |     // recording data again.
 | 
					
						
							|  |  |  |     int lastRecordingId; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If the previous work was the same, we won't need to retrieve its parts
 | 
					
						
							|  |  |  |     // from the database again.
 | 
					
						
							|  |  |  |     int lastWorkId; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // This will always contain the parts of the current work.
 | 
					
						
							|  |  |  |     List<Work> workParts = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (var i = 0; i < playlist.length; i++) { | 
					
						
							|  |  |  |       // The data that will be stored in the resulting ProgramItem.
 | 
					
						
							|  |  |  |       int newRecordingId; | 
					
						
							|  |  |  |       List<Work> newWorkParts = []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       final track = playlist[i]; | 
					
						
							|  |  |  |       final recordingId = track.track.recordingId; | 
					
						
							|  |  |  |       final partIds = track.track.partIds; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // newRecordingId will be null, if the recording ID is the same. This
 | 
					
						
							|  |  |  |       // also means, that the work is the same, so workParts doesn't have to
 | 
					
						
							|  |  |  |       // be updated either.
 | 
					
						
							|  |  |  |       if (recordingId != lastRecordingId) { | 
					
						
							|  |  |  |         lastRecordingId = recordingId; | 
					
						
							|  |  |  |         newRecordingId = recordingId; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         final recording = | 
					
						
							|  |  |  |             await backend.db.recordingById(recordingId).getSingle(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (recording.work != lastWorkId) { | 
					
						
							|  |  |  |           workParts = await backend.db.workParts(recording.work).get(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         lastWorkId = recording.work; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (final partId in partIds) { | 
					
						
							|  |  |  |         newWorkParts.add(workParts[partId]); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       newItems.add(ProgramItem( | 
					
						
							|  |  |  |         recordingId: newRecordingId, | 
					
						
							|  |  |  |         workParts: newWorkParts, | 
					
						
							|  |  |  |       )); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Check, whether we are still a part of the widget tree, because this
 | 
					
						
							|  |  |  |     // function might take some time.
 | 
					
						
							|  |  |  |     if (mounted) { | 
					
						
							|  |  |  |       setState(() { | 
					
						
							|  |  |  |         items = newItems; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-02 21:05:49 +01:00
										 |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							|  |  |  |     return Scaffold( | 
					
						
							|  |  |  |       appBar: AppBar( | 
					
						
							|  |  |  |         leading: IconButton( | 
					
						
							|  |  |  |           icon: const Icon(Icons.keyboard_arrow_down), | 
					
						
							|  |  |  |           onPressed: () => Navigator.pop(context), | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |         title: Text('Program'), | 
					
						
							|  |  |  |       ), | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |       body: StreamBuilder<int>( | 
					
						
							|  |  |  |         stream: backend.player.currentIndex, | 
					
						
							| 
									
										
										
										
											2020-04-22 10:01:50 +02:00
										 |  |  |         builder: (context, snapshot) { | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |           if (snapshot.hasData) { | 
					
						
							|  |  |  |             return ListView.builder( | 
					
						
							|  |  |  |               itemCount: items.length, | 
					
						
							|  |  |  |               itemBuilder: (context, index) { | 
					
						
							|  |  |  |                 return InkWell( | 
					
						
							|  |  |  |                   child: ProgramTile( | 
					
						
							|  |  |  |                     item: items[index], | 
					
						
							|  |  |  |                     isPlaying: index == snapshot?.data, | 
					
						
							|  |  |  |                   ), | 
					
						
							|  |  |  |                   onTap: () { | 
					
						
							|  |  |  |                     backend.player.skipTo(index); | 
					
						
							|  |  |  |                   }, | 
					
						
							|  |  |  |                 ); | 
					
						
							| 
									
										
										
										
											2020-04-22 10:01:50 +02:00
										 |  |  |               }, | 
					
						
							|  |  |  |             ); | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             return Container(); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       ), | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |       bottomNavigationBar: BottomAppBar( | 
					
						
							|  |  |  |         child: Column( | 
					
						
							|  |  |  |           mainAxisSize: MainAxisSize.min, | 
					
						
							|  |  |  |           children: <Widget>[ | 
					
						
							|  |  |  |             Slider( | 
					
						
							|  |  |  |               value: position, | 
					
						
							|  |  |  |               onChangeStart: (_) { | 
					
						
							|  |  |  |                 seeking = true; | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |               onChangeEnd: (pos) { | 
					
						
							|  |  |  |                 seeking = false; | 
					
						
							| 
									
										
										
										
											2020-04-18 13:50:38 +02:00
										 |  |  |                 backend.player.seekTo(pos); | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |               }, | 
					
						
							|  |  |  |               onChanged: (pos) { | 
					
						
							|  |  |  |                 setState(() { | 
					
						
							|  |  |  |                   position = pos; | 
					
						
							|  |  |  |                 }); | 
					
						
							|  |  |  |               }, | 
					
						
							|  |  |  |             ), | 
					
						
							| 
									
										
										
										
											2019-12-02 22:05:07 +01:00
										 |  |  |             Row( | 
					
						
							|  |  |  |               children: <Widget>[ | 
					
						
							| 
									
										
										
										
											2019-12-03 10:24:23 +01:00
										 |  |  |                 Padding( | 
					
						
							|  |  |  |                   padding: const EdgeInsets.only(left: 24.0), | 
					
						
							| 
									
										
										
										
											2020-04-18 13:50:38 +02:00
										 |  |  |                   child: StreamBuilder<Duration>( | 
					
						
							|  |  |  |                     stream: backend.player.position, | 
					
						
							|  |  |  |                     builder: (context, snapshot) { | 
					
						
							|  |  |  |                       if (snapshot.hasData) { | 
					
						
							|  |  |  |                         return DurationText(snapshot.data); | 
					
						
							|  |  |  |                       } else { | 
					
						
							|  |  |  |                         return Container(); | 
					
						
							|  |  |  |                       } | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                   ), | 
					
						
							| 
									
										
										
										
											2019-12-03 10:24:23 +01:00
										 |  |  |                 ), | 
					
						
							|  |  |  |                 Spacer(), | 
					
						
							| 
									
										
										
										
											2019-12-02 22:05:07 +01:00
										 |  |  |                 IconButton( | 
					
						
							|  |  |  |                   icon: const Icon(Icons.skip_previous), | 
					
						
							| 
									
										
										
										
											2020-04-22 10:01:50 +02:00
										 |  |  |                   onPressed: () { | 
					
						
							|  |  |  |                     backend.player.skipToPrevious(); | 
					
						
							|  |  |  |                   }, | 
					
						
							| 
									
										
										
										
											2019-12-02 22:05:07 +01:00
										 |  |  |                 ), | 
					
						
							|  |  |  |                 PlayPauseButton(), | 
					
						
							|  |  |  |                 IconButton( | 
					
						
							|  |  |  |                   icon: const Icon(Icons.skip_next), | 
					
						
							| 
									
										
										
										
											2020-04-22 10:01:50 +02:00
										 |  |  |                   onPressed: () { | 
					
						
							|  |  |  |                     backend.player.skipToNext(); | 
					
						
							|  |  |  |                   }, | 
					
						
							| 
									
										
										
										
											2019-12-02 22:05:07 +01:00
										 |  |  |                 ), | 
					
						
							| 
									
										
										
										
											2019-12-03 10:24:23 +01:00
										 |  |  |                 Spacer(), | 
					
						
							|  |  |  |                 Padding( | 
					
						
							|  |  |  |                   padding: const EdgeInsets.only(right: 20.0), | 
					
						
							| 
									
										
										
										
											2020-04-18 13:50:38 +02:00
										 |  |  |                   child: StreamBuilder<Duration>( | 
					
						
							|  |  |  |                     stream: backend.player.duration, | 
					
						
							|  |  |  |                     builder: (context, snapshot) { | 
					
						
							|  |  |  |                       if (snapshot.hasData) { | 
					
						
							|  |  |  |                         return DurationText(snapshot.data); | 
					
						
							|  |  |  |                       } else { | 
					
						
							|  |  |  |                         return Container(); | 
					
						
							|  |  |  |                       } | 
					
						
							|  |  |  |                     }, | 
					
						
							|  |  |  |                   ), | 
					
						
							| 
									
										
										
										
											2019-12-03 10:24:23 +01:00
										 |  |  |                 ), | 
					
						
							| 
									
										
										
										
											2019-12-02 22:05:07 +01:00
										 |  |  |               ], | 
					
						
							|  |  |  |             ), | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |           ], | 
					
						
							|  |  |  |         ), | 
					
						
							|  |  |  |       ), | 
					
						
							| 
									
										
										
										
											2019-12-02 21:05:49 +01:00
										 |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   void dispose() { | 
					
						
							|  |  |  |     super.dispose(); | 
					
						
							| 
									
										
										
										
											2020-04-24 19:56:14 +02:00
										 |  |  |     playerActiveSubscription.cancel(); | 
					
						
							| 
									
										
										
										
											2020-04-24 19:53:29 +02:00
										 |  |  |     playlistSubscription.cancel(); | 
					
						
							| 
									
										
										
										
											2019-12-02 21:38:47 +01:00
										 |  |  |     positionSubscription.cancel(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-12-02 21:05:49 +01:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-18 13:50:38 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | class DurationText extends StatelessWidget { | 
					
						
							|  |  |  |   final Duration duration; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DurationText(this.duration); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   @override | 
					
						
							|  |  |  |   Widget build(BuildContext context) { | 
					
						
							|  |  |  |     final minutes = duration.inMinutes; | 
					
						
							|  |  |  |     final seconds = (duration - Duration(minutes: minutes)).inSeconds; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     final secondsString = seconds >= 10 ? seconds.toString() : '0$seconds'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return Text('$minutes:$secondsString'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |