2023-06-23 16:23:46 -04:00
|
|
|
import 'dart:isolate';
|
|
|
|
import 'dart:ui';
|
|
|
|
|
|
|
|
import 'package:file_picker/file_picker.dart';
|
2023-06-17 10:58:32 -04:00
|
|
|
import 'package:flutter/material.dart';
|
2023-06-23 16:23:46 -04:00
|
|
|
import 'package:flutter_downloader/flutter_downloader.dart';
|
2023-10-21 08:02:28 -04:00
|
|
|
import 'package:intl/intl.dart';
|
2023-06-23 16:23:46 -04:00
|
|
|
import 'package:path/path.dart';
|
|
|
|
import 'package:provider/provider.dart';
|
2023-10-12 12:33:04 -04:00
|
|
|
import 'package:recon/auxiliary.dart';
|
|
|
|
import 'package:recon/clients/inventory_client.dart';
|
|
|
|
import 'package:share_plus/share_plus.dart';
|
2023-06-17 10:58:32 -04:00
|
|
|
|
2023-06-23 16:23:46 -04:00
|
|
|
class InventoryBrowserAppBar extends StatefulWidget {
|
2023-06-17 10:58:32 -04:00
|
|
|
const InventoryBrowserAppBar({super.key});
|
|
|
|
|
2023-06-23 16:23:46 -04:00
|
|
|
@override
|
|
|
|
State<InventoryBrowserAppBar> createState() => _InventoryBrowserAppBarState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _InventoryBrowserAppBarState extends State<InventoryBrowserAppBar> {
|
|
|
|
final ReceivePort _port = ReceivePort();
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
|
|
|
|
IsolateNameServer.registerPortWithName(_port.sendPort, 'downloader_send_port');
|
|
|
|
_port.listen((dynamic data) {
|
|
|
|
// Not useful yet? idk...
|
2023-10-31 16:50:47 -04:00
|
|
|
// String id = data[0];
|
|
|
|
// DownloadTaskStatus status = data[1];
|
|
|
|
// int progress = data[2];
|
2023-06-23 16:23:46 -04:00
|
|
|
});
|
|
|
|
|
|
|
|
FlutterDownloader.registerCallback(downloadCallback);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
IsolateNameServer.removePortNameMapping('downloader_send_port');
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
@pragma('vm:entry-point')
|
|
|
|
static void downloadCallback(String id, int status, int progress) {
|
|
|
|
final SendPort? send = IsolateNameServer.lookupPortByName('downloader_send_port');
|
|
|
|
send?.send([id, status, progress]);
|
|
|
|
}
|
|
|
|
|
2023-06-17 10:58:32 -04:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
2023-07-11 12:44:20 -04:00
|
|
|
return Consumer<InventoryClient>(
|
2023-10-21 08:02:28 -04:00
|
|
|
builder: (BuildContext context, InventoryClient iClient, Widget? _) {
|
2023-07-11 12:44:20 -04:00
|
|
|
return AnimatedSwitcher(
|
|
|
|
duration: const Duration(milliseconds: 350),
|
|
|
|
transitionBuilder: (child, animation) => FadeTransition(
|
|
|
|
opacity: animation,
|
|
|
|
child: child,
|
|
|
|
),
|
|
|
|
child: !iClient.isAnyRecordSelected
|
|
|
|
? AppBar(
|
|
|
|
key: const ValueKey("default-appbar"),
|
|
|
|
title: const Text("Inventory"),
|
2023-10-21 08:02:28 -04:00
|
|
|
actions: [
|
|
|
|
PopupMenuButton(
|
|
|
|
icon: const Icon(Icons.swap_vert),
|
|
|
|
onSelected: (value) {
|
|
|
|
iClient.sortReverse = value;
|
|
|
|
},
|
|
|
|
itemBuilder: (context) {
|
|
|
|
return [
|
|
|
|
PopupMenuItem(
|
|
|
|
value: false,
|
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
Icon(
|
|
|
|
Icons.arrow_upward,
|
|
|
|
color: iClient.sortReverse == false
|
|
|
|
? Theme.of(context).colorScheme.primary
|
|
|
|
: Theme.of(context).colorScheme.onSurface,
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
width: 8,
|
|
|
|
),
|
|
|
|
Text(
|
|
|
|
"Ascending",
|
|
|
|
style: TextStyle(
|
|
|
|
color: iClient.sortReverse == false
|
|
|
|
? Theme.of(context).colorScheme.primary
|
|
|
|
: Theme.of(context).colorScheme.onSurface,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
PopupMenuItem(
|
|
|
|
value: true,
|
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
Icon(Icons.arrow_downward,
|
|
|
|
color: iClient.sortReverse == true
|
|
|
|
? Theme.of(context).colorScheme.primary
|
|
|
|
: Theme.of(context).colorScheme.onSurface),
|
|
|
|
const SizedBox(
|
|
|
|
width: 8,
|
|
|
|
),
|
|
|
|
Text(
|
|
|
|
"Descending",
|
|
|
|
style: TextStyle(
|
|
|
|
color: iClient.sortReverse == true
|
|
|
|
? Theme.of(context).colorScheme.primary
|
|
|
|
: Theme.of(context).colorScheme.onSurface,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
)
|
|
|
|
];
|
|
|
|
},
|
|
|
|
),
|
|
|
|
Padding(
|
|
|
|
padding: const EdgeInsets.only(right: 8.0),
|
|
|
|
child: PopupMenuButton(
|
|
|
|
icon: const Icon(Icons.sort),
|
|
|
|
onSelected: (value) {
|
|
|
|
iClient.sortMode = value;
|
|
|
|
},
|
|
|
|
itemBuilder: (context) {
|
|
|
|
return SortMode.values
|
|
|
|
.map(
|
|
|
|
(e) => PopupMenuItem(
|
|
|
|
value: e,
|
|
|
|
child: Row(
|
|
|
|
children: [
|
|
|
|
Icon(
|
|
|
|
e.icon,
|
|
|
|
color: iClient.sortMode == e
|
|
|
|
? Theme.of(context).colorScheme.primary
|
|
|
|
: Theme.of(context).colorScheme.onSurface,
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
width: 8,
|
|
|
|
),
|
|
|
|
Text(
|
|
|
|
toBeginningOfSentenceCase(e.name) ?? e.name,
|
|
|
|
style: TextStyle(
|
|
|
|
color: iClient.sortMode == e
|
|
|
|
? Theme.of(context).colorScheme.primary
|
|
|
|
: Theme.of(context).colorScheme.onSurface,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.toList();
|
|
|
|
},
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
2023-07-11 12:44:20 -04:00
|
|
|
)
|
|
|
|
: AppBar(
|
|
|
|
key: const ValueKey("selection-appbar"),
|
|
|
|
title: Text("${iClient.selectedRecordCount} Selected"),
|
|
|
|
leading: IconButton(
|
|
|
|
onPressed: () {
|
|
|
|
iClient.clearSelectedRecords();
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.close),
|
|
|
|
),
|
|
|
|
actions: [
|
2023-10-12 12:33:04 -04:00
|
|
|
if (iClient.selectedRecordCount == 1 &&
|
|
|
|
(iClient.selectedRecords.firstOrNull?.isLink == true ||
|
|
|
|
iClient.selectedRecords.firstOrNull?.isItem == true))
|
|
|
|
IconButton(
|
|
|
|
onPressed: () {
|
|
|
|
Share.share(iClient.selectedRecords.first.assetUri);
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.share),
|
|
|
|
),
|
2023-07-11 12:44:20 -04:00
|
|
|
if (iClient.onlyFilesSelected)
|
|
|
|
IconButton(
|
|
|
|
onPressed: () async {
|
|
|
|
final selectedRecords = iClient.selectedRecords;
|
2023-06-23 16:23:46 -04:00
|
|
|
|
2023-07-11 12:44:20 -04:00
|
|
|
final assetUris = selectedRecords.map((record) => record.assetUri).toList();
|
|
|
|
final thumbUris = selectedRecords.map((record) => record.thumbnailUri).toList();
|
2023-06-23 16:23:46 -04:00
|
|
|
|
2023-07-11 12:44:20 -04:00
|
|
|
final selectedUris = await showDialog<List<String>>(
|
|
|
|
context: context,
|
|
|
|
builder: (context) {
|
|
|
|
return AlertDialog(
|
|
|
|
icon: const Icon(Icons.download),
|
|
|
|
title: const Text("Download what?"),
|
|
|
|
content: Column(
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
|
|
|
const Divider(),
|
|
|
|
const SizedBox(
|
|
|
|
height: 8,
|
|
|
|
),
|
|
|
|
TextButton.icon(
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop(assetUris);
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.data_object),
|
|
|
|
label: Text(
|
|
|
|
"Asset${iClient.selectedRecordCount != 1 ? "s" : ""} (${assetUris.map((e) => extension(e)).toList().unique().join(", ")})",
|
2023-06-23 16:23:46 -04:00
|
|
|
),
|
2023-07-11 12:44:20 -04:00
|
|
|
),
|
|
|
|
TextButton.icon(
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop(thumbUris);
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.image),
|
|
|
|
label: Text(
|
|
|
|
"Thumbnail${iClient.selectedRecordCount != 1 ? "s" : ""} (${thumbUris.map((e) => extension(e)).toList().unique().join(", ")})",
|
2023-06-23 16:23:46 -04:00
|
|
|
),
|
2023-07-11 12:44:20 -04:00
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
if (selectedUris == null) return;
|
2023-06-23 16:23:46 -04:00
|
|
|
|
2023-07-11 12:44:20 -04:00
|
|
|
final directory = await FilePicker.platform.getDirectoryPath(dialogTitle: "Download to...");
|
|
|
|
if (directory == null) {
|
|
|
|
if (context.mounted) {
|
|
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
|
|
const SnackBar(
|
|
|
|
content: Text("Selection aborted."),
|
|
|
|
),
|
|
|
|
);
|
2023-06-23 16:23:46 -04:00
|
|
|
}
|
2023-07-11 12:44:20 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (directory == "/") {
|
|
|
|
if (context.mounted) {
|
|
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
|
|
const SnackBar(
|
|
|
|
content: Text("Selected directory is invalid"),
|
|
|
|
),
|
2023-06-23 16:23:46 -04:00
|
|
|
);
|
|
|
|
}
|
2023-07-11 12:44:20 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (var record in selectedRecords) {
|
2023-11-01 10:22:58 -04:00
|
|
|
final uri = selectedUris == thumbUris ? record.thumbnailUri : record.assetUri;
|
2023-07-11 12:44:20 -04:00
|
|
|
await FlutterDownloader.enqueue(
|
2023-09-29 03:51:46 -04:00
|
|
|
url: Aux.resdbToHttp(uri),
|
2023-07-11 12:44:20 -04:00
|
|
|
savedDir: directory,
|
|
|
|
showNotification: true,
|
|
|
|
openFileFromNotification: false,
|
|
|
|
fileName:
|
|
|
|
"${record.id.split("-")[1]}-${record.formattedName.toString()}${extension(uri)}",
|
|
|
|
);
|
|
|
|
}
|
|
|
|
iClient.clearSelectedRecords();
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.download),
|
2023-06-23 16:23:46 -04:00
|
|
|
),
|
2023-07-11 12:44:20 -04:00
|
|
|
const SizedBox(
|
|
|
|
width: 4,
|
|
|
|
),
|
|
|
|
IconButton(
|
|
|
|
onPressed: () async {
|
|
|
|
var loading = false;
|
|
|
|
await showDialog(
|
|
|
|
context: context,
|
|
|
|
builder: (context) {
|
|
|
|
return StatefulBuilder(
|
|
|
|
builder: (context, setState) {
|
|
|
|
return AlertDialog(
|
|
|
|
icon: const Icon(Icons.delete),
|
|
|
|
title: Text(iClient.selectedRecordCount == 1
|
|
|
|
? "Really delete this Record?"
|
|
|
|
: "Really delete ${iClient.selectedRecordCount} Records?"),
|
|
|
|
content: const Text("This action cannot be undone!"),
|
|
|
|
actionsAlignment: MainAxisAlignment.spaceBetween,
|
|
|
|
actions: [
|
|
|
|
TextButton(
|
|
|
|
onPressed: loading
|
|
|
|
? null
|
|
|
|
: () {
|
|
|
|
Navigator.of(context).pop(false);
|
|
|
|
},
|
|
|
|
child: const Text("Cancel"),
|
|
|
|
),
|
|
|
|
Row(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
|
|
|
if (loading)
|
|
|
|
const SizedBox.square(
|
|
|
|
dimension: 16,
|
|
|
|
child: CircularProgressIndicator(strokeWidth: 2),
|
2023-06-23 16:23:46 -04:00
|
|
|
),
|
2023-07-11 12:44:20 -04:00
|
|
|
const SizedBox(
|
|
|
|
width: 4,
|
|
|
|
),
|
|
|
|
TextButton(
|
|
|
|
onPressed: loading
|
|
|
|
? null
|
|
|
|
: () async {
|
|
|
|
setState(() {
|
|
|
|
loading = true;
|
|
|
|
});
|
|
|
|
try {
|
|
|
|
await iClient.deleteSelectedRecords();
|
|
|
|
} catch (e) {
|
2023-06-23 16:23:46 -04:00
|
|
|
if (context.mounted) {
|
2023-07-11 12:44:20 -04:00
|
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
|
|
SnackBar(
|
|
|
|
content: Text("Failed to delete one or more records: $e"),
|
|
|
|
),
|
|
|
|
);
|
2023-06-23 16:23:46 -04:00
|
|
|
}
|
2023-07-11 12:44:20 -04:00
|
|
|
setState(() {
|
|
|
|
loading = false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
if (context.mounted) {
|
|
|
|
Navigator.of(context).pop(true);
|
|
|
|
}
|
|
|
|
iClient.reloadCurrentDirectory();
|
|
|
|
},
|
|
|
|
style: TextButton.styleFrom(
|
|
|
|
foregroundColor: Theme.of(context).colorScheme.error,
|
2023-06-23 16:23:46 -04:00
|
|
|
),
|
2023-07-11 12:44:20 -04:00
|
|
|
child: const Text("Delete"),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
},
|
|
|
|
icon: const Icon(Icons.delete),
|
|
|
|
),
|
|
|
|
const SizedBox(
|
|
|
|
width: 4,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
2023-06-17 10:58:32 -04:00
|
|
|
);
|
|
|
|
}
|
2023-06-23 16:23:46 -04:00
|
|
|
}
|