musicus_mobile/lib/app.dart

149 lines
4.3 KiB
Dart
Raw Normal View History

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';
import 'widgets/player_bar.dart';
class App extends StatelessWidget {
2019-12-02 21:05:49 +01:00
@override
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) {
return Material(
color: Theme.of(context).scaffoldBackgroundColor,
);
} 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();
},
),
],
),
);
} else {
return Content();
}
},
),
);
}
2019-12-02 21:05:49 +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) {
// 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: '/',
),
bottomNavigationBar: SizeTransition(
sizeFactor: CurvedAnimation(
curve: Curves.easeOut,
parent: playerBarAnimation,
2019-12-02 21:05:49 +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
}