aves/lib/widgets/common/basic/query_bar.dart
2021-11-04 10:44:31 +09:00

95 lines
2.9 KiB
Dart

import 'package:aves/theme/durations.dart';
import 'package:aves/theme/icons.dart';
import 'package:aves/utils/debouncer.dart';
import 'package:aves/widgets/common/extensions/build_context.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/widgets.dart';
class QueryBar extends StatefulWidget {
final ValueNotifier<String> queryNotifier;
final FocusNode? focusNode;
final IconData? icon;
final String? hintText;
final bool editable;
const QueryBar({
Key? key,
required this.queryNotifier,
this.focusNode,
this.icon,
this.hintText,
this.editable = true,
}) : super(key: key);
@override
_QueryBarState createState() => _QueryBarState();
}
class _QueryBarState extends State<QueryBar> {
final Debouncer _debouncer = Debouncer(delay: Durations.searchDebounceDelay);
late TextEditingController _controller;
ValueNotifier<String> get queryNotifier => widget.queryNotifier;
@override
void initState() {
super.initState();
_controller = TextEditingController(text: queryNotifier.value);
}
@override
Widget build(BuildContext context) {
final clearButton = IconButton(
icon: const Icon(AIcons.clear),
onPressed: widget.editable
? () {
_controller.clear();
queryNotifier.value = '';
}
: null,
tooltip: context.l10n.clearTooltip,
);
return Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Expanded(
child: TextField(
controller: _controller,
focusNode: widget.focusNode ?? FocusNode(),
decoration: InputDecoration(
icon: Padding(
padding: const EdgeInsetsDirectional.only(start: 16),
child: Icon(widget.icon ?? AIcons.filter),
),
hintText: widget.hintText ?? MaterialLocalizations.of(context).searchFieldLabel,
hintStyle: Theme.of(context).inputDecorationTheme.hintStyle,
),
textInputAction: TextInputAction.search,
onChanged: (s) => _debouncer(() => queryNotifier.value = s.trim()),
enabled: widget.editable,
),
),
ConstrainedBox(
constraints: const BoxConstraints(minWidth: 16),
child: ValueListenableBuilder<TextEditingValue>(
valueListenable: _controller,
builder: (context, value, child) => AnimatedSwitcher(
duration: Durations.appBarActionChangeAnimation,
transitionBuilder: (child, animation) => FadeTransition(
opacity: animation,
child: SizeTransition(
axis: Axis.horizontal,
sizeFactor: animation,
child: child,
),
),
child: value.text.isNotEmpty ? clearButton : const SizedBox(),
),
),
)
],
);
}
}