musicus_mobile/mobile/lib/editors/recording.dart

216 lines
6.2 KiB
Dart
Raw Normal View History

2020-03-22 16:49:20 +01:00
import 'package:flutter/material.dart';
2020-04-24 22:41:52 +02:00
import 'package:musicus_database/musicus_database.dart';
2020-03-22 16:49:20 +01:00
import '../backend.dart';
2020-04-26 15:35:45 +02:00
import '../editors/performance.dart';
import '../selectors/recording.dart';
2020-03-22 16:49:20 +01:00
import '../selectors/work.dart';
2020-04-26 15:35:45 +02:00
/// Screen for editing a recording.
///
/// If the user has finished editing, the result will be returned using the
/// navigator as a [RecordingSelectorResult] object.
2020-03-22 16:49:20 +01:00
class RecordingEditor extends StatefulWidget {
/// The recording to edit.
2020-04-26 18:29:21 +02:00
///
/// If this is null, a new recording will be created.
final RecordingInfo recordingInfo;
2020-03-22 16:49:20 +01:00
RecordingEditor({
this.recordingInfo,
2020-03-22 16:49:20 +01:00
});
@override
_RecordingEditorState createState() => _RecordingEditorState();
}
class _RecordingEditorState extends State<RecordingEditor> {
2020-04-12 11:50:32 +02:00
final commentController = TextEditingController();
2020-04-26 18:29:21 +02:00
bool uploading = false;
2020-04-26 15:35:45 +02:00
WorkInfo workInfo;
List<PerformanceInfo> performanceInfos = [];
2020-03-22 16:49:20 +01:00
@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);
}();
2020-03-22 16:49:20 +01:00
}
}
@override
Widget build(BuildContext context) {
final backend = Backend.of(context);
Future<void> selectWork() async {
2020-04-26 15:35:45 +02:00
final WorkInfo newWorkInfo = await Navigator.push(
2020-03-22 16:49:20 +01:00
context,
MaterialPageRoute(
builder: (context) => WorkSelector(),
fullscreenDialog: true,
));
2020-04-26 15:35:45 +02:00
if (newWorkInfo != null) {
2020-03-22 16:49:20 +01:00
setState(() {
2020-04-26 15:35:45 +02:00
workInfo = newWorkInfo;
2020-03-22 16:49:20 +01:00
});
}
}
2020-04-26 15:35:45 +02:00
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;
});
}
},
));
}
2020-03-22 16:49:20 +01:00
return Scaffold(
appBar: AppBar(
title: Text('Recording'),
actions: <Widget>[
2020-04-26 18:29:21 +02:00
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'),
));
}
},
2020-04-26 16:48:05 +02:00
),
2020-03-22 16:49:20 +01:00
],
),
body: ListView(
children: <Widget>[
2020-04-26 15:35:45 +02:00
workInfo != null
? ListTile(
2020-04-26 15:35:45 +02:00
title: Text(workInfo.work.title),
subtitle: Text(workInfo.composers
.map((p) => '${p.firstName} ${p.lastName}')
.join(', ')),
onTap: selectWork,
)
2020-03-22 16:49:20 +01:00
: ListTile(
title: Text('Work'),
subtitle: Text('Select work'),
onTap: selectWork,
),
2020-04-12 11:50:32 +02:00
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',
),
),
),
2020-03-22 16:49:20 +01:00
ListTile(
title: Text('Performers'),
trailing: IconButton(
icon: const Icon(Icons.add),
onPressed: () async {
2020-04-26 15:35:45 +02:00
final PerformanceInfo model = await Navigator.push(
2020-03-22 16:49:20 +01:00
context,
MaterialPageRoute(
2020-04-26 15:35:45 +02:00
builder: (context) => PerformanceEditor(),
2020-03-22 16:49:20 +01:00
fullscreenDialog: true,
));
if (model != null) {
setState(() {
2020-04-26 15:35:45 +02:00
performanceInfos.add(model);
2020-03-22 16:49:20 +01:00
});
}
},
),
),
2020-04-26 15:35:45 +02:00
...performanceTiles,
2020-03-22 16:49:20 +01:00
],
),
);
}
}