mirror of
https://github.com/gui-cs/Terminal.Gui.git
synced 2026-02-10 04:03:41 +01:00
* Initial plan * CommandContext infrastructure: WeakReference and extension methods Co-authored-by: tig <585482+tig@users.noreply.github.com> * Align CommandContext infrastructure with PR #4620 - Update ViewExtensions.cs to use Terminal.Gui.ViewBase namespace - Update WeakReferenceExtensions.cs to match PR #4620 format - Change CommandContext constructor to take WeakReference<View>? directly - Add CommandContext.ToString() implementation - Update ICommandContext.Source documentation - Update View.Command.cs to explicitly create WeakReference - Update Shortcut.cs to handle WeakReference Source - Update ComboBox.cs Source comparison - Update Dialog.cs and DialogTResult.cs pattern matching - Update MenuBar.cs and PopoverMenu.cs to work with WeakReference - Update ScrollSlider.cs and OptionSelector.cs Remaining: Fix Examples code and tests Co-authored-by: tig <585482+tig@users.noreply.github.com> * Fix Examples code for WeakReference Source - Update UICatalogRunnable.cs to use TryGetTarget for Data access - Update Menus.cs to use ToIdentifyingString() and TryGetTarget - Update MouseTester.cs to use ToIdentifyingString() for all Source.Id access All Examples now build successfully Co-authored-by: tig <585482+tig@users.noreply.github.com> * Update tests for WeakReference Source - Fix CommandContextTests.cs to wrap Source in WeakReference - Fix InputBindingTests.cs to wrap Source in WeakReference - Update assertions to use TryGetTarget for safe access - All CommandContext and InputBinding tests pass Co-authored-by: tig <585482+tig@users.noreply.github.com> * Add TryGetSource extension methods and replace usage - Add TryGetSource extension to WeakReferenceExtensions.cs - Add CommandContextExtensions.cs with TryGetSource for ICommandContext - Replace 14 instances of Source?.TryGetTarget pattern across codebase - Updated Terminal.Gui files: ComboBox, Dialog, DialogTResult, MenuBar, PopoverMenu, OptionSelector, Shortcut - Updated Examples files: UICatalogRunnable, Menus - All files build successfully Co-authored-by: tig <585482+tig@users.noreply.github.com> * Add tests for TryGetSource extension methods - Add 6 tests for TryGetSource extension methods - Test WeakReference<View>.TryGetSource with valid/null references - Test ICommandContext.TryGetSource with valid/null contexts - Test pattern matching usage with TryGetSource - All 23 CommandContext tests pass - Full test suite: 15,094 passed Co-authored-by: tig <585482+tig@users.noreply.github.com> * code cleanup --------- Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: tig <585482+tig@users.noreply.github.com> Co-authored-by: Tig <tig@users.noreply.github.com>
1012 lines
30 KiB
C#
1012 lines
30 KiB
C#
#nullable disable
|
|
|
|
//
|
|
// ComboBox.cs: ComboBox control
|
|
//
|
|
// Authors:
|
|
// Ross Ferguson (ross.c.ferguson@btinternet.com)
|
|
//
|
|
|
|
using System.Collections.ObjectModel;
|
|
|
|
namespace Terminal.Gui.Views;
|
|
|
|
/// <summary>Provides a drop-down list of items the user can select from.</summary>
|
|
[Obsolete ("ComboBox is obsolete and will be removed before v2 Beta. See Issue ##2404.")]
|
|
public class ComboBox : View, IDesignable
|
|
{
|
|
private readonly ComboListView _listview;
|
|
private readonly int _minimumHeight = 2;
|
|
private readonly TextField _search;
|
|
private readonly ObservableCollection<object> _searchSet = [];
|
|
private bool _autoHide = true;
|
|
private bool _hideDropdownListOnClick;
|
|
private int _lastSelectedItem = -1;
|
|
private int _selectedItem = -1;
|
|
private IListDataSource _source;
|
|
private string _text = "";
|
|
|
|
/// <summary>Public constructor</summary>
|
|
public ComboBox ()
|
|
{
|
|
CanFocus = true;
|
|
_search = new TextField { CanFocus = true, TabStop = TabBehavior.NoStop };
|
|
|
|
_listview = new ComboListView (this, HideDropdownListOnClick) { CanFocus = true, TabStop = TabBehavior.NoStop };
|
|
|
|
_search.TextChanged += Search_Changed;
|
|
|
|
_listview.Y = Pos.Bottom (_search);
|
|
|
|
_listview.Accepting += (sender, args) =>
|
|
{
|
|
// This prevents Accepted from bubbling up to the combobox
|
|
args.Handled = true;
|
|
|
|
// But OpenSelectedItem won't be fired because of that. So do it here.
|
|
SelectText ();
|
|
};
|
|
|
|
_listview.ValueChanged += (sender, e) =>
|
|
{
|
|
if (e.NewValue >= 0 && !HideDropdownListOnClick && _searchSet.Count > 0)
|
|
{
|
|
SetValue (_searchSet [e.NewValue.Value]);
|
|
}
|
|
};
|
|
Add (_search, _listview);
|
|
|
|
// BUGBUG: This should not be needed; LayoutComplete will handle
|
|
Initialized += (s, e) => ProcessLayout ();
|
|
|
|
// On resize
|
|
SubViewsLaidOut += (sender, a) => ProcessLayout ();
|
|
|
|
SuperViewChanged += (s, e) =>
|
|
{
|
|
// Determine if this view is hosted inside a dialog and is the only control
|
|
for (View view = SuperView; view != null; view = view.SuperView)
|
|
{
|
|
if (view is Dialog && SuperView is { } && SuperView.SubViews.Count == 1 && SuperView.SubViews.ElementAt (0) == this)
|
|
{
|
|
_autoHide = false;
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
SetNeedsLayout ();
|
|
SetNeedsDraw ();
|
|
ShowHideList (Text);
|
|
};
|
|
|
|
// Things this view knows how to do
|
|
AddCommand (Command.Accept,
|
|
ctx =>
|
|
{
|
|
if (ctx?.TryGetSource (out View? sourceView) == true && sourceView == _search)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
return ActivateSelected (ctx);
|
|
});
|
|
AddCommand (Command.Toggle, () => ExpandCollapse ());
|
|
AddCommand (Command.Expand, () => Expand ());
|
|
AddCommand (Command.Collapse, () => Collapse ());
|
|
AddCommand (Command.Down, MoveDown);
|
|
AddCommand (Command.Up, MoveUp);
|
|
AddCommand (Command.PageDown, () => PageDown ());
|
|
AddCommand (Command.PageUp, () => PageUp ());
|
|
AddCommand (Command.Start, () => MoveHome ());
|
|
AddCommand (Command.End, () => MoveEnd ());
|
|
AddCommand (Command.Cancel, () => CancelSelected ());
|
|
AddCommand (Command.UnixEmulation, () => UnixEmulation ());
|
|
|
|
// Default keybindings for this view
|
|
KeyBindings.Add (Key.F4, Command.Toggle);
|
|
KeyBindings.Add (Key.CursorDown, Command.Down);
|
|
KeyBindings.Add (Key.CursorUp, Command.Up);
|
|
KeyBindings.Add (Key.PageDown, Command.PageDown);
|
|
KeyBindings.Add (Key.PageUp, Command.PageUp);
|
|
KeyBindings.Add (Key.Home, Command.Start);
|
|
KeyBindings.Add (Key.End, Command.End);
|
|
KeyBindings.Add (Key.Esc, Command.Cancel);
|
|
KeyBindings.Add (Key.U.WithCtrl, Command.UnixEmulation);
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
protected override bool OnSettingScheme (ValueChangingEventArgs<Scheme> args)
|
|
{
|
|
_listview.SetScheme (args.NewValue);
|
|
|
|
return base.OnSettingScheme (args);
|
|
}
|
|
|
|
/// <summary>Gets or sets if the drop-down list can be hide with a button click event.</summary>
|
|
public bool HideDropdownListOnClick { get => _hideDropdownListOnClick; set => _hideDropdownListOnClick = _listview.HideDropdownListOnClick = value; }
|
|
|
|
/// <summary>Gets the drop-down list state, expanded or collapsed.</summary>
|
|
public bool IsShow { get; private set; }
|
|
|
|
/// <summary>If set to true, no changes to the text will be allowed.</summary>
|
|
public bool ReadOnly
|
|
{
|
|
get => _search.ReadOnly;
|
|
set => _search.ReadOnly = value;
|
|
|
|
//if (_search.ReadOnly)
|
|
//{
|
|
// if (_search.Scheme is { })
|
|
// {
|
|
// _search.Scheme = new Scheme (_search.Scheme) { Normal = _search.Scheme.Focus };
|
|
// }
|
|
//}
|
|
}
|
|
|
|
/// <summary>Current search text</summary>
|
|
public string SearchText { get => _search.Text; set => SetSearchText (value); }
|
|
|
|
/// <summary>Gets the index of the currently selected item in the <see cref="Source"/></summary>
|
|
/// <value>The selected item or -1 none selected.</value>
|
|
public int SelectedItem
|
|
{
|
|
get => _selectedItem;
|
|
set
|
|
{
|
|
if (_selectedItem != value && (value == -1 || (_source is { } && value > -1 && value < _source.Count)))
|
|
{
|
|
_selectedItem = _lastSelectedItem = value;
|
|
|
|
if (_selectedItem != -1)
|
|
{
|
|
SetValue (_source.ToList () [_selectedItem].ToString (), true);
|
|
}
|
|
else
|
|
{
|
|
SetValue ("", true);
|
|
}
|
|
|
|
OnSelectedChanged ();
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>Gets or sets the <see cref="IListDataSource"/> backing this <see cref="ComboBox"/>, enabling custom rendering.</summary>
|
|
/// <value>The source.</value>
|
|
/// <remarks>Use <see cref="SetSource{T}"/> to set a new <see cref="ObservableCollection{T}"/> source.</remarks>
|
|
public IListDataSource Source
|
|
{
|
|
get => _source;
|
|
set
|
|
{
|
|
_source = value;
|
|
|
|
// Only need to refresh list if its been added to a container view
|
|
if (SuperView is { } && SuperView.SubViews.Contains (this))
|
|
{
|
|
Text = string.Empty;
|
|
SetNeedsDraw ();
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>The text of the currently selected list item</summary>
|
|
public override string Text
|
|
{
|
|
get => _text;
|
|
set
|
|
{
|
|
// Guard against base constructor calling before _search is initialized
|
|
if (_search is null)
|
|
{
|
|
_text = value;
|
|
|
|
return;
|
|
}
|
|
|
|
SetSearchText (value);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Collapses the drop-down list. Returns true if the state changed or false if it was already collapsed and no
|
|
/// action was taken
|
|
/// </summary>
|
|
public virtual bool Collapse ()
|
|
{
|
|
if (!IsShow)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
IsShow = false;
|
|
HideList ();
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>This event is raised when the drop-down list is collapsed.</summary>
|
|
public event EventHandler Collapsed;
|
|
|
|
/// <summary>
|
|
/// Expands the drop-down list. Returns true if the state changed or false if it was already expanded and no
|
|
/// action was taken
|
|
/// </summary>
|
|
public virtual bool Expand ()
|
|
{
|
|
if (IsShow)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
SetSearchSet ();
|
|
IsShow = true;
|
|
ShowList ();
|
|
FocusSelectedItem ();
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>This event is raised when the drop-down list is expanded.</summary>
|
|
public event EventHandler Expanded;
|
|
|
|
/// <inheritdoc/>
|
|
protected override bool OnMouseEvent (Mouse me)
|
|
{
|
|
if (me.Position!.Value.X == Viewport.Right - 1 && me.Position!.Value.Y == Viewport.Top && me.Flags == MouseFlags.LeftButtonPressed && _autoHide)
|
|
{
|
|
if (IsShow)
|
|
{
|
|
IsShow = false;
|
|
HideList ();
|
|
}
|
|
else
|
|
{
|
|
SetSearchSet ();
|
|
|
|
IsShow = true;
|
|
ShowList ();
|
|
FocusSelectedItem ();
|
|
}
|
|
|
|
return me.Handled = true;
|
|
}
|
|
|
|
if (me.Flags == MouseFlags.LeftButtonPressed)
|
|
{
|
|
if (!_search.HasFocus)
|
|
{
|
|
_search.SetFocus ();
|
|
}
|
|
|
|
return me.Handled = true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>Virtual method which invokes the <see cref="Collapsed"/> event.</summary>
|
|
public virtual void OnCollapsed () => Collapsed?.Invoke (this, EventArgs.Empty);
|
|
|
|
/// <inheritdoc/>
|
|
protected override bool OnDrawingContent (DrawContext context)
|
|
{
|
|
if (!_autoHide)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
SetAttributeForRole (Enabled ? VisualRole.Focus : VisualRole.Disabled);
|
|
AddRune (Viewport.Right - 1, 0, Glyphs.DownArrow);
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>Virtual method which invokes the <see cref="Expanded"/> event.</summary>
|
|
public virtual void OnExpanded () => Expanded?.Invoke (this, EventArgs.Empty);
|
|
|
|
/// <inheritdoc/>
|
|
protected override void OnHasFocusChanged (bool newHasFocus, View previousFocusedView, View view)
|
|
{
|
|
if (newHasFocus)
|
|
{
|
|
if (!_search.HasFocus && !_listview.HasFocus)
|
|
{
|
|
_search.SetFocus ();
|
|
}
|
|
_search.InsertionPoint = _search.Text.GetRuneCount ();
|
|
}
|
|
else
|
|
{
|
|
if (_source?.Count > 0 && _selectedItem > -1 && _selectedItem < _source.Count - 1 && _text != _source.ToList () [_selectedItem].ToString ())
|
|
{
|
|
SetValue (_source.ToList () [_selectedItem].ToString ());
|
|
}
|
|
|
|
if (_autoHide && IsShow && view != this && view != _search && view != _listview)
|
|
{
|
|
IsShow = false;
|
|
HideList ();
|
|
}
|
|
else if (_listview.TabStop?.HasFlag (TabBehavior.TabStop) ?? false)
|
|
{
|
|
_listview.TabStop = TabBehavior.NoStop;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>Invokes the OnOpenSelectedItem event if it is defined.</summary>
|
|
/// <returns></returns>
|
|
public virtual bool OnOpenSelectedItem ()
|
|
{
|
|
string value = _search.Text;
|
|
_lastSelectedItem = SelectedItem;
|
|
OpenSelectedItem?.Invoke (this, new ListViewItemEventArgs (SelectedItem, value));
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>Invokes the SelectedChanged event if it is defined.</summary>
|
|
/// <returns></returns>
|
|
public virtual bool OnSelectedChanged ()
|
|
{
|
|
// Note: Cannot rely on "listview.SelectedItem != lastSelectedItem" because the list is dynamic.
|
|
// So we cannot optimize. Ie: Don't call if not changed
|
|
SelectedItemChanged?.Invoke (this, new ListViewItemEventArgs (SelectedItem, _search.Text));
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>This event is raised when the user Double Clicks on an item or presses ENTER to open the selected item.</summary>
|
|
public event EventHandler<ListViewItemEventArgs> OpenSelectedItem;
|
|
|
|
/// <summary>This event is raised when the selected item in the <see cref="ComboBox"/> has changed.</summary>
|
|
public event EventHandler<ListViewItemEventArgs> SelectedItemChanged;
|
|
|
|
/// <summary>Sets the source of the <see cref="ComboBox"/> to an <see cref="ObservableCollection{T}"/>.</summary>
|
|
/// <value>An object implementing the INotifyCollectionChanged and INotifyPropertyChanged interface.</value>
|
|
/// <remarks>
|
|
/// Use the <see cref="Source"/> property to set a new <see cref="IListDataSource"/> source and use custom
|
|
/// rendering.
|
|
/// </remarks>
|
|
public void SetSource<T> (ObservableCollection<T> source)
|
|
{
|
|
if (source is null)
|
|
{
|
|
Source = null;
|
|
}
|
|
else
|
|
{
|
|
_listview.SetSource (source);
|
|
Source = _listview.Source;
|
|
}
|
|
}
|
|
|
|
private bool ActivateSelected (ICommandContext commandContext)
|
|
{
|
|
if (HasItems ())
|
|
{
|
|
if (SelectText ())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return RaiseAccepting (commandContext) == true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/// <summary>Internal height of dynamic search list</summary>
|
|
/// <returns></returns>
|
|
private int CalculateHeight ()
|
|
{
|
|
if (!IsInitialized || Viewport.Height == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
return Math.Min (Math.Max (Viewport.Height - 1, _minimumHeight - 1),
|
|
_searchSet?.Count > 0 ? _searchSet.Count : IsShow ? Math.Max (Viewport.Height - 1, _minimumHeight - 1) : 0);
|
|
}
|
|
|
|
private bool CancelSelected ()
|
|
{
|
|
if (HasFocus)
|
|
{
|
|
_search.SetFocus ();
|
|
}
|
|
|
|
if (ReadOnly || HideDropdownListOnClick)
|
|
{
|
|
SelectedItem = _lastSelectedItem;
|
|
|
|
if (SelectedItem > -1 && _listview.Source?.Count > 0)
|
|
{
|
|
Text = _listview.Source.ToList () [SelectedItem]?.ToString ();
|
|
}
|
|
}
|
|
else if (!ReadOnly)
|
|
{
|
|
Text = string.Empty;
|
|
_selectedItem = _lastSelectedItem;
|
|
OnSelectedChanged ();
|
|
}
|
|
|
|
return Collapse ();
|
|
}
|
|
|
|
/// <summary>Toggles the expand/collapse state of the sublist in the combo box</summary>
|
|
/// <returns></returns>
|
|
private bool ExpandCollapse ()
|
|
{
|
|
if (_search.HasFocus || _listview.HasFocus)
|
|
{
|
|
if (!IsShow)
|
|
{
|
|
return Expand ();
|
|
}
|
|
|
|
return Collapse ();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private void FocusSelectedItem ()
|
|
{
|
|
if (_listview.Source?.Count > 0)
|
|
{
|
|
_listview.SelectedItem = SelectedItem > -1 ? SelectedItem : 0;
|
|
}
|
|
_listview.TabStop = TabBehavior.TabStop;
|
|
_listview.SetFocus ();
|
|
OnExpanded ();
|
|
}
|
|
|
|
private int GetSelectedItemFromSource (string searchText)
|
|
{
|
|
if (_source is null)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
for (var i = 0; i < _searchSet.Count; i++)
|
|
{
|
|
if (_searchSet [i].ToString () == searchText)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
private bool HasItems () => Source?.Count > 0;
|
|
|
|
/// <summary>Hide the search list</summary>
|
|
/// Consider making public
|
|
private void HideList ()
|
|
{
|
|
if (_lastSelectedItem != _selectedItem)
|
|
{
|
|
OnOpenSelectedItem ();
|
|
}
|
|
|
|
Reset (true);
|
|
_listview.ClearViewport ();
|
|
_listview.TabStop = TabBehavior.NoStop;
|
|
SuperView?.MoveSubViewToStart (this);
|
|
|
|
// BUGBUG: SetNeedsDraw takes Viewport relative coordinates, not Screen
|
|
Rectangle rect = _listview.ViewportToScreen (_listview.IsInitialized ? _listview.Viewport : Rectangle.Empty);
|
|
SuperView?.SetNeedsDraw (rect);
|
|
OnCollapsed ();
|
|
}
|
|
|
|
private bool? MoveDown ()
|
|
{
|
|
if (_search.HasFocus)
|
|
{
|
|
// jump to list
|
|
if (_searchSet?.Count > 0)
|
|
{
|
|
_listview.TabStop = TabBehavior.TabStop;
|
|
_listview.SetFocus ();
|
|
|
|
if (_listview.SelectedItem is { })
|
|
{
|
|
SetValue (_searchSet [_listview.SelectedItem.Value]);
|
|
}
|
|
else
|
|
{
|
|
_listview.SelectedItem = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private bool? MoveEnd ()
|
|
{
|
|
if (!IsShow && _search.HasFocus)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
if (HasItems ())
|
|
{
|
|
_listview.MoveEnd ();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool? MoveHome ()
|
|
{
|
|
if (!IsShow && _search.HasFocus)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
if (HasItems ())
|
|
{
|
|
_listview.MoveHome ();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool? MoveUp ()
|
|
{
|
|
if (HasItems ())
|
|
{
|
|
return _listview.MoveUp ();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private bool? MoveUpList ()
|
|
{
|
|
if (_listview.HasFocus && _listview.SelectedItem == 0 && _searchSet?.Count > 0) // jump back to search
|
|
{
|
|
_search.InsertionPoint = _search.Text.GetRuneCount ();
|
|
_search.SetFocus ();
|
|
}
|
|
else
|
|
{
|
|
MoveUp ();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool PageDown ()
|
|
{
|
|
if (HasItems ())
|
|
{
|
|
_listview.MovePageDown ();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool PageUp ()
|
|
{
|
|
if (HasItems ())
|
|
{
|
|
_listview.MovePageUp ();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// TODO: Upgrade Combobox to use Dim.Auto instead of all this stuff.
|
|
private void ProcessLayout ()
|
|
{
|
|
if (Viewport.Height < _minimumHeight && (Height is null || Height is DimAbsolute))
|
|
{
|
|
Height = _minimumHeight;
|
|
}
|
|
|
|
// BUGBUG: This uses Viewport. Should use ContentSize
|
|
if ((!_autoHide && Viewport.Width > 0 && _search.Frame.Width != Viewport.Width)
|
|
|| (_autoHide && Viewport.Width > 0 && _search.Frame.Width != Viewport.Width - 1))
|
|
{
|
|
_search.Width = _listview.Width = _autoHide ? Viewport.Width - 1 : Viewport.Width;
|
|
_listview.Height = CalculateHeight ();
|
|
_search.SetRelativeLayout (GetContentSize ());
|
|
_listview.SetRelativeLayout (GetContentSize ());
|
|
}
|
|
}
|
|
|
|
/// <summary>Reset to full original list</summary>
|
|
private void Reset (bool keepSearchText = false)
|
|
{
|
|
if (!keepSearchText)
|
|
{
|
|
SetSearchText (string.Empty);
|
|
}
|
|
|
|
ResetSearchSet ();
|
|
|
|
_listview.SetSource (_searchSet);
|
|
_listview.Height = CalculateHeight ();
|
|
|
|
if (SubViews.Count > 0 && HasFocus)
|
|
{
|
|
_search.SetFocus ();
|
|
}
|
|
}
|
|
|
|
private void ResetSearchSet (bool noCopy = false)
|
|
{
|
|
_listview.SuspendCollectionChangedEvent ();
|
|
_searchSet.Clear ();
|
|
_listview.ResumeSuspendCollectionChangedEvent ();
|
|
|
|
if (_autoHide || noCopy)
|
|
{
|
|
return;
|
|
}
|
|
|
|
SetSearchSet ();
|
|
}
|
|
|
|
private void Search_Changed (object sender, EventArgs e)
|
|
{
|
|
if (_source is null)
|
|
{
|
|
// Object initialization
|
|
return;
|
|
}
|
|
|
|
ShowHideList (Text);
|
|
}
|
|
|
|
private void ShowHideList (string oldText)
|
|
{
|
|
if (string.IsNullOrEmpty (_search.Text) && string.IsNullOrEmpty (oldText))
|
|
{
|
|
ResetSearchSet ();
|
|
}
|
|
else if (_search.Text != oldText)
|
|
{
|
|
if (_search.Text.Length < oldText.Length)
|
|
{
|
|
_selectedItem = -1;
|
|
}
|
|
|
|
IsShow = true;
|
|
ResetSearchSet (true);
|
|
|
|
if (!string.IsNullOrEmpty (_search.Text))
|
|
{
|
|
_listview.SuspendCollectionChangedEvent ();
|
|
|
|
foreach (object item in _source.ToList ())
|
|
{
|
|
// Iterate to preserver object type and force deep copy
|
|
if (item.ToString ().StartsWith (_search.Text, StringComparison.CurrentCultureIgnoreCase))
|
|
{
|
|
_searchSet.Add (item);
|
|
}
|
|
}
|
|
|
|
_listview.ResumeSuspendCollectionChangedEvent ();
|
|
}
|
|
}
|
|
|
|
if (HasFocus)
|
|
{
|
|
ShowList ();
|
|
}
|
|
else if (_autoHide)
|
|
{
|
|
IsShow = false;
|
|
HideList ();
|
|
}
|
|
}
|
|
|
|
private bool SelectText ()
|
|
{
|
|
IsShow = false;
|
|
_listview.TabStop = TabBehavior.NoStop;
|
|
|
|
if (_listview.Source!.Count == 0 || (_searchSet?.Count ?? 0) == 0)
|
|
{
|
|
_text = "";
|
|
HideList ();
|
|
IsShow = false;
|
|
|
|
return false;
|
|
}
|
|
|
|
SetValue (_listview.SelectedItem is { } ? _searchSet [_listview.SelectedItem.Value] : _text);
|
|
_search.InsertionPoint = _search.Text.GetColumns ();
|
|
ShowHideList (Text);
|
|
OnOpenSelectedItem ();
|
|
Reset (true);
|
|
HideList ();
|
|
IsShow = false;
|
|
|
|
return true;
|
|
}
|
|
|
|
private void SetSearchSet ()
|
|
{
|
|
if (Source is null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// PERF: At the request of @dodexahedron in the comment https://github.com/gui-cs/Terminal.Gui/pull/3552#discussion_r1648112410.
|
|
_listview.SuspendCollectionChangedEvent ();
|
|
|
|
// force deep copy
|
|
foreach (object item in Source.ToList ())
|
|
{
|
|
_searchSet.Add (item);
|
|
}
|
|
|
|
_listview.ResumeSuspendCollectionChangedEvent ();
|
|
}
|
|
|
|
// Sets the search text field Text as well as our own Text property
|
|
private void SetSearchText (string value)
|
|
{
|
|
_search.Text = value;
|
|
_text = value;
|
|
}
|
|
|
|
private void SetValue (object text, bool isFromSelectedItem = false)
|
|
{
|
|
// TOOD: The fact we have to suspend events to change the text makes this feel very hacky.
|
|
_search.TextChanged -= Search_Changed;
|
|
|
|
// Note we set _text, to avoid set_Text from setting _search.Text again
|
|
_text = _search.Text = text.ToString ();
|
|
_search.InsertionPoint = 0;
|
|
_search.TextChanged += Search_Changed;
|
|
|
|
if (!isFromSelectedItem)
|
|
{
|
|
_selectedItem = GetSelectedItemFromSource (_text);
|
|
OnSelectedChanged ();
|
|
}
|
|
}
|
|
|
|
/// <summary>Show the search list</summary>
|
|
/// Consider making public
|
|
private void ShowList ()
|
|
{
|
|
_listview.SuspendCollectionChangedEvent ();
|
|
_listview.SetSource (_searchSet);
|
|
_listview.ResumeSuspendCollectionChangedEvent ();
|
|
|
|
_listview.ClearViewport ();
|
|
_listview.Height = CalculateHeight ();
|
|
SuperView?.MoveSubViewToStart (this);
|
|
}
|
|
|
|
private bool UnixEmulation ()
|
|
{
|
|
// Unix emulation
|
|
Reset ();
|
|
|
|
return true;
|
|
}
|
|
|
|
private class ComboListView : ListView
|
|
{
|
|
private ComboBox _container;
|
|
private bool _hideDropdownListOnClick;
|
|
private int _highlighted = -1;
|
|
private bool _isFocusing;
|
|
public ComboListView (ComboBox container, bool hideDropdownListOnClick) => SetInitialProperties (container, hideDropdownListOnClick);
|
|
|
|
public ComboListView (ComboBox container, ObservableCollection<string> source, bool hideDropdownListOnClick)
|
|
{
|
|
Source = new ListWrapper<string> (source);
|
|
SetInitialProperties (container, hideDropdownListOnClick);
|
|
}
|
|
|
|
public bool HideDropdownListOnClick
|
|
{
|
|
get => _hideDropdownListOnClick;
|
|
set
|
|
{
|
|
_hideDropdownListOnClick = value;
|
|
MouseHoldRepeat = value ? MouseFlags.LeftButtonReleased : null;
|
|
}
|
|
}
|
|
|
|
protected override bool OnMouseEvent (Mouse me)
|
|
{
|
|
bool isMousePositionValid = IsMousePositionValid (me);
|
|
|
|
var res = false;
|
|
|
|
if (isMousePositionValid)
|
|
{
|
|
// We're derived from ListView and it overrides OnMouseEvent, so we need to call it
|
|
res = base.OnMouseEvent (me);
|
|
}
|
|
|
|
if (HideDropdownListOnClick && me.Flags == MouseFlags.LeftButtonClicked)
|
|
{
|
|
if (!isMousePositionValid && !_isFocusing)
|
|
{
|
|
_container.IsShow = false;
|
|
_container.HideList ();
|
|
}
|
|
else if (isMousePositionValid)
|
|
{
|
|
return RaiseAccepting (new CommandContext (Command.Accept, new WeakReference<View> (this), new InputBinding ())) == true;
|
|
}
|
|
else
|
|
{
|
|
_isFocusing = false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
if (me.Flags == MouseFlags.PositionReport && HideDropdownListOnClick)
|
|
{
|
|
if (isMousePositionValid)
|
|
{
|
|
_highlighted = Math.Min (TopItem + me.Position!.Value.Y, Source.Count);
|
|
SetNeedsDraw ();
|
|
}
|
|
|
|
_isFocusing = false;
|
|
|
|
return true;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
protected override bool OnDrawingContent (DrawContext context)
|
|
{
|
|
Attribute current = GetAttributeForRole (VisualRole.Focus);
|
|
SetAttribute (current);
|
|
Move (0, 0);
|
|
Rectangle f = Frame;
|
|
int item = TopItem;
|
|
bool focused = HasFocus;
|
|
int col = ShowMarks ? 2 : 0;
|
|
int start = LeftItem;
|
|
|
|
for (var row = 0; row < f.Height; row++, item++)
|
|
{
|
|
bool isSelected = item == _container.SelectedItem;
|
|
bool isHighlighted = _hideDropdownListOnClick && item == _highlighted;
|
|
|
|
Attribute newcolor;
|
|
|
|
if (isHighlighted || (isSelected && !_hideDropdownListOnClick))
|
|
{
|
|
newcolor = focused ? GetAttributeForRole (VisualRole.Focus) : GetAttributeForRole (VisualRole.HotNormal);
|
|
}
|
|
else if (isSelected && _hideDropdownListOnClick)
|
|
{
|
|
newcolor = focused ? GetAttributeForRole (VisualRole.HotFocus) : GetAttributeForRole (VisualRole.HotNormal);
|
|
}
|
|
else
|
|
{
|
|
newcolor = GetAttributeForRole (VisualRole.Normal);
|
|
}
|
|
|
|
if (newcolor != current)
|
|
{
|
|
SetAttribute (newcolor);
|
|
current = newcolor;
|
|
}
|
|
|
|
Move (0, row);
|
|
|
|
if (Source is null || item >= Source.Count)
|
|
{
|
|
for (var c = 0; c < f.Width; c++)
|
|
{
|
|
AddRune (0, row, (Rune)' ');
|
|
}
|
|
}
|
|
else
|
|
{
|
|
var rowEventArgs = new ListViewRowEventArgs (item);
|
|
OnRowRender (rowEventArgs);
|
|
|
|
if (rowEventArgs.RowAttribute is { } && current != rowEventArgs.RowAttribute)
|
|
{
|
|
current = (Attribute)rowEventArgs.RowAttribute;
|
|
SetAttribute (current);
|
|
}
|
|
|
|
if (ShowMarks)
|
|
{
|
|
AddRune (Source.IsMarked (item) ? MarkMultiple ? Glyphs.CheckStateChecked : Glyphs.Selected :
|
|
MarkMultiple ? Glyphs.CheckStateUnChecked : Glyphs.UnSelected);
|
|
AddRune ((Rune)' ');
|
|
}
|
|
|
|
Source.Render (this, isSelected, item, col, row, f.Width - col, start);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
protected override void OnHasFocusChanged (bool newHasFocus, [CanBeNull] View previousFocusedView, [CanBeNull] View focusedView)
|
|
{
|
|
if (newHasFocus)
|
|
{
|
|
if (_hideDropdownListOnClick)
|
|
{
|
|
_isFocusing = true;
|
|
_highlighted = _container.SelectedItem;
|
|
App?.Mouse.GrabMouse (this);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (_hideDropdownListOnClick)
|
|
{
|
|
_isFocusing = false;
|
|
_highlighted = _container.SelectedItem;
|
|
App?.Mouse.UngrabMouse ();
|
|
}
|
|
}
|
|
}
|
|
|
|
protected override void OnValueChanged (ValueChangedEventArgs<int?> args)
|
|
{
|
|
base.OnValueChanged (args);
|
|
|
|
if (SelectedItem is null)
|
|
{
|
|
return;
|
|
}
|
|
_highlighted = SelectedItem.Value;
|
|
}
|
|
|
|
private bool IsMousePositionValid (Mouse me)
|
|
{
|
|
if (me.Position!.Value.X >= 0 && me.Position!.Value.X < Frame.Width && me.Position!.Value.Y >= 0 && me.Position!.Value.Y < Frame.Height)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private void SetInitialProperties (ComboBox container, bool hideDropdownListOnClick)
|
|
{
|
|
_container = container ?? throw new ArgumentNullException (nameof (container), @"ComboBox container cannot be null.");
|
|
HideDropdownListOnClick = hideDropdownListOnClick;
|
|
AddCommand (Command.Up, () => _container.MoveUpList ());
|
|
}
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public bool EnableForDesign ()
|
|
{
|
|
ObservableCollection<string> source = new (["Combo Item 1", "Combo Item two", "Combo Item Quattro", "Last Combo Item"]);
|
|
SetSource (source);
|
|
Height = Dim.Auto (DimAutoStyle.Content, source.Count + 1);
|
|
|
|
return true;
|
|
}
|
|
}
|