2019-12-03 12:23:41 +01:00
|
|
|
import 'dart:async';
|
|
|
|
|
|
2019-12-02 21:05:49 +01:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
|
|
|
|
|
import 'backend.dart';
|
|
|
|
|
import 'screens/home.dart';
|
2020-03-28 10:18:52 +01:00
|
|
|
import 'selectors/files.dart';
|
2019-12-02 21:05:49 +01:00
|
|
|
import 'widgets/player_bar.dart';
|
|
|
|
|
|
2020-03-28 09:16:42 +01:00
|
|
|
class App extends StatelessWidget {
|
2019-12-02 21:05:49 +01:00
|
|
|
@override
|
2020-03-28 09:16:42 +01:00
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
final backend = Backend.of(context);
|
|
|
|
|
|
|
|
|
|
return MaterialApp(
|
|
|
|
|
title: 'Musicus',
|
|
|
|
|
theme: ThemeData(
|
|
|
|
|
brightness: Brightness.dark,
|
|
|
|
|
accentColor: Colors.amber,
|
|
|
|
|
textSelectionColor: Colors.grey[600],
|
|
|
|
|
cursorColor: Colors.amber,
|
|
|
|
|
textSelectionHandleColor: Colors.amber,
|
|
|
|
|
toggleableActiveColor: Colors.amber,
|
|
|
|
|
// Added for sliders and FABs. Not everything seems to obey this.
|
|
|
|
|
colorScheme: ColorScheme.dark(
|
|
|
|
|
primary: Colors.amber,
|
|
|
|
|
secondary: Colors.amber,
|
|
|
|
|
),
|
|
|
|
|
fontFamily: 'Libertinus Sans',
|
|
|
|
|
),
|
|
|
|
|
home: Builder(
|
|
|
|
|
builder: (context) {
|
|
|
|
|
if (backend.status == BackendStatus.loading) {
|
2020-03-28 09:54:04 +01:00
|
|
|
return Material(
|
2020-03-28 09:16:42 +01:00
|
|
|
color: Theme.of(context).scaffoldBackgroundColor,
|
|
|
|
|
);
|
2020-03-28 09:54:04 +01:00
|
|
|
} else if (backend.status == BackendStatus.needsPermissions) {
|
|
|
|
|
return Material(
|
|
|
|
|
color: Theme.of(context).scaffoldBackgroundColor,
|
|
|
|
|
child: Column(
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
Text(
|
|
|
|
|
'Musicus needs permissions\nto access your files.',
|
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
|
style: Theme.of(context).textTheme.headline6,
|
|
|
|
|
),
|
|
|
|
|
SizedBox(
|
|
|
|
|
height: 16.0,
|
|
|
|
|
),
|
|
|
|
|
ListTile(
|
|
|
|
|
leading: const Icon(Icons.done),
|
|
|
|
|
title: Text('Grant permissions'),
|
|
|
|
|
onTap: () {
|
|
|
|
|
backend.requestPermissions();
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
ListTile(
|
|
|
|
|
leading: const Icon(Icons.settings),
|
|
|
|
|
title: Text('Open system\'s app settings'),
|
|
|
|
|
onTap: () {
|
|
|
|
|
backend.openAppSettings();
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
);
|
2020-03-28 10:18:52 +01:00
|
|
|
} else if (backend.status == BackendStatus.setup) {
|
|
|
|
|
return Material(
|
|
|
|
|
color: Theme.of(context).scaffoldBackgroundColor,
|
|
|
|
|
child: Column(
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
Text(
|
|
|
|
|
'Choose the base path for\nyour music library.',
|
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
|
style: Theme.of(context).textTheme.headline6,
|
|
|
|
|
),
|
|
|
|
|
SizedBox(
|
|
|
|
|
height: 16.0,
|
|
|
|
|
),
|
|
|
|
|
ListTile(
|
|
|
|
|
leading: const Icon(Icons.folder_open),
|
|
|
|
|
title: Text('Choose path'),
|
|
|
|
|
onTap: () async {
|
|
|
|
|
final path = await Navigator.push<String>(
|
|
|
|
|
context,
|
|
|
|
|
MaterialPageRoute(
|
|
|
|
|
builder: (context) => FilesSelector(
|
|
|
|
|
mode: FilesSelectorMode.directory,
|
|
|
|
|
),
|
|
|
|
|
fullscreenDialog: true,
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (path != null) {
|
|
|
|
|
backend.setMusicLibraryPath(path);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
);
|
2020-03-28 09:16:42 +01:00
|
|
|
} else {
|
|
|
|
|
return Content();
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
2019-12-02 21:05:49 +01:00
|
|
|
}
|
|
|
|
|
|
2020-03-28 09:16:42 +01:00
|
|
|
class Content extends StatefulWidget {
|
|
|
|
|
@override
|
|
|
|
|
_ContentState createState() => _ContentState();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class _ContentState extends State<Content> with SingleTickerProviderStateMixin {
|
2019-12-02 21:05:49 +01:00
|
|
|
final nestedNavigator = GlobalKey<NavigatorState>();
|
|
|
|
|
|
|
|
|
|
AnimationController playerBarAnimation;
|
2020-03-28 08:51:45 +01:00
|
|
|
BackendState backend;
|
2019-12-03 12:23:41 +01:00
|
|
|
StreamSubscription<bool> playerActiveSubscription;
|
2019-12-02 21:05:49 +01:00
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void initState() {
|
|
|
|
|
super.initState();
|
|
|
|
|
|
|
|
|
|
playerBarAnimation = AnimationController(
|
|
|
|
|
vsync: this,
|
|
|
|
|
duration: Duration(milliseconds: 300),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void didChangeDependencies() {
|
|
|
|
|
super.didChangeDependencies();
|
|
|
|
|
|
|
|
|
|
backend = Backend.of(context);
|
2019-12-03 12:33:43 +01:00
|
|
|
playerBarAnimation.value = backend.playerActive.value ? 1.0 : 0.0;
|
2019-12-03 12:23:41 +01:00
|
|
|
|
|
|
|
|
if (playerActiveSubscription != null) {
|
|
|
|
|
playerActiveSubscription.cancel();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
playerActiveSubscription = backend.playerActive.listen((active) =>
|
2019-12-02 21:05:49 +01:00
|
|
|
active ? playerBarAnimation.forward() : playerBarAnimation.reverse());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
2020-03-28 09:16:42 +01:00
|
|
|
// The nested Navigator is for every screen from which the player bar at
|
|
|
|
|
// the bottom should be accessible. The WillPopScope widget intercepts
|
|
|
|
|
// taps on the system back button and redirects them to the nested
|
|
|
|
|
// navigator.
|
|
|
|
|
return WillPopScope(
|
|
|
|
|
onWillPop: () async => !(await nestedNavigator.currentState.maybePop()),
|
|
|
|
|
child: Scaffold(
|
|
|
|
|
body: Navigator(
|
|
|
|
|
key: nestedNavigator,
|
|
|
|
|
onGenerateRoute: (settings) => settings.name == '/'
|
|
|
|
|
? MaterialPageRoute(
|
|
|
|
|
builder: (context) => HomeScreen(),
|
|
|
|
|
)
|
|
|
|
|
: null,
|
|
|
|
|
initialRoute: '/',
|
2019-12-02 21:58:13 +01:00
|
|
|
),
|
2020-03-28 09:16:42 +01:00
|
|
|
bottomNavigationBar: SizeTransition(
|
|
|
|
|
sizeFactor: CurvedAnimation(
|
|
|
|
|
curve: Curves.easeOut,
|
|
|
|
|
parent: playerBarAnimation,
|
2019-12-02 21:05:49 +01:00
|
|
|
),
|
2020-03-28 09:16:42 +01:00
|
|
|
axisAlignment: -1.0,
|
|
|
|
|
child: PlayerBar(),
|
2019-12-02 21:05:49 +01:00
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
2019-12-03 12:23:41 +01:00
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
void dispose() {
|
|
|
|
|
super.dispose();
|
|
|
|
|
playerActiveSubscription.cancel();
|
|
|
|
|
}
|
2019-12-02 21:05:49 +01:00
|
|
|
}
|