mirror of
https://gitee.com/ant-design-blazor/ant-design-blazor.git
synced 2024-12-15 01:11:52 +08:00
152a574577
* feat(module:overlay): OverlayTrigger not bound to a div * feat(module:overlay): OverlayTrigger not bound to a div * feat(module:overlay): Logic transfer to single Overlay * feat(module:overlay): remove obsolete duplication * feat(module:Tooltip): Add for unbounded oncontextmenu event handler * feat(module:tooltip): unbound js event listeners remove * docs(module:tooltip): unbound explanation * fix(module:button): attach Ref to top level html element @ref * feat(module:dropdown&tooltip&popconfirm&popover): Overlay not bound to a div * docs(module:dropdown&tooltip&popconfirm&popover): unbound explanation * feat(module:OverlayTrigger): common logic relocation * feat(module:overlaytrigger): Overlay not bound to a div * feat(module:DatePicker): Overlay not bound to a div * feat(module:select): Overlay not boud to div * fix(module:select): onclickarrow event relocation * fix(module:select): rename Show to OnArrowClick * feat(module:avatar): Overlay not bound to a div * docs(module:avatar): demo switch to unbound version * feat(module:autocomplete): partial OverlayTrigger not bound to a div * feat(module:slider): tooltip * docs(module:slider): tooltip * fix(module:overlay): add SetVisible method * feat: set Ref where missing, performance components register Ref when missing IsFixed flag for CascadeValue changed hard-code sequence numbers when using RenderTreeBuilder Rate component use Tooltip Unbound version Tabs test fix * fix: revert changes (accidental) * feat(module:upload): tooltip with unbound usage * feat(module:table): column use of unbound tooltip * feat(module:autocomplete):overlay unbound from div * fix(module:upload): missing div restore Co-authored-by: James Yeung <shunjiey@hotmail.com>
434 lines
13 KiB
C#
434 lines
13 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Data;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using AntDesign.Internal;
|
|
using AntDesign.JsInterop;
|
|
using Microsoft.AspNetCore.Components;
|
|
using Microsoft.AspNetCore.Components.Web;
|
|
using OneOf;
|
|
|
|
namespace AntDesign
|
|
{
|
|
public partial class AutoComplete<TOption> : AntInputComponentBase<string>, IAutoCompleteRef
|
|
{
|
|
#region Parameters
|
|
|
|
[Parameter]
|
|
public string Placeholder { get; set; }
|
|
|
|
[Parameter]
|
|
public bool Disabled { get; set; }
|
|
|
|
[Parameter]
|
|
public bool DefaultActiveFirstOption { get; set; } = true;
|
|
[Parameter]
|
|
public bool Backfill { get; set; } = false;
|
|
|
|
/// <summary>
|
|
/// 列表对象集合
|
|
/// </summary>
|
|
private List<AutoCompleteOption> AutoCompleteOptions { get; set; } = new List<AutoCompleteOption>();
|
|
|
|
/// <summary>
|
|
/// 列表数据集合
|
|
/// </summary>
|
|
private List<AutoCompleteDataItem<TOption>> _optionDataItems = new List<AutoCompleteDataItem<TOption>>();
|
|
|
|
/// <summary>
|
|
/// 列表绑定数据源集合
|
|
/// </summary>
|
|
private IEnumerable<TOption> _options;
|
|
[Parameter]
|
|
public IEnumerable<TOption> Options
|
|
{
|
|
get
|
|
{
|
|
return _options;
|
|
}
|
|
set
|
|
{
|
|
_options = value;
|
|
_optionDataItems = _options?.Select(x => new AutoCompleteDataItem<TOption>(x, x.ToString())).ToList() ?? new List<AutoCompleteDataItem<TOption>>();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 绑定列表数据项格式的数据源
|
|
/// </summary>
|
|
[Parameter]
|
|
public IEnumerable<AutoCompleteDataItem<TOption>> OptionDataItems
|
|
{
|
|
get
|
|
{
|
|
return _optionDataItems;
|
|
}
|
|
set
|
|
{
|
|
_optionDataItems = value.ToList();
|
|
}
|
|
}
|
|
|
|
[Parameter]
|
|
public EventCallback<AutoCompleteOption> OnSelectionChange { get; set; }
|
|
[Parameter]
|
|
public EventCallback<AutoCompleteOption> OnActiveChange { get; set; }
|
|
|
|
[Parameter]
|
|
public EventCallback<ChangeEventArgs> OnInput { get; set; }
|
|
|
|
[Parameter]
|
|
public EventCallback<bool> OnPanelVisibleChange { get; set; }
|
|
|
|
[Parameter]
|
|
public RenderFragment ChildContent { get; set; }
|
|
|
|
/// <summary>
|
|
/// 选项模板
|
|
/// </summary>
|
|
[Parameter]
|
|
public RenderFragment<AutoCompleteDataItem<TOption>> OptionTemplate { get; set; }
|
|
|
|
/// <summary>
|
|
/// 格式化选项,可以自定义显示格式
|
|
/// </summary>
|
|
[Parameter]
|
|
public Func<AutoCompleteDataItem<TOption>, string> OptionFormat { get; set; }
|
|
|
|
/// <summary>
|
|
/// 所有选项模板
|
|
/// </summary>
|
|
[Parameter]
|
|
public RenderFragment OverlayTemplate { get; set; }
|
|
|
|
/// <summary>
|
|
/// 对比,用于两个对象比较是否相同
|
|
/// </summary>
|
|
[Parameter]
|
|
public Func<object, object, bool> CompareWith { get; set; } = (o1, o2) => o1?.ToString() == o2?.ToString();
|
|
|
|
/// <summary>
|
|
/// 过滤表达式
|
|
/// </summary>
|
|
[Parameter]
|
|
public Func<AutoCompleteDataItem<TOption>, string, bool> FilterExpression { get; set; } = (option, value) => string.IsNullOrEmpty(value) ? true : option.Label.Contains(value, StringComparison.InvariantCultureIgnoreCase);
|
|
|
|
/// <summary>
|
|
/// 允许过滤
|
|
/// </summary>
|
|
[Parameter]
|
|
public bool AllowFilter { get; set; } = true;
|
|
|
|
[Parameter]
|
|
public OneOf<int?, string> Width { get; set; }
|
|
|
|
[Parameter]
|
|
public string OverlayClassName { get; set; }
|
|
|
|
[Parameter]
|
|
public string OverlayStyle { get; set; }
|
|
|
|
[Parameter]
|
|
public string PopupContainerSelector { get; set; } = "body";
|
|
|
|
#endregion Parameters
|
|
|
|
//private ElementReference _divRef;
|
|
private OverlayTrigger _overlayTrigger;
|
|
|
|
public object SelectedValue { get; set; }
|
|
/// <summary>
|
|
/// 选择的项
|
|
/// </summary>
|
|
[Parameter]
|
|
public AutoCompleteOption SelectedItem { get; set; }
|
|
|
|
/// <summary>
|
|
/// 高亮的项目
|
|
/// </summary>
|
|
public object ActiveValue { get; set; }
|
|
|
|
|
|
[Parameter]
|
|
public bool ShowPanel { get; set; } = false;
|
|
|
|
private bool _isOptionsZero = true;
|
|
|
|
private IAutoCompleteInput _inputComponent;
|
|
|
|
public void SetInputComponent(IAutoCompleteInput input)
|
|
{
|
|
_inputComponent = input;
|
|
}
|
|
|
|
#region 子控件触发事件
|
|
public async Task InputFocus(FocusEventArgs e)
|
|
{
|
|
if (!_isOptionsZero)
|
|
{
|
|
this.OpenPanel();
|
|
}
|
|
}
|
|
|
|
public async Task InputInput(ChangeEventArgs args)
|
|
{
|
|
SelectedValue = args?.Value;
|
|
if (OnInput.HasDelegate) await OnInput.InvokeAsync(args);
|
|
StateHasChanged();
|
|
}
|
|
|
|
public async Task InputKeyDown(KeyboardEventArgs args)
|
|
{
|
|
var key = args.Key;
|
|
|
|
if (this.ShowPanel)
|
|
{
|
|
if (key == "Escape" || key == "Tab")
|
|
{
|
|
this.ClosePanel();
|
|
}
|
|
else if (key == "Enter" && this.ActiveValue != null)
|
|
{
|
|
await SetSelectedItem(GetActiveItem());
|
|
}
|
|
}
|
|
else if (!_isOptionsZero)
|
|
{
|
|
this.OpenPanel();
|
|
}
|
|
|
|
if (key == "ArrowUp")
|
|
{
|
|
this.SetPreviousItemActive();
|
|
}
|
|
else if (key == "ArrowDown")
|
|
{
|
|
this.SetNextItemActive();
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
protected override void OnParametersSet()
|
|
{
|
|
base.OnParametersSet();
|
|
ResetActiveItem();
|
|
}
|
|
|
|
protected override async Task OnFirstAfterRenderAsync()
|
|
{
|
|
await SetOverlayWidth();
|
|
await base.OnFirstAfterRenderAsync();
|
|
}
|
|
|
|
|
|
|
|
public void AddOption(AutoCompleteOption option)
|
|
{
|
|
AutoCompleteOptions.Add(option);
|
|
}
|
|
|
|
public void RemoveOption(AutoCompleteOption option)
|
|
{
|
|
if (AutoCompleteOptions?.Contains(option) == true)
|
|
AutoCompleteOptions?.Remove(option);
|
|
}
|
|
|
|
public IList<AutoCompleteDataItem<TOption>> GetOptionItems()
|
|
{
|
|
if (_optionDataItems != null)
|
|
{
|
|
if (FilterExpression != null && AllowFilter == true && SelectedValue != null)
|
|
return _optionDataItems.Where(x => FilterExpression(x, SelectedValue?.ToString())).ToList();
|
|
else
|
|
return _optionDataItems;
|
|
}
|
|
else
|
|
{
|
|
return new List<AutoCompleteDataItem<TOption>>();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 打开面板
|
|
/// </summary>
|
|
public void OpenPanel()
|
|
{
|
|
if (this.ShowPanel == false)
|
|
{
|
|
this.ShowPanel = true;
|
|
|
|
_overlayTrigger.Show();
|
|
|
|
ResetActiveItem();
|
|
StateHasChanged();
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// 关闭面板
|
|
/// </summary>
|
|
public void ClosePanel()
|
|
{
|
|
if (this.ShowPanel == true)
|
|
{
|
|
this.ShowPanel = false;
|
|
|
|
_overlayTrigger.Close();
|
|
|
|
StateHasChanged();
|
|
}
|
|
}
|
|
|
|
public AutoCompleteOption GetActiveItem()
|
|
{
|
|
return AutoCompleteOptions.FirstOrDefault(x => CompareWith(x.Value, this.ActiveValue));
|
|
}
|
|
|
|
//设置高亮的对象
|
|
public void SetActiveItem(AutoCompleteOption item)
|
|
{
|
|
this.ActiveValue = item == null ? default(TOption) : item.Value;
|
|
if (OnActiveChange.HasDelegate) OnActiveChange.InvokeAsync(item);
|
|
StateHasChanged();
|
|
}
|
|
|
|
//设置下一个激活
|
|
public void SetNextItemActive()
|
|
{
|
|
var opts = AutoCompleteOptions.Where(x => x.Disabled == false).ToList();
|
|
var nextItem = opts.IndexOf(GetActiveItem());
|
|
if (nextItem == -1 || nextItem == opts.Count - 1)
|
|
SetActiveItem(opts.FirstOrDefault());
|
|
else
|
|
SetActiveItem(opts[nextItem + 1]);
|
|
|
|
if (Backfill)
|
|
_inputComponent.SetValue(this.ActiveValue);
|
|
}
|
|
|
|
//设置上一个激活
|
|
public void SetPreviousItemActive()
|
|
{
|
|
var opts = AutoCompleteOptions.Where(x => x.Disabled == false).ToList();
|
|
var nextItem = opts.IndexOf(GetActiveItem());
|
|
if (nextItem == -1 || nextItem == 0)
|
|
SetActiveItem(opts.LastOrDefault());
|
|
else
|
|
SetActiveItem(opts[nextItem - 1]);
|
|
|
|
if (Backfill)
|
|
_inputComponent.SetValue(this.ActiveValue);
|
|
}
|
|
|
|
private void ResetActiveItem()
|
|
{
|
|
var items = GetOptionItems();
|
|
_isOptionsZero = items.Count == 0 && Options != null;
|
|
if (items.Any(x => CompareWith(x.Value, this.ActiveValue)) == false)
|
|
{
|
|
// 如果当前激活项找在列表中不存在,那么我们需要做一些处理
|
|
if (items.Any(x => CompareWith(x.Value, this.SelectedValue)))
|
|
{
|
|
this.ActiveValue = this.SelectedValue;
|
|
}
|
|
else if (DefaultActiveFirstOption == true && items.Count > 0)
|
|
{
|
|
this.ActiveValue = items.FirstOrDefault().Value;
|
|
}
|
|
else
|
|
{
|
|
this.ActiveValue = null;
|
|
}
|
|
}
|
|
|
|
if (_overlayTrigger != null && ShowPanel)
|
|
{
|
|
// if options count == 0 then close overlay
|
|
if (_isOptionsZero && _overlayTrigger.IsOverlayShow())
|
|
{
|
|
_overlayTrigger.Close();
|
|
}
|
|
// if options count > 0 then open overlay
|
|
else if (!_isOptionsZero && !_overlayTrigger.IsOverlayShow())
|
|
{
|
|
_overlayTrigger.Show();
|
|
}
|
|
}
|
|
}
|
|
|
|
public async Task SetSelectedItem(AutoCompleteOption item)
|
|
{
|
|
if (item != null)
|
|
{
|
|
this.SelectedValue = item.Value;
|
|
this.SelectedItem = item;
|
|
_inputComponent?.SetValue(this.SelectedItem.Label);
|
|
|
|
if (OnSelectionChange.HasDelegate) await OnSelectionChange.InvokeAsync(this.SelectedItem);
|
|
}
|
|
this.ClosePanel();
|
|
}
|
|
|
|
bool _parPanelVisible = false;
|
|
|
|
private async void OnOverlayTriggerVisibleChange(bool visible)
|
|
{
|
|
if (OnPanelVisibleChange.HasDelegate && _parPanelVisible != visible)
|
|
{
|
|
_parPanelVisible = visible;
|
|
await OnPanelVisibleChange.InvokeAsync(visible);
|
|
}
|
|
|
|
if (this.ShowPanel != visible)
|
|
{
|
|
this.ShowPanel = visible;
|
|
}
|
|
}
|
|
|
|
private string _minWidth = "";
|
|
private async Task SetOverlayWidth()
|
|
{
|
|
string newWidth;
|
|
if (Width.Value != null)
|
|
{
|
|
var w = Width.Match<string>(f0 => $"{f0}px", f1 => f1);
|
|
newWidth = $"min-width:{w}";
|
|
}
|
|
else
|
|
{
|
|
Element element = await JsInvokeAsync<Element>(JSInteropConstants.GetDomInfo, _overlayTrigger.RefBack.Current); ;
|
|
//Element element;
|
|
//if (_divRef.Id != null)
|
|
//{
|
|
// element = await JsInvokeAsync<Element>(JSInteropConstants.GetDomInfo, _divRef);
|
|
//}
|
|
//else
|
|
//{
|
|
// element = await JsInvokeAsync<Element>(JSInteropConstants.GetDomInfo, _overlayTrigger.RefBack.Current);
|
|
//}
|
|
newWidth = $"min-width:{element.clientWidth}px";
|
|
}
|
|
if (newWidth != _minWidth) _minWidth = newWidth;
|
|
}
|
|
}
|
|
|
|
public class AutoCompleteDataItem<TOption>
|
|
{
|
|
public AutoCompleteDataItem() { }
|
|
|
|
public AutoCompleteDataItem(TOption value, string label)
|
|
{
|
|
Value = value;
|
|
Label = label;
|
|
}
|
|
|
|
public TOption Value { get; set; }
|
|
public string Label { get; set; }
|
|
|
|
public bool IsDisabled { get; set; }
|
|
}
|
|
}
|
|
|