123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197 |
- using System;
- using System.IO;
- using System.Linq;
- namespace Terminal.Gui {
- /// <summary>
- /// Autocomplete for a <see cref="TextField"/> which shows suggestions within the box.
- /// Displayed suggestions can be completed using the tab key.
- /// </summary>
- public class AppendAutocomplete : AutocompleteBase {
- private TextField textField;
- /// <inheritdoc/>
- public override View HostControl { get => textField; set => textField = (TextField)value; }
- /// <summary>
- /// The color used for rendering the appended text. Note that only
- /// <see cref="ColorScheme.Normal"/> is used and then only <see cref="Attribute.Foreground"/>
- /// (Background comes from <see cref="HostControl"/>).
- /// </summary>
- public override ColorScheme ColorScheme { get; set; }
- /// <summary>
- /// Creates a new instance of the <see cref="AppendAutocomplete"/> class.
- /// </summary>
- public AppendAutocomplete (TextField textField)
- {
- this.textField = textField;
- SelectionKey = Key.Tab;
- ColorScheme = new ColorScheme{
- Normal = new Attribute(Color.DarkGray,0),
- Focus = new Attribute(Color.DarkGray,0),
- HotNormal = new Attribute(Color.DarkGray,0),
- HotFocus = new Attribute(Color.DarkGray,0),
- Disabled = new Attribute(Color.DarkGray,0),
- };
- }
- /// <inheritdoc/>
- public override void ClearSuggestions ()
- {
- base.ClearSuggestions ();
- textField.SetNeedsDisplay ();
- }
- /// <inheritdoc/>
- public override bool MouseEvent (MouseEvent me, bool fromHost = false)
- {
- return false;
- }
- /// <inheritdoc/>
- public override bool ProcessKey (KeyEvent kb)
- {
- var key = kb.Key;
- if (key == SelectionKey) {
- return this.AcceptSelectionIfAny ();
- } else
- if (key == Key.CursorUp) {
- return this.CycleSuggestion (1);
- } else
- if (key == Key.CursorDown) {
- return this.CycleSuggestion (-1);
- }
- else if(key == CloseKey && Suggestions.Any())
- {
- ClearSuggestions();
- _suspendSuggestions = true;
- return true;
- }
- if(char.IsLetterOrDigit((char)kb.KeyValue))
- {
- _suspendSuggestions = false;
- }
- return false;
- }
- bool _suspendSuggestions = false;
- /// <inheritdoc/>
- public override void GenerateSuggestions (AutocompleteContext context)
- {
- if(_suspendSuggestions)
- {
- return;
- }
- base.GenerateSuggestions (context);
- }
- /// <summary>
- /// Renders the current suggestion into the <see cref="TextField"/>
- /// </summary>
- public override void RenderOverlay (Point renderAt)
- {
- if (!this.MakingSuggestion ()) {
- return;
- }
- // draw it like its selected even though its not
- Application.Driver.SetAttribute (new Attribute (ColorScheme.Normal.Foreground, textField.ColorScheme.Focus.Background));
- textField.Move (textField.Text.Length, 0);
- var suggestion = this.Suggestions.ElementAt (this.SelectedIdx);
- var fragment = suggestion.Replacement.Substring (suggestion.Remove);
- int spaceAvailable = textField.Bounds.Width - textField.Text.ConsoleWidth;
- int spaceRequired = fragment.Sum(c=>Rune.ColumnWidth(c));
- if(spaceAvailable < spaceRequired)
- {
- fragment = new string(
- fragment.TakeWhile(c=> (spaceAvailable -= Rune.ColumnWidth(c)) >= 0)
- .ToArray()
- );
- }
- Application.Driver.AddStr (fragment);
- }
- /// <summary>
- /// Accepts the current autocomplete suggestion displaying in the text box.
- /// Returns true if a valid suggestion was being rendered and acceptable or
- /// false if no suggestion was showing.
- /// </summary>
- /// <returns></returns>
- internal bool AcceptSelectionIfAny ()
- {
- if (this.MakingSuggestion ()) {
- var insert = this.Suggestions.ElementAt (this.SelectedIdx);
- var newText = textField.Text.ToString ();
- newText = newText.Substring (0, newText.Length - insert.Remove);
- newText += insert.Replacement;
- textField.Text = newText;
- this.MoveCursorToEnd ();
- this.ClearSuggestions ();
- return true;
- }
- return false;
- }
- internal void MoveCursorToEnd ()
- {
- textField.ClearAllSelection ();
- textField.CursorPosition = textField.Text.Length;
- }
- internal void SetTextTo (FileSystemInfo fileSystemInfo)
- {
- var newText = fileSystemInfo.FullName;
- if (fileSystemInfo is DirectoryInfo) {
- newText += System.IO.Path.DirectorySeparatorChar;
- }
- textField.Text = newText;
- this.MoveCursorToEnd ();
- }
- internal bool CursorIsAtEnd ()
- {
- return textField.CursorPosition == textField.Text.Length;
- }
- /// <summary>
- /// Returns true if there is a suggestion that can be made and the control
- /// is in a state where user would expect to see auto-complete (i.e. focused and
- /// cursor in right place).
- /// </summary>
- /// <returns></returns>
- private bool MakingSuggestion ()
- {
- return Suggestions.Any () && this.SelectedIdx != -1 && textField.HasFocus && this.CursorIsAtEnd ();
- }
- private bool CycleSuggestion (int direction)
- {
- if (this.Suggestions.Count <= 1) {
- return false;
- }
- this.SelectedIdx = (this.SelectedIdx + direction) % this.Suggestions.Count;
- if (this.SelectedIdx < 0) {
- this.SelectedIdx = this.Suggestions.Count () - 1;
- }
- textField.SetNeedsDisplay ();
- return true;
- }
- }
- }
|