cursor_autocomplete_options 1.4.0  cursor_autocomplete_options: ^1.4.0 copied to clipboard
cursor_autocomplete_options: ^1.4.0 copied to clipboard
Add autocomplete dialog options to the cursor of a textfield.
cursor_autocomplete_options #
Flutter already have a autocomplete widget. But it dosen't match the expected ui pattern that normaly we see in desktop and web versions. That's because the autocomplete section is bellow the textfield and we can't change that in the autocomplete api. For that reason this package has been made. To give the possibility to display a listview of options right bellow the cursor indicator with each autocompletion option in a list tile displayed in a overlay.
🌟 Test the live demo! #
Test the package in a online web demo:
https://igormidev.github.io/cursor_autocomplete_options
Getting started #
First, import the pacakge: #
import 'package:cursor_autocomplete_options/cursor_autocomplete_options.dart';
Setting variables #
Inside the widget that contains the textfield, that needs to be statefull to dispose the resources after using it, we will create the following widgets:
late final OptionsController<String> optionsController;  
final FocusNode textfieldFocusNode = FocusNode(); 
final TextEditingController textEditingController = TextEditingController();
⚠️ Important! #
We need to create a focus node for the textfield because the OptionsController needs to be able to focus on the textfield again after focusing in the listtile. Also, the app needs to now the textfield to be abble to calculate where to put the caracters. So don't forget to put the textfield focus node and the textEditingController in your TextFormField widget.
After that, in the init state we will initialize the OptionsController: #
@override
void initState() {
  super.initState();
  optionsController = OptionsController<String>(
    textfieldFocusNode: textfieldFocusNode,
    textEditingController: textEditingController,
    context: context,
    onSelectInsertInCursor: (option) {
      return option;
    },
  );
}
⚠️ Important! #
If your context get's updated constantly, then the context that you used in the context parammeter of OptionsController can became deprecated and no more longer valid. So you need to update it. You can do this by calling the updateContext(context) function of your controller in the build method of your statefull widget, for exemple.
This is not a problem if you never rebuild the widget and the context never get's deprecated.
Bellow is a visual exemple:
@override
Widget build(BuildContext context) {
  // Add this line in your build method of the statefull
  // widget the textfield will be contained at:
  optionsController.updateContext(context); 
 
  return ... // Your widget
}
About generic type <T> of OptionsController #
The OptionsController receives a generic T value. This can be any model, but if you just want to add in the cursor a text you can use <String> as the generic type.
⚠️ Important! #
If the <T> is diferent of String, you need to pass the field optionAsString to cast it into a String. So this parameter becomes mandatory if you are not using <String> as the generic type. Otherwise, if the generic type is <String>, you don't need to pass this parametter because the package already know it's is a string.
Now, you can use the textfield with the values we created: #
TextFormField(
    focusNode: textfieldFocusNode, // <= Don't forget!
    controller: textEditingController, // <= Don't forget!
    style: const TextStyle(height: 1, fontSize: 18), // Don't forget *height* parametter!
    ...
),
⚠️ Important! #
You need to initialize a field called style in the textfield with the parameter height of it beeing 1. I can't be other value.
The other values of the Textstyle, like fontSize, can be anyone that you like.
⚠️ Important! #
Don't forget to dispose the controller and textfield in the end. Notice: you DON'T need to dispose the textfield focus node because `OptionsController.dispose()`` will already dispose it for you. See the following exemple:
@override
void dispose() {
  super.dispose();
  // Don't forget to dispose!
  optionsController.dispose();
  textEditingController.dispose();
}
Usage #
You can trigger to show the dialog in the following cursor when ever you wan't. Remember that the textfield has to be focused in order to the package reconize the cursor and it's location. For that the package will the focus node of the textfield to see if it is focused.
To trigger the overlay to appear with the options, you will call the funcion showOptionsMenu() in your OptionsController controller passing the list of options that you wan't
to give to the user. This function will receive the suggestion parameters that is a list of items of the type <T>, the same type of your OptionController<T>.
After the showOptionsMenu() function is trigged, it will show the dialog with the options and then, after the user select's the option, the app will trigger both functions onSelectedOption and onSelectInsertInCursor (the one's that are not null).
You can trigger it, for example, when the user types "#" in the textfield. But this is totally open for you about when to trigger it. Bellow is a example of this case:
TextFormField(
  ... // Other atributes
  onChanged: (value) {
    if (value.isEmpty) return;
    final cursorPositionIndex =
      textEditingController.selection.base.offset;
    final typedValue = value[cursorPositionIndex - 1];
    final isTypedCaracterHashtag = typedValue == '#';
    if (isTypedCaracterHashtag) {
      optionsController.showOptionsMenu(suggestion);
    }
  },
)
You can see this full exemple by clicking here
Customize the card widget #
Want to customize your card widget?
Use showOptionsMenuWithWrapperBuilder and create a wrapper above the Listview with the ListTiles with the options.
Manipulating selections #
You can manipulate and determine what will be done with the selection with two major functions: onSelectedOption and onSelectInsertInCursor.
We don't recommend to use both of them, but technically there is no problem in using both.
onSelectedOption #
Will give you complete controll of what to do with the selected option. So if you want you can manipulate the textfield values yourself.
onSelectInsertInCursor #
This is a pre-built option builded above onSelectedOption that manipulate the TextEditingController in order to insert in the current cursor possition the return of this function. Also, you can change the position of the cursor after adding the text with the passing a cursorIndexChangeQuantity in the return payload. Check the InsertInCursorPayload for more info.
Configurations #
Change card size #
Inside the OptionsController, you can manipulate the width and height of the options listview with the fields: overlayCardHeight and overlayCardWeight.
Debounce #
You can configure a debouncer in the trigger of the options overlay. The default is 300ms.
Controll when to close dialog #
Maybe, you don't want to close the dialog after doing your login in onSelectedOption or onSelectInsertInCursor. So you can set the willAutomaticallyCloseDialogAfterSelection field to false and the dialog will stop closing after selecting a value.
Made with ❤ by Igor Miranda 
If you like the package, give a 👍