import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:fluttertoast/fluttertoast.dart'; import 'package:freezer/settings.dart'; import 'package:freezer/ui/menu.dart'; import 'package:freezer/api/importer.dart'; import 'package:freezer/api/spotify.dart'; import 'package:freezer/ui/elements.dart'; import 'package:freezer/translations.i18n.dart'; import 'package:spotify/spotify.dart' as spotify; import 'package:url_launcher/url_launcher.dart'; import 'dart:async'; class SpotifyImporterV1 extends StatefulWidget { @override _SpotifyImporterV1State createState() => _SpotifyImporterV1State(); } class _SpotifyImporterV1State extends State { String _url; bool _error = false; bool _loading = false; SpotifyPlaylist _data; //Load URL Future _load() async { setState(() { _error = false; _loading = true; }); try { String uri = await SpotifyScrapper.resolveUrl(_url); //Error/NonPlaylist if (uri == null || uri.split(':')[1] != 'playlist') { throw Exception(); } //Load SpotifyPlaylist data = await SpotifyScrapper.playlist(uri); setState(() => _data = data); return; } catch (e, st) { print('$e, $st'); setState(() { _error = true; _loading = false; }); return; } } //Start importing Future _start() async { List tracks = _data.toImporter(); await importer.start(context, _data.name, _data.description, tracks); } @override Widget build(BuildContext context) { return Scaffold( appBar: FreezerAppBar('Importer'.i18n), body: ListView( children: [ ListTile( title: Text('Currently supporting only Spotify, with 100 tracks limit'.i18n), subtitle: Text('Due to API limitations'.i18n), leading: Icon( Icons.warning, color: Colors.deepOrangeAccent, ), ), FreezerDivider(), Container(height: 16.0,), Text( 'Enter your playlist link below'.i18n, textAlign: TextAlign.center, style: TextStyle( fontSize: 20.0 ), ), Padding( padding: EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0), child: Row( children: [ Expanded( child: TextField( onChanged: (String s) => _url = s, onSubmitted: (String s) { _url = s; _load(); }, decoration: InputDecoration( hintText: 'URL' ), ), ), IconButton( icon: Icon(Icons.search), onPressed: () => _load(), ) ], ), ), Container(height: 8.0,), if (_data == null && _loading) Row( mainAxisAlignment: MainAxisAlignment.center, children: [ CircularProgressIndicator() ], ), if (_error) ListTile( title: Text('Error loading URL!'.i18n), leading: Icon(Icons.error, color: Colors.red,), ), //Playlist if (_data != null) ...[ FreezerDivider(), ListTile( title: Text(_data.name), subtitle: Text((_data.description ?? '') == '' ? '${_data.tracks.length} tracks' : _data.description), leading: Image.network(_data.image??'http://cdn-images.deezer.com/images/cover//256x256-000000-80-0-0.jpg') ), ImporterSettings(), Padding( padding: EdgeInsets.symmetric(vertical: 4.0, horizontal: 16.0), child: ElevatedButton( child: Text('Start import'.i18n), onPressed: () async { await _start(); Navigator.of(context).pushReplacement(MaterialPageRoute( builder: (context) => ImporterStatusScreen() )); }, ), ), ] ], ), ); } } class ImporterSettings extends StatefulWidget { @override _ImporterSettingsState createState() => _ImporterSettingsState(); } class _ImporterSettingsState extends State { @override Widget build(BuildContext context) { return Column( mainAxisSize: MainAxisSize.min, children: [ ListTile( title: Text('Download imported tracks'.i18n), leading: Switch( value: importer.download, onChanged: (v) => setState(() => importer.download = v), ), ), ], ); } } class ImporterStatusScreen extends StatefulWidget { @override _ImporterStatusScreenState createState() => _ImporterStatusScreenState(); } class _ImporterStatusScreenState extends State { bool _done = false; StreamSubscription _subscription; @override void initState() { //If import done mark as not done, to prevent double routing if (importer.done) { _done = true; importer.done = false; } //Update _subscription = importer.updateStream.listen((event) { setState(() { //Unset done so this page doesn't reopen if (importer.done) { _done = true; importer.done = false; }; }); }); super.initState(); } @override void dispose() { if (_subscription != null) _subscription.cancel(); super.dispose(); } @override Widget build(BuildContext context) { return Scaffold( appBar: FreezerAppBar('Importing...'.i18n), body: ListView( children: [ // Spinner if (!_done) Padding( padding: EdgeInsets.symmetric(vertical: 8.0), child: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ CircularProgressIndicator() ], ), ), // Progress indicator Container( child: Row( mainAxisSize: MainAxisSize.max, mainAxisAlignment: MainAxisAlignment.spaceAround, children: [ Row( mainAxisSize: MainAxisSize.min, children: [ Icon(Icons.import_export, size: 24.0,), Container(width: 4.0,), Text('${importer.ok+importer.error}/${importer.tracks.length}', style: TextStyle(fontSize: 24.0),) ], ), Row( mainAxisSize: MainAxisSize.min, children: [ Icon(Icons.done, size: 24.0,), Container(width: 4.0,), Text('${importer.ok}', style: TextStyle(fontSize: 24.0),) ], ), Row( mainAxisSize: MainAxisSize.min, children: [ Icon(Icons.error, size: 24.0,), Container(width: 4.0,), Text('${importer.error}', style: TextStyle(fontSize: 24.0),), ], ), //When Done if (_done) TextButton( child: Text('Playlist menu'.i18n), onPressed: () { MenuSheet m = MenuSheet(context); m.defaultPlaylistMenu(importer.playlist); }, ) ], ), ), Container(height: 8.0), FreezerDivider(), //Tracks ...List.generate(importer.tracks.length, (i) { ImporterTrack t = importer.tracks[i]; return ListTile( leading: t.state.icon, title: Text(t.title), subtitle: Text( t.artists.join(", "), maxLines: 1, ), ); }) ], ), ); } } class SpotifyImporterV2 extends StatefulWidget { @override _SpotifyImporterV2State createState() => _SpotifyImporterV2State(); } class _SpotifyImporterV2State extends State { bool _authorizing = false; String _clientId; String _clientSecret; SpotifyAPIWrapper spotify; //Spotify authorization flow Future _authorize() async { setState(() => _authorizing = true); spotify = SpotifyAPIWrapper(); await spotify.authorize(_clientId, _clientSecret); //Save credentials settings.spotifyClientId = _clientId; settings.spotifyClientSecret = _clientSecret; await settings.save(); setState(() => _authorizing = false); //Redirect Navigator.of(context).pushReplacement(MaterialPageRoute( builder: (context) => SpotifyImporterV2Main(spotify) )); } @override void initState() { _clientId = settings.spotifyClientId; _clientSecret = settings.spotifyClientSecret; super.initState(); } @override void dispose() { //Stop server if (spotify != null) { spotify.cancelAuthorize(); } super.dispose(); } @override Widget build(BuildContext context) { return Scaffold( appBar: FreezerAppBar("Spotify Importer v2".i18n), body: ListView( children: [ Padding( padding: EdgeInsets.symmetric(vertical: 12.0, horizontal: 8.0), child: Text( "This importer requires Spotify Client ID and Client Secret. To obtain them:".i18n, textAlign: TextAlign.center, style: TextStyle( fontSize: 18.0, ), ), ), Padding( padding: EdgeInsets.symmetric(horizontal: 8.0), child: Text( "1. Go to: developer.spotify.com/dashboard and create an app.".i18n, textAlign: TextAlign.center, style: TextStyle( fontSize: 16.0, ), ) ), Padding( padding: EdgeInsets.symmetric(vertical: 4.0, horizontal: 16.0), child: ElevatedButton( child: Text("Open in Browser".i18n), onPressed: () { launch("https://developer.spotify.com/dashboard"); }, ), ), Container(height: 16.0), Padding( padding: EdgeInsets.symmetric(horizontal: 8.0), child: Text( "2. In the app you just created go to settings, and set the Redirect URL to: ".i18n + "http://localhost:42069", textAlign: TextAlign.center, style: TextStyle( fontSize: 16.0, ), ) ), Padding( padding: EdgeInsets.symmetric(vertical: 4.0, horizontal: 16.0), child: ElevatedButton( child: Text("Copy the Redirect URL".i18n), onPressed: () async { await Clipboard.setData(new ClipboardData(text: "http://localhost:42069")); Fluttertoast.showToast(msg: "Copied".i18n, gravity: ToastGravity.BOTTOM, toastLength: Toast.LENGTH_SHORT); }, ), ), Padding( padding: EdgeInsets.symmetric(horizontal: 16.0, vertical: 16.0), child: Row( mainAxisSize: MainAxisSize.max, children: [ Flexible( child: TextField( controller: TextEditingController(text: _clientId), decoration: InputDecoration( labelText: "Client ID".i18n ), onChanged: (v) => setState(() => _clientId = v), ), ), Container(width: 16.0), Flexible( child: TextField( controller: TextEditingController(text: _clientSecret), obscureText: true, decoration: InputDecoration( labelText: "Client Secret".i18n ), onChanged: (v) => setState(() => _clientSecret = v), ), ), ], ), ), Padding( padding: EdgeInsets.symmetric(vertical: 4.0, horizontal: 16.0), child: ElevatedButton( child: Text("Authorize".i18n), onPressed: (_clientId != null && _clientSecret != null && !_authorizing) ? () => _authorize() : null ), ), if (_authorizing) Padding( padding: EdgeInsets.symmetric(vertical: 8.0), child: Row( mainAxisAlignment: MainAxisAlignment.center, children: [ CircularProgressIndicator() ], ), ) ], ), ); } } class SpotifyImporterV2Main extends StatefulWidget { SpotifyAPIWrapper spotify; SpotifyImporterV2Main(this.spotify, {Key key}): super(key: key); @override _SpotifyImporterV2MainState createState() => _SpotifyImporterV2MainState(); } class _SpotifyImporterV2MainState extends State { String _url; bool _urlLoading = false; spotify.Playlist _urlPlaylist; bool _playlistsLoading = true; List _playlists; @override void initState() { _loadPlaylists(); super.initState(); } //Load playlists Future _loadPlaylists() async { var pages = widget.spotify.spotify.users.playlists(widget.spotify.me.id); _playlists = List.from(await pages.all()); setState(() => _playlistsLoading = false); } Future _loadUrl() async { setState(() => _urlLoading = true); //Resolve URL try { String uri = await SpotifyScrapper.resolveUrl(_url); //Error/NonPlaylist if (uri == null || uri.split(':')[1] != 'playlist') { throw Exception(); } //Get playlist spotify.Playlist playlist = await widget.spotify.spotify.playlists.get(uri.split(":")[2]); setState(() { _urlLoading = false; _urlPlaylist = playlist; }); } catch (e) { Fluttertoast.showToast(msg: "Invalid/Unsupported URL".i18n, gravity: ToastGravity.BOTTOM, toastLength: Toast.LENGTH_SHORT); setState(() => _urlLoading = false); return; } } Future _startImport(String title, String description, String id) async { //Show loading dialog showDialog( context: context, barrierDismissible: false, builder: (context) => WillPopScope( onWillPop: () => Future.value(false), child: AlertDialog( title: Text("Please wait...".i18n), content: Row( mainAxisAlignment: MainAxisAlignment.center, children: [CircularProgressIndicator()], ) ) ) ); try { //Fetch entire playlist var pages = widget.spotify.spotify.playlists.getTracksByPlaylistId(id); var all = await pages.all(); //Map to importer track List tracks = all.map((t) => ImporterTrack(t.name, t.artists.map((a) => a.name).toList(), isrc: t.externalIds.isrc)).toList(); await importer.start(context, title, description, tracks); //Route Navigator.of(context).pop(); Navigator.of(context).pushReplacement(MaterialPageRoute( builder: (context) => ImporterStatusScreen() )); } catch (e) { Fluttertoast.showToast(msg: e.toString(), gravity: ToastGravity.BOTTOM, toastLength: Toast.LENGTH_SHORT); Navigator.of(context).pop(); return; } } @override Widget build(BuildContext context) { return Scaffold( appBar: FreezerAppBar("Spotify Importer v2".i18n), body: ListView( children: [ Padding( padding: EdgeInsets.symmetric(vertical: 8.0, horizontal: 16.0), child: Text( 'Logged in as: '.i18n + widget.spotify.me.displayName, maxLines: 1, textAlign: TextAlign.center, style: TextStyle( fontSize: 18.0, fontWeight: FontWeight.bold ) ), ), FreezerDivider(), Container(height: 4.0), Text( "Options".i18n, textAlign: TextAlign.center, style: TextStyle( fontSize: 18.0, fontWeight: FontWeight.bold ), ), ImporterSettings(), FreezerDivider(), Container(height: 4.0), Text( "Import playlists by URL".i18n, textAlign: TextAlign.center, style: TextStyle( fontSize: 18.0, fontWeight: FontWeight.bold ), ), Padding( padding: EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0), child: Row( children: [ Expanded( child: TextField( decoration: InputDecoration( hintText: "URL".i18n ), onChanged: (v) => setState(() => _url = v) ), ), IconButton( icon: Icon(Icons.search), onPressed: () => _loadUrl(), ) ], ) ), if (_urlLoading) Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Padding( padding: EdgeInsets.symmetric(vertical: 4.0), child: CircularProgressIndicator(), ) ], ), if (_urlPlaylist != null) ListTile( title: Text(_urlPlaylist.name), subtitle: Text(_urlPlaylist.description ?? ''), leading: Image.network(_urlPlaylist.images.first?.url??"http://cdn-images.deezer.com/images/cover//256x256-000000-80-0-0.jpg") ), if (_urlPlaylist != null) Padding( padding: EdgeInsets.symmetric(horizontal: 16.0), child: ElevatedButton( child: Text("Import".i18n), onPressed: () { _startImport(_urlPlaylist.name, _urlPlaylist.description, _urlPlaylist.id); } ) ), // Playlists FreezerDivider(), Container(height: 4.0), Text( "Playlists".i18n, textAlign: TextAlign.center, style: TextStyle( fontSize: 18.0, fontWeight: FontWeight.bold ) ), Container(height: 4.0), if (_playlistsLoading) Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Padding( padding: EdgeInsets.symmetric(vertical: 4.0), child: CircularProgressIndicator(), ) ], ), if (!_playlistsLoading && _playlists != null) ...List.generate(_playlists.length, (i) { spotify.PlaylistSimple p = _playlists[i]; return ListTile( title: Text(p.name, maxLines: 1), subtitle: Text(p.owner.displayName, maxLines: 1), leading: Image.network(p.images.first?.url??"http://cdn-images.deezer.com/images/cover//256x256-000000-80-0-0.jpg"), onTap: () { _startImport(p.name, "", p.id); }, ); }) ], ) ); } }