Files
Terminal.Gui/Terminal.Gui/Views/ComboBox.cs
Copilot f2d260a853 Add TryGetSource extension methods for WeakReference<View> access (#4694)
* 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>
2026-02-06 14:44:04 -07:00

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;
}
}