158 lines
5.3 KiB
Dart
158 lines
5.3 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:aves/theme/durations.dart';
|
|
import 'package:aves/utils/android_file_utils.dart';
|
|
import 'package:aves/widgets/common/extensions/build_context.dart';
|
|
import 'package:collection/collection.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/widgets.dart';
|
|
import 'package:path/path.dart';
|
|
|
|
import 'aves_dialog.dart';
|
|
|
|
class CreateAlbumDialog extends StatefulWidget {
|
|
@override
|
|
_CreateAlbumDialogState createState() => _CreateAlbumDialogState();
|
|
}
|
|
|
|
class _CreateAlbumDialogState extends State<CreateAlbumDialog> {
|
|
final ScrollController _scrollController = ScrollController();
|
|
final TextEditingController _nameController = TextEditingController();
|
|
final FocusNode _nameFieldFocusNode = FocusNode();
|
|
final ValueNotifier<bool> _existsNotifier = ValueNotifier(false);
|
|
final ValueNotifier<bool> _isValidNotifier = ValueNotifier(false);
|
|
Set<StorageVolume> _allVolumes;
|
|
StorageVolume _primaryVolume, _selectedVolume;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
_allVolumes = androidFileUtils.storageVolumes;
|
|
_primaryVolume = _allVolumes.firstWhere((volume) => volume.isPrimary, orElse: () => _allVolumes.first);
|
|
_selectedVolume = _primaryVolume;
|
|
_nameFieldFocusNode.addListener(_onFocus);
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_nameController.dispose();
|
|
_nameFieldFocusNode.removeListener(_onFocus);
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final volumeTiles = <Widget>[];
|
|
if (_allVolumes.length > 1) {
|
|
final byPrimary = groupBy<StorageVolume, bool>(_allVolumes, (volume) => volume.isPrimary);
|
|
int compare(StorageVolume a, StorageVolume b) => compareAsciiUpperCase(a.path, b.path);
|
|
final primaryVolumes = byPrimary[true]..sort(compare);
|
|
final otherVolumes = byPrimary[false]..sort(compare);
|
|
volumeTiles.addAll([
|
|
Padding(
|
|
padding: AvesDialog.contentHorizontalPadding + EdgeInsets.only(top: 20),
|
|
child: Text(context.l10n.newAlbumDialogStorageLabel),
|
|
),
|
|
...primaryVolumes.map((volume) => _buildVolumeTile(context, volume)),
|
|
...otherVolumes.map((volume) => _buildVolumeTile(context, volume)),
|
|
SizedBox(height: 8),
|
|
]);
|
|
}
|
|
|
|
return AvesDialog(
|
|
context: context,
|
|
title: context.l10n.newAlbumDialogTitle,
|
|
scrollController: _scrollController,
|
|
scrollableContent: [
|
|
...volumeTiles,
|
|
Padding(
|
|
padding: AvesDialog.contentHorizontalPadding + EdgeInsets.only(bottom: 8),
|
|
child: ValueListenableBuilder<bool>(
|
|
valueListenable: _existsNotifier,
|
|
builder: (context, exists, child) {
|
|
return TextField(
|
|
controller: _nameController,
|
|
focusNode: _nameFieldFocusNode,
|
|
decoration: InputDecoration(
|
|
labelText: context.l10n.newAlbumDialogNameLabel,
|
|
helperText: exists ? context.l10n.newAlbumDialogNameLabelAlreadyExistsHelper : '',
|
|
),
|
|
autofocus: _allVolumes.length == 1,
|
|
onChanged: (_) => _validate(),
|
|
onSubmitted: (_) => _submit(context),
|
|
);
|
|
}),
|
|
),
|
|
],
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () => Navigator.pop(context),
|
|
child: Text(MaterialLocalizations.of(context).cancelButtonLabel),
|
|
),
|
|
ValueListenableBuilder<bool>(
|
|
valueListenable: _isValidNotifier,
|
|
builder: (context, isValid, child) {
|
|
return TextButton(
|
|
onPressed: isValid ? () => _submit(context) : null,
|
|
child: Text(context.l10n.createAlbumButtonLabel),
|
|
);
|
|
},
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
Widget _buildVolumeTile(BuildContext context, StorageVolume volume) => RadioListTile<StorageVolume>(
|
|
value: volume,
|
|
groupValue: _selectedVolume,
|
|
onChanged: (volume) {
|
|
_selectedVolume = volume;
|
|
_validate();
|
|
setState(() {});
|
|
},
|
|
title: Text(
|
|
volume.getDescription(context),
|
|
softWrap: false,
|
|
overflow: TextOverflow.fade,
|
|
maxLines: 1,
|
|
),
|
|
subtitle: Text(
|
|
volume.path,
|
|
softWrap: false,
|
|
overflow: TextOverflow.fade,
|
|
maxLines: 1,
|
|
),
|
|
);
|
|
|
|
void _onFocus() async {
|
|
// when the field gets focus, we wait for the soft keyboard to appear
|
|
// then scroll to the bottom to make sure the field is in view
|
|
if (_nameFieldFocusNode.hasFocus) {
|
|
await Future.delayed(Durations.softKeyboardDisplayDelay);
|
|
_scrollToBottom();
|
|
}
|
|
}
|
|
|
|
void _scrollToBottom() {
|
|
_scrollController.animateTo(
|
|
_scrollController.position.maxScrollExtent,
|
|
duration: Durations.dialogFieldReachAnimation,
|
|
curve: Curves.easeInOut,
|
|
);
|
|
}
|
|
|
|
String _buildAlbumPath(String name) {
|
|
if (name == null || name.isEmpty) return '';
|
|
return join(_selectedVolume.path, 'Pictures', name);
|
|
}
|
|
|
|
Future<void> _validate() async {
|
|
final newName = _nameController.text ?? '';
|
|
final path = _buildAlbumPath(newName);
|
|
final exists = newName.isNotEmpty && await Directory(path).exists();
|
|
_existsNotifier.value = exists;
|
|
_isValidNotifier.value = newName.isNotEmpty;
|
|
}
|
|
|
|
void _submit(BuildContext context) => Navigator.pop(context, _buildAlbumPath(_nameController.text));
|
|
}
|