mirror of
https://github.com/johrpan/musicus_mobile.git
synced 2025-10-26 10:47:25 +01:00
Add basic files selector
This required to add a platform channel with a method to get the toplevel storage devices. The app will need to request the WRITE_EXTERNAL_STORAGE permission later. Also the dependency on "path" was introduced.
This commit is contained in:
parent
02e283b8cc
commit
24a4911665
4 changed files with 213 additions and 2 deletions
|
|
@ -1,6 +1,8 @@
|
||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
package="de.johrpan.musicus">
|
package="de.johrpan.musicus">
|
||||||
|
|
||||||
|
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
|
||||||
|
|
||||||
<application
|
<application
|
||||||
android:name="io.flutter.app.FlutterApplication"
|
android:name="io.flutter.app.FlutterApplication"
|
||||||
android:label="Musicus"
|
android:label="Musicus"
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,39 @@
|
||||||
package de.johrpan.musicus
|
package de.johrpan.musicus
|
||||||
|
|
||||||
import androidx.annotation.NonNull;
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.core.content.ContextCompat
|
||||||
import io.flutter.embedding.android.FlutterActivity
|
import io.flutter.embedding.android.FlutterActivity
|
||||||
import io.flutter.embedding.engine.FlutterEngine
|
import io.flutter.embedding.engine.FlutterEngine
|
||||||
|
import io.flutter.plugin.common.MethodChannel
|
||||||
import io.flutter.plugins.GeneratedPluginRegistrant
|
import io.flutter.plugins.GeneratedPluginRegistrant
|
||||||
|
|
||||||
class MainActivity: FlutterActivity() {
|
class MainActivity : FlutterActivity() {
|
||||||
|
private val CHANNEL = "de.johrpan.musicus/platform"
|
||||||
|
|
||||||
override fun configureFlutterEngine(@NonNull flutterEngine: FlutterEngine) {
|
override fun configureFlutterEngine(@NonNull flutterEngine: FlutterEngine) {
|
||||||
GeneratedPluginRegistrant.registerWith(flutterEngine);
|
GeneratedPluginRegistrant.registerWith(flutterEngine)
|
||||||
|
|
||||||
|
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, CHANNEL).setMethodCallHandler { call, result ->
|
||||||
|
if (call.method == "getStorageRoots") {
|
||||||
|
result.success(getStorageRoots())
|
||||||
|
} else {
|
||||||
|
result.notImplemented()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getStorageRoots(): ArrayList<String> {
|
||||||
|
val result = ArrayList<String>()
|
||||||
|
|
||||||
|
ContextCompat.getExternalFilesDirs(this, null).forEach {
|
||||||
|
val path = it.absolutePath;
|
||||||
|
val index = path.lastIndexOf("/Android/data/")
|
||||||
|
|
||||||
|
if (index > 0) {
|
||||||
|
result.add(path.substring(0, index))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
181
lib/selectors/files.dart
Normal file
181
lib/selectors/files.dart
Normal file
|
|
@ -0,0 +1,181 @@
|
||||||
|
import 'dart:io';
|
||||||
|
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
|
import 'package:path/path.dart' as path;
|
||||||
|
|
||||||
|
class FilesSelector extends StatefulWidget {
|
||||||
|
@override
|
||||||
|
_FilesSelectorState createState() => _FilesSelectorState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _FilesSelectorState extends State<FilesSelector> {
|
||||||
|
static const platform = MethodChannel('de.johrpan.musicus/platform');
|
||||||
|
|
||||||
|
List<Directory> storageRoots = [];
|
||||||
|
List<Directory> directories = [];
|
||||||
|
List<FileSystemEntity> contents = [];
|
||||||
|
Set<String> selectedPaths = {};
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
|
||||||
|
platform.invokeListMethod<String>('getStorageRoots').then((sr) {
|
||||||
|
setState(() {
|
||||||
|
storageRoots = sr.map((path) => Directory(path)).toList();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
Widget body;
|
||||||
|
|
||||||
|
if (directories.isEmpty) {
|
||||||
|
if (storageRoots != null) {
|
||||||
|
body = ListView(
|
||||||
|
children: storageRoots
|
||||||
|
.map((dir) => ListTile(
|
||||||
|
leading: const Icon(Icons.storage),
|
||||||
|
title: Text(dir.path),
|
||||||
|
onTap: () {
|
||||||
|
setState(() {
|
||||||
|
directories.add(dir);
|
||||||
|
});
|
||||||
|
|
||||||
|
openDirectory(dir);
|
||||||
|
},
|
||||||
|
))
|
||||||
|
.toList(),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
body = Container();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (contents != null) {
|
||||||
|
body = ListView(
|
||||||
|
children: contents.map((fse) {
|
||||||
|
Widget result;
|
||||||
|
|
||||||
|
if (fse is Directory) {
|
||||||
|
result = ListTile(
|
||||||
|
leading: const Icon(Icons.folder),
|
||||||
|
title: Text(path.basename(fse.path)),
|
||||||
|
onTap: () {
|
||||||
|
setState(() {
|
||||||
|
directories.add(fse);
|
||||||
|
});
|
||||||
|
|
||||||
|
openDirectory(fse);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
} else if (fse is File) {
|
||||||
|
result = CheckboxListTile(
|
||||||
|
value: selectedPaths.contains(fse.path),
|
||||||
|
secondary: Icon(Icons.insert_drive_file),
|
||||||
|
title: Text(path.basename(fse.path)),
|
||||||
|
onChanged: (selected) {
|
||||||
|
setState(() {
|
||||||
|
if (selected) {
|
||||||
|
selectedPaths.add(fse.path);
|
||||||
|
} else {
|
||||||
|
selectedPaths.remove(fse.path);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}).toList(),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
body = Container();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return WillPopScope(
|
||||||
|
child: Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text('Choose files'),
|
||||||
|
actions: <Widget>[
|
||||||
|
FlatButton(
|
||||||
|
child: Text('DONE'),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.pop(context, selectedPaths);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
body: Column(
|
||||||
|
children: <Widget>[
|
||||||
|
Material(
|
||||||
|
elevation: 2.0,
|
||||||
|
child: ListTile(
|
||||||
|
leading: directories.isNotEmpty ? IconButton(
|
||||||
|
icon: const Icon(Icons.arrow_upward),
|
||||||
|
onPressed: up,
|
||||||
|
) : null,
|
||||||
|
title: Text(directories.isEmpty
|
||||||
|
? 'Storage devices'
|
||||||
|
: directories.last.path),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Expanded(
|
||||||
|
child: body,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onWillPop: () {
|
||||||
|
if (directories.isNotEmpty) {
|
||||||
|
up();
|
||||||
|
return Future.value(false);
|
||||||
|
} else {
|
||||||
|
return Future.value(true);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> openDirectory(Directory directory) async {
|
||||||
|
setState(() {
|
||||||
|
contents.clear();
|
||||||
|
});
|
||||||
|
|
||||||
|
final fses = await directory.list().toList();
|
||||||
|
fses.sort((fse1, fse2) {
|
||||||
|
int compareBasenames() =>
|
||||||
|
path.basename(fse1.path).compareTo(path.basename(fse2.path));
|
||||||
|
|
||||||
|
if (fse1 is Directory) {
|
||||||
|
if (fse2 is Directory) {
|
||||||
|
return compareBasenames();
|
||||||
|
} else {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
} else if (fse2 is Directory) {
|
||||||
|
return 1;
|
||||||
|
} else {
|
||||||
|
return compareBasenames();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
contents = fses;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void up() {
|
||||||
|
if (directories.isNotEmpty) {
|
||||||
|
setState(() {
|
||||||
|
directories.removeLast();
|
||||||
|
});
|
||||||
|
|
||||||
|
if (directories.isNotEmpty) {
|
||||||
|
openDirectory(directories.last);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -12,6 +12,7 @@ dependencies:
|
||||||
flutter:
|
flutter:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
moor_flutter:
|
moor_flutter:
|
||||||
|
path:
|
||||||
rxdart:
|
rxdart:
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue