2019-04-27 14:40:27 +02:00
|
|
|
import 'package:comunic/helpers/search_helper.dart';
|
|
|
|
import 'package:comunic/lists/users_list.dart';
|
|
|
|
import 'package:comunic/models/user.dart';
|
|
|
|
import 'package:comunic/ui/tiles/simple_user_tile.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
|
|
|
|
/// Pick user widget
|
|
|
|
///
|
|
|
|
/// Allows to choose a user by starting to type his name and tapping on it
|
|
|
|
/// when his name appear
|
|
|
|
///
|
|
|
|
/// @author Pierre HUBERT
|
|
|
|
|
|
|
|
typedef OnSelectUserCallback = void Function(User);
|
|
|
|
|
|
|
|
class PickUserWidget extends StatefulWidget {
|
|
|
|
final OnSelectUserCallback onSelectUser;
|
2020-05-02 15:30:19 +02:00
|
|
|
final void Function(String) onValueChange;
|
2019-04-27 16:23:08 +02:00
|
|
|
final String label;
|
|
|
|
final bool resetOnChoose;
|
|
|
|
final bool keepFocusOnChoose;
|
2019-05-01 09:24:50 +02:00
|
|
|
final bool enabled;
|
2019-04-27 16:23:08 +02:00
|
|
|
|
2020-05-02 15:30:19 +02:00
|
|
|
const PickUserWidget({
|
|
|
|
Key key,
|
|
|
|
@required this.onSelectUser,
|
|
|
|
@required this.label,
|
|
|
|
this.resetOnChoose = false,
|
|
|
|
this.keepFocusOnChoose = false,
|
|
|
|
this.onValueChange,
|
|
|
|
this.enabled = true,
|
|
|
|
}) : assert(onSelectUser != null),
|
2019-04-27 16:23:08 +02:00
|
|
|
assert(label != null),
|
|
|
|
assert(resetOnChoose != null),
|
|
|
|
assert(keepFocusOnChoose != null),
|
2019-04-27 14:40:27 +02:00
|
|
|
super(key: key);
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => _PickUserWidgetState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _PickUserWidgetState extends State<PickUserWidget> {
|
|
|
|
// Helper
|
|
|
|
final SearchHelper _searchHelper = SearchHelper();
|
|
|
|
|
|
|
|
// Widget properties
|
|
|
|
final FocusNode _focusNode = FocusNode();
|
|
|
|
final TextEditingController _controller = TextEditingController();
|
|
|
|
OverlayEntry _overlayEntry;
|
|
|
|
UsersList _suggestions;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
|
|
|
|
|
|
|
_focusNode.addListener(() {
|
|
|
|
if (_focusNode.hasFocus) {
|
|
|
|
//Check for focus
|
2019-04-27 16:23:08 +02:00
|
|
|
//_showOverlay();
|
2019-04-27 14:40:27 +02:00
|
|
|
} else {
|
|
|
|
//Remove overlay
|
|
|
|
_removeOverlay();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return TextField(
|
|
|
|
focusNode: _focusNode,
|
|
|
|
onChanged: (s) => _updateSuggestions(),
|
|
|
|
controller: _controller,
|
2019-05-01 09:24:50 +02:00
|
|
|
enabled: widget.enabled,
|
2019-04-27 16:23:08 +02:00
|
|
|
decoration: InputDecoration(
|
|
|
|
labelText: widget.label,
|
|
|
|
alignLabelWithHint: true,
|
|
|
|
),
|
2019-04-27 14:40:27 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
OverlayEntry _createOverlayEntry() {
|
|
|
|
RenderBox renderBox = context.findRenderObject();
|
|
|
|
final size = renderBox.size;
|
|
|
|
final offset = renderBox.localToGlobal(Offset.zero);
|
|
|
|
|
|
|
|
return OverlayEntry(builder: (c) {
|
|
|
|
return Positioned(
|
|
|
|
left: offset.dx,
|
|
|
|
top: offset.dy + size.height + 5.0,
|
|
|
|
width: size.width,
|
|
|
|
child: Material(
|
|
|
|
elevation: 4.0,
|
|
|
|
child: ListView.builder(
|
|
|
|
padding: EdgeInsets.zero,
|
|
|
|
shrinkWrap: true,
|
|
|
|
itemCount: _suggestions == null ? 0 : _suggestions.length,
|
|
|
|
itemBuilder: (c, i) => SimpleUserTile(
|
2020-05-02 15:30:19 +02:00
|
|
|
user: _suggestions[i],
|
|
|
|
onTap: _userTapped,
|
|
|
|
),
|
2019-04-27 14:40:27 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-04-27 16:23:08 +02:00
|
|
|
void _showOverlay() {
|
|
|
|
_overlayEntry = _createOverlayEntry();
|
|
|
|
Overlay.of(context).insert(_overlayEntry);
|
|
|
|
}
|
|
|
|
|
2019-04-27 14:40:27 +02:00
|
|
|
void _removeOverlay() {
|
|
|
|
if (_overlayEntry != null) {
|
|
|
|
_overlayEntry.remove();
|
|
|
|
_overlayEntry = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// This method get called each time the input value is updated
|
|
|
|
Future<void> _updateSuggestions() async {
|
2020-05-02 15:30:19 +02:00
|
|
|
if (widget.onValueChange != null) widget.onValueChange(_controller.text);
|
2019-04-27 14:40:27 +02:00
|
|
|
|
2020-05-02 15:30:19 +02:00
|
|
|
if (_controller.text.length == 0) return _removeOverlay();
|
|
|
|
|
|
|
|
final results = await _searchHelper.searchUser(_controller.text);
|
2019-04-27 14:40:27 +02:00
|
|
|
|
|
|
|
if (results == null) return;
|
|
|
|
|
|
|
|
_suggestions = results;
|
2019-04-27 16:23:08 +02:00
|
|
|
if (_overlayEntry != null)
|
|
|
|
_overlayEntry.markNeedsBuild();
|
|
|
|
else
|
|
|
|
_showOverlay();
|
2019-04-27 14:40:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Method called each time a user is tapped (selected)
|
|
|
|
void _userTapped(User user) {
|
2019-04-27 16:23:08 +02:00
|
|
|
// Hide overlay
|
2019-04-27 14:40:27 +02:00
|
|
|
_removeOverlay();
|
|
|
|
|
2019-04-27 16:23:08 +02:00
|
|
|
// Unfocus if required
|
|
|
|
if (!widget.keepFocusOnChoose) {
|
|
|
|
_focusNode.unfocus();
|
|
|
|
}
|
|
|
|
|
|
|
|
//Check if name has to remain in input
|
|
|
|
if (widget.resetOnChoose) {
|
|
|
|
_controller.text = "";
|
|
|
|
} else
|
|
|
|
_controller.text = user.fullName;
|
|
|
|
|
|
|
|
//Callback
|
2019-04-27 14:40:27 +02:00
|
|
|
widget.onSelectUser(user);
|
|
|
|
}
|
|
|
|
}
|