2020-06-01 14:09:28 +08:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.Threading.Tasks;
|
2021-01-10 13:19:07 +08:00
|
|
|
|
using AntDesign.core.Extensions;
|
2021-04-28 11:54:19 +08:00
|
|
|
|
using AntDesign.Datepicker.Locale;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
using AntDesign.Internal;
|
|
|
|
|
using AntDesign.JsInterop;
|
|
|
|
|
using Microsoft.AspNetCore.Components;
|
2021-04-28 11:54:19 +08:00
|
|
|
|
using Microsoft.JSInterop;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
using OneOf;
|
|
|
|
|
|
|
|
|
|
namespace AntDesign
|
|
|
|
|
{
|
2020-06-03 10:24:16 +08:00
|
|
|
|
public class DatePickerBase<TValue> : AntInputComponentBase<TValue>, IDatePicker
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2020-07-05 00:06:34 +08:00
|
|
|
|
DateTime? IDatePicker.HoverDateTime { get; set; }
|
|
|
|
|
|
|
|
|
|
public const int START_PICKER_INDEX = 0;
|
|
|
|
|
public const int END_PICKER_INDEX = 0;
|
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
[Parameter]
|
|
|
|
|
public string PrefixCls { get; set; } = "ant-picker";
|
|
|
|
|
|
|
|
|
|
protected string _picker;
|
|
|
|
|
protected bool _isSetPicker = false;
|
2021-01-10 13:19:07 +08:00
|
|
|
|
private bool _isNullableEvaluated;
|
|
|
|
|
private bool _isNullable;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Stores information if TValue is a nullable type.
|
|
|
|
|
/// </summary>
|
|
|
|
|
protected bool IsNullable
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
if (!_isNullableEvaluated)
|
|
|
|
|
{
|
|
|
|
|
Type type = typeof(TValue);
|
|
|
|
|
if (type.IsAssignableFrom(typeof(DateTime?)) || type.IsAssignableFrom(typeof(DateTime?[])))
|
|
|
|
|
{
|
|
|
|
|
_isNullable = true;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_isNullable = false;
|
|
|
|
|
}
|
|
|
|
|
_isNullableEvaluated = true;
|
|
|
|
|
}
|
|
|
|
|
return _isNullable;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public string Picker
|
|
|
|
|
{
|
|
|
|
|
get => _picker;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
_isSetPicker = true;
|
|
|
|
|
_picker = value;
|
2020-06-16 12:31:46 +08:00
|
|
|
|
InitPicker(value);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public string PopupContainerSelector { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool Disabled { get; set; } = false;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool Bordered { get; set; } = true;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool AutoFocus { get; set; } = false;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool Open { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool InputReadOnly { get; set; } = false;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool ShowToday { get; set; } = true;
|
|
|
|
|
|
2020-11-25 11:08:12 +08:00
|
|
|
|
[Parameter]
|
2021-02-05 16:50:12 +08:00
|
|
|
|
public DatePickerLocale Locale
|
|
|
|
|
{
|
|
|
|
|
get { return _locale; }
|
2021-03-12 17:02:11 +08:00
|
|
|
|
set
|
|
|
|
|
{
|
2021-02-05 16:50:12 +08:00
|
|
|
|
_locale = value;
|
|
|
|
|
_isLocaleSetOutside = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-07 22:46:50 +08:00
|
|
|
|
|
2020-06-16 12:31:46 +08:00
|
|
|
|
[Parameter]
|
2021-02-05 16:50:12 +08:00
|
|
|
|
public CultureInfo CultureInfo
|
|
|
|
|
{
|
|
|
|
|
get { return _cultureInfo; }
|
2021-03-12 17:02:11 +08:00
|
|
|
|
set
|
|
|
|
|
{
|
2021-02-05 16:50:12 +08:00
|
|
|
|
_cultureInfo = value;
|
|
|
|
|
_isCultureSetOutside = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-16 12:31:46 +08:00
|
|
|
|
|
2020-11-25 11:08:12 +08:00
|
|
|
|
public bool IsShowTime { get; protected set; }
|
2020-06-01 14:09:28 +08:00
|
|
|
|
public string ShowTimeFormat { get; protected set; } = "HH:mm:ss";
|
|
|
|
|
protected OneOf<bool, string> _showTime = null;
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
private bool _timeFormatProvided;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
[Parameter]
|
|
|
|
|
public OneOf<bool, string> ShowTime
|
|
|
|
|
{
|
|
|
|
|
get => _showTime;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
_showTime = value;
|
|
|
|
|
|
|
|
|
|
value.Switch(booleanValue =>
|
|
|
|
|
{
|
|
|
|
|
IsShowTime = booleanValue;
|
|
|
|
|
}, strValue =>
|
|
|
|
|
{
|
|
|
|
|
IsShowTime = true;
|
2021-04-28 11:54:19 +08:00
|
|
|
|
_timeFormatProvided = true;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
ShowTimeFormat = strValue;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
2020-07-06 19:09:01 +08:00
|
|
|
|
public bool AllowClear { get; set; } = true;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
protected string[] _placeholders = new string[] { "", "" };
|
|
|
|
|
protected OneOf<string, string[]> _placeholder;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public OneOf<string, string[]> Placeholder
|
|
|
|
|
{
|
|
|
|
|
get => _placeholder;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
_placeholder = value;
|
|
|
|
|
value.Switch(single =>
|
|
|
|
|
{
|
|
|
|
|
_placeholders[0] = single;
|
|
|
|
|
}, arr =>
|
|
|
|
|
{
|
|
|
|
|
_placeholders[0] = arr.Length > 0 ? arr[0] : _placeholders[0];
|
|
|
|
|
_placeholders[1] = arr.Length > 1 ? arr[1] : _placeholders[1];
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public string PopupStyle { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public string ClassName { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public string DropdownClassName { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public string Format { get; set; }
|
|
|
|
|
|
2021-01-10 13:19:07 +08:00
|
|
|
|
private TValue _defaultValue;
|
2021-03-12 17:02:11 +08:00
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
[Parameter]
|
2021-01-10 13:19:07 +08:00
|
|
|
|
public TValue DefaultValue
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
|
|
|
|
get => _defaultValue;
|
2021-01-10 13:19:07 +08:00
|
|
|
|
set => _defaultValue = SortValue(value);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-10 13:19:07 +08:00
|
|
|
|
protected bool[] UseDefaultPickerValue { get; } = new bool[2];
|
|
|
|
|
private TValue _defaultPickerValue;
|
2021-03-12 17:02:11 +08:00
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
[Parameter]
|
2021-01-10 13:19:07 +08:00
|
|
|
|
public TValue DefaultPickerValue
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
|
|
|
|
get => _defaultPickerValue;
|
2021-01-10 13:19:07 +08:00
|
|
|
|
set => _defaultPickerValue = SortValue(value);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public RenderFragment SuffixIcon { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public RenderFragment RenderExtraFooter { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public EventCallback<bool> OnOpenChange { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
2020-10-29 01:20:11 +08:00
|
|
|
|
public EventCallback<DateTimeChangedEventArgs> OnPanelChange { get; set; }
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, bool> DisabledDate { get; set; } = null;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, int[]> DisabledHours { get; set; } = null;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, int[]> DisabledMinutes { get; set; } = null;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, int[]> DisabledSeconds { get; set; } = null;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, DatePickerDisabledTime> DisabledTime { get; set; } = null;
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, DateTime, RenderFragment> DateRender { get; set; }
|
|
|
|
|
|
|
|
|
|
// TODO: need locale
|
|
|
|
|
[Parameter]
|
|
|
|
|
public Func<DateTime, RenderFragment> MonthCellRender { get; set; }
|
|
|
|
|
|
2020-06-03 10:24:16 +08:00
|
|
|
|
public DateTime CurrentDate { get; set; } = DateTime.Now;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
2021-01-10 13:19:07 +08:00
|
|
|
|
protected DateTime[] PickerValues { get; } = new DateTime[] { DateTime.Now, DateTime.Now };
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
public bool IsRange { get; set; }
|
|
|
|
|
|
|
|
|
|
protected DatePickerInput _inputStart;
|
|
|
|
|
protected DatePickerInput _inputEnd;
|
|
|
|
|
protected OverlayTrigger _dropDown;
|
|
|
|
|
|
|
|
|
|
protected string _activeBarStyle = "";
|
2020-08-14 12:43:48 +08:00
|
|
|
|
protected string _rangeArrowStyle = "";
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
protected DatePickerStatus[] _pickerStatus
|
|
|
|
|
= new DatePickerStatus[] { new DatePickerStatus(), new DatePickerStatus() };
|
|
|
|
|
|
|
|
|
|
protected Stack<string> _prePickerStack = new Stack<string>();
|
|
|
|
|
protected bool _isClose = true;
|
|
|
|
|
protected bool _needRefresh;
|
2021-04-28 11:54:19 +08:00
|
|
|
|
protected bool _duringManualInput;
|
2021-02-05 16:50:12 +08:00
|
|
|
|
private bool _isCultureSetOutside;
|
|
|
|
|
private bool _isLocaleSetOutside;
|
|
|
|
|
private CultureInfo _cultureInfo = LocaleProvider.CurrentLocale.CurrentCulture;
|
|
|
|
|
private DatePickerLocale _locale = LocaleProvider.CurrentLocale.DatePicker;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
2021-03-12 17:02:11 +08:00
|
|
|
|
protected ClassMapper _panelClassMapper = new ClassMapper();
|
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
protected override void OnInitialized()
|
|
|
|
|
{
|
|
|
|
|
// set default picker type
|
|
|
|
|
if (_isSetPicker == false)
|
|
|
|
|
{
|
|
|
|
|
Picker = DatePickerType.Date;
|
|
|
|
|
}
|
|
|
|
|
this.SetClass();
|
|
|
|
|
|
|
|
|
|
base.OnInitialized();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override Task SetParametersAsync(ParameterView parameters)
|
|
|
|
|
{
|
|
|
|
|
_needRefresh = true;
|
|
|
|
|
|
|
|
|
|
return base.SetParametersAsync(parameters);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected void SetClass()
|
|
|
|
|
{
|
|
|
|
|
this.ClassMapper.Clear()
|
|
|
|
|
.Add(PrefixCls)
|
2021-03-12 17:02:11 +08:00
|
|
|
|
.Get(() => $"{PrefixCls}-{Size}")
|
|
|
|
|
.If($"{PrefixCls}-rtl", () => RTL)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
.If($"{PrefixCls}-borderless", () => Bordered == false)
|
|
|
|
|
.If($"{PrefixCls}-disabled", () => Disabled == true)
|
|
|
|
|
.If($"{ClassName}", () => !string.IsNullOrEmpty(ClassName))
|
|
|
|
|
.If($"{PrefixCls}-range", () => IsRange == true)
|
|
|
|
|
.If($"{PrefixCls}-focused", () => AutoFocus == true)
|
|
|
|
|
//.If($"{PrefixCls}-normal", () => Image.IsT1 && Image.AsT1 == Empty.PRESENTED_IMAGE_SIMPLE)
|
|
|
|
|
//.If($"{PrefixCls}-{Direction}", () => Direction.IsIn("ltr", "rlt"))
|
|
|
|
|
;
|
2021-03-12 17:02:11 +08:00
|
|
|
|
|
|
|
|
|
_panelClassMapper
|
|
|
|
|
.Add($"{PrefixCls}-panel")
|
|
|
|
|
.If($"{PrefixCls}-panel-rtl", () => RTL);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected async override Task OnAfterRenderAsync(bool firstRender)
|
|
|
|
|
{
|
|
|
|
|
await base.OnAfterRenderAsync(firstRender);
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
if (firstRender)
|
|
|
|
|
{
|
|
|
|
|
await Js.InvokeVoidAsync(JSInteropConstants.AddPreventKeys, _inputStart.Ref, new[] { "ArrowUp", "ArrowDown" });
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
if (_needRefresh && IsRange)
|
|
|
|
|
{
|
|
|
|
|
if (_inputStart.IsOnFocused)
|
|
|
|
|
{
|
2021-04-17 22:14:01 +08:00
|
|
|
|
HtmlElement element = await JsInvokeAsync<HtmlElement>(JSInteropConstants.GetDomInfo, _inputStart.Ref);
|
|
|
|
|
_activeBarStyle = $"width: {element.ClientWidth - 10}px; position: absolute; transform: translate3d(0px, 0px, 0px);";
|
2020-08-14 12:43:48 +08:00
|
|
|
|
_rangeArrowStyle = $"left: 12px";
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
else if (_inputEnd.IsOnFocused)
|
|
|
|
|
{
|
2021-04-17 22:14:01 +08:00
|
|
|
|
HtmlElement element = await JsInvokeAsync<HtmlElement>(JSInteropConstants.GetDomInfo, _inputEnd.Ref);
|
|
|
|
|
int translateDistance = element.ClientWidth + 16;
|
2021-04-06 20:02:53 +08:00
|
|
|
|
|
|
|
|
|
if (RTL)
|
|
|
|
|
{
|
|
|
|
|
translateDistance = -translateDistance;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-17 22:14:01 +08:00
|
|
|
|
_activeBarStyle = $"width: {element.ClientWidth - 10}px; position: absolute; transform: translate3d({translateDistance}px, 0px, 0px);";
|
|
|
|
|
_rangeArrowStyle = $"left: {element.ClientWidth + 30}px";
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_activeBarStyle = "display: none";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
StateHasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_needRefresh = false;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
protected override void Dispose(bool disposing)
|
|
|
|
|
{
|
|
|
|
|
base.Dispose(disposing);
|
|
|
|
|
_ = InvokeAsync(async () =>
|
|
|
|
|
{
|
|
|
|
|
await Js.InvokeVoidAsync(JSInteropConstants.RemovePreventKeys, _inputStart.Ref);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
protected string GetInputValue(int index = 0)
|
|
|
|
|
{
|
|
|
|
|
DateTime? tryGetValue = GetIndexValue(index);
|
|
|
|
|
if (tryGetValue == null)
|
|
|
|
|
{
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DateTime value = (DateTime)tryGetValue;
|
|
|
|
|
|
2020-08-14 12:43:48 +08:00
|
|
|
|
return GetFormatValue(value, index);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected void ChangeFocusTarget(bool inputStartFocus, bool inputEndFocus)
|
|
|
|
|
{
|
|
|
|
|
if (!IsRange)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2021-04-28 11:54:19 +08:00
|
|
|
|
_duringManualInput = false;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
_needRefresh = true;
|
|
|
|
|
_inputStart.IsOnFocused = inputStartFocus;
|
|
|
|
|
_inputEnd.IsOnFocused = inputEndFocus;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
protected virtual async Task OnSelect(DateTime date)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2021-04-28 11:54:19 +08:00
|
|
|
|
int index = GetOnFocusPickerIndex();
|
|
|
|
|
_duringManualInput = false;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
// InitPicker is the finally value
|
|
|
|
|
if (_picker == _pickerStatus[index]._initPicker)
|
|
|
|
|
{
|
|
|
|
|
ChangeValue(date, index);
|
|
|
|
|
|
|
|
|
|
// auto focus the other input
|
|
|
|
|
if (IsRange && (!IsShowTime || Picker == DatePickerType.Time))
|
|
|
|
|
{
|
2020-08-14 12:43:48 +08:00
|
|
|
|
if (index == 0 && !_pickerStatus[1]._currentShowHadSelectValue && !_inputEnd.IsOnFocused)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
|
|
|
|
await Blur(0);
|
|
|
|
|
await Focus(1);
|
|
|
|
|
}
|
2020-08-14 12:43:48 +08:00
|
|
|
|
else if (index == 1 && !_pickerStatus[0]._currentShowHadSelectValue && !_inputStart.IsOnFocused)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
|
|
|
|
await Blur(1);
|
|
|
|
|
await Focus(0);
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-04-28 11:54:19 +08:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
await Focus(index);
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_picker = _prePickerStack.Pop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ChangePickerValue(date, index);
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
protected virtual async Task OnBlur(int index)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-16 12:31:46 +08:00
|
|
|
|
protected void InitPicker(string picker)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2021-02-05 16:50:12 +08:00
|
|
|
|
if (_isCultureSetOutside && !_isLocaleSetOutside)
|
|
|
|
|
Locale = LocaleProvider.GetLocale(_cultureInfo.Name).DatePicker;
|
|
|
|
|
|
2020-06-16 12:31:46 +08:00
|
|
|
|
if (string.IsNullOrEmpty(_pickerStatus[0]._initPicker))
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2020-06-16 12:31:46 +08:00
|
|
|
|
_pickerStatus[0]._initPicker = picker;
|
|
|
|
|
}
|
|
|
|
|
if (string.IsNullOrEmpty(_pickerStatus[1]._initPicker))
|
|
|
|
|
{
|
|
|
|
|
_pickerStatus[1]._initPicker = picker;
|
|
|
|
|
}
|
|
|
|
|
if (IsRange)
|
|
|
|
|
{
|
2020-11-25 11:08:12 +08:00
|
|
|
|
(string first, string second) = DatePickerPlaceholder.GetRangePlaceHolderByType(picker, Locale);
|
2020-06-16 12:31:46 +08:00
|
|
|
|
_placeholders[0] = first;
|
|
|
|
|
_placeholders[1] = second;
|
2021-01-10 13:19:07 +08:00
|
|
|
|
|
2020-06-16 12:31:46 +08:00
|
|
|
|
DateTime now = DateTime.Now;
|
2020-09-09 00:52:14 +08:00
|
|
|
|
PickerValues[1] = picker switch
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2020-06-16 12:31:46 +08:00
|
|
|
|
DatePickerType.Date => now.AddMonths(1),
|
|
|
|
|
DatePickerType.Week => now.AddMonths(1),
|
|
|
|
|
DatePickerType.Month => now.AddYears(1),
|
|
|
|
|
DatePickerType.Decade => now.AddYears(1),
|
|
|
|
|
DatePickerType.Quarter => now.AddYears(1),
|
|
|
|
|
DatePickerType.Year => now.AddYears(10),
|
|
|
|
|
_ => now,
|
|
|
|
|
};
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
2021-01-10 13:19:07 +08:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
string first = DatePickerPlaceholder.GetPlaceholderByType(picker, Locale);
|
|
|
|
|
_placeholders[0] = first;
|
|
|
|
|
_placeholders[1] = first;
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-14 12:43:48 +08:00
|
|
|
|
protected virtual void UpdateCurrentValueAsString(int index = 0)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
|
|
|
|
if (EditContext != null)
|
|
|
|
|
{
|
|
|
|
|
CurrentValueAsString = GetInputValue(index);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Close()
|
|
|
|
|
{
|
2021-04-28 11:54:19 +08:00
|
|
|
|
_duringManualInput = false;
|
2020-07-07 19:13:34 +08:00
|
|
|
|
_dropDown?.Hide();
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task Focus(int index = 0)
|
|
|
|
|
{
|
|
|
|
|
DatePickerInput input = null;
|
|
|
|
|
|
|
|
|
|
if (index == 0)
|
|
|
|
|
{
|
|
|
|
|
input = _inputStart;
|
|
|
|
|
}
|
|
|
|
|
else if (index == 1 && IsRange)
|
|
|
|
|
{
|
|
|
|
|
input = _inputEnd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (input != null)
|
|
|
|
|
{
|
|
|
|
|
input.IsOnFocused = true;
|
2021-04-15 14:19:26 +08:00
|
|
|
|
await FocusAsync(input.Ref);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
_needRefresh = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task Blur(int index = 0)
|
|
|
|
|
{
|
|
|
|
|
DatePickerInput input = null;
|
2021-04-28 11:54:19 +08:00
|
|
|
|
_duringManualInput = false;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
if (index == 0)
|
|
|
|
|
{
|
|
|
|
|
input = _inputStart;
|
|
|
|
|
}
|
|
|
|
|
else if (index == 1 && IsRange)
|
|
|
|
|
{
|
|
|
|
|
input = _inputEnd;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (input != null)
|
|
|
|
|
{
|
|
|
|
|
input.IsOnFocused = false;
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.Blur, input.Ref);
|
2020-06-01 14:09:28 +08:00
|
|
|
|
_needRefresh = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int GetOnFocusPickerIndex()
|
|
|
|
|
{
|
|
|
|
|
if (_inputStart != null && _inputStart.IsOnFocused)
|
|
|
|
|
{
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_inputEnd != null && _inputEnd.IsOnFocused)
|
|
|
|
|
{
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2021-03-12 17:02:11 +08:00
|
|
|
|
/// Get pickerValue by picker index. Note that index refers to a picker panel
|
2021-01-10 13:19:07 +08:00
|
|
|
|
/// and not to input text. For RangePicker 2 inputs generate 2 panels.
|
2020-06-01 14:09:28 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="index"></param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
public DateTime GetIndexPickerValue(int index)
|
|
|
|
|
{
|
2021-01-10 13:19:07 +08:00
|
|
|
|
int tempIndex = GetOnFocusPickerIndex();
|
|
|
|
|
if (index == 0)
|
|
|
|
|
{
|
|
|
|
|
return PickerValues[tempIndex];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-03-12 17:02:11 +08:00
|
|
|
|
//First picker panel will show the value, second panel shows next
|
2021-01-10 13:19:07 +08:00
|
|
|
|
//expected value that depends on Picker type
|
|
|
|
|
return Picker switch
|
|
|
|
|
{
|
|
|
|
|
DatePickerType.Date => PickerValues[tempIndex].AddMonths(1),
|
|
|
|
|
DatePickerType.Week => PickerValues[tempIndex].AddMonths(1),
|
|
|
|
|
DatePickerType.Month => PickerValues[tempIndex].AddYears(1),
|
|
|
|
|
DatePickerType.Decade => PickerValues[tempIndex].AddYears(1),
|
|
|
|
|
DatePickerType.Quarter => PickerValues[tempIndex].AddYears(1),
|
|
|
|
|
DatePickerType.Year => PickerValues[tempIndex].AddYears(10),
|
|
|
|
|
_ => DateTime.Now,
|
|
|
|
|
};
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-14 12:43:48 +08:00
|
|
|
|
public void ChangePlaceholder(string placeholder, int index = 0)
|
|
|
|
|
{
|
|
|
|
|
_placeholders[index] = placeholder;
|
|
|
|
|
|
|
|
|
|
StateHasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
private string _internalFormat;
|
|
|
|
|
private string InternalFormat
|
2020-08-14 12:43:48 +08:00
|
|
|
|
{
|
2021-04-28 11:54:19 +08:00
|
|
|
|
get
|
2020-08-14 12:43:48 +08:00
|
|
|
|
{
|
2021-04-28 11:54:19 +08:00
|
|
|
|
if (string.IsNullOrEmpty(_internalFormat))
|
|
|
|
|
{
|
|
|
|
|
if (!string.IsNullOrEmpty(Format))
|
|
|
|
|
_internalFormat = Format;
|
|
|
|
|
else
|
|
|
|
|
_internalFormat = _pickerStatus[0]._initPicker switch
|
|
|
|
|
{
|
|
|
|
|
DatePickerType.Date => GetTimeFormat(),
|
|
|
|
|
DatePickerType.Month => Locale.Lang.YearMonthFormat,
|
|
|
|
|
DatePickerType.Year => Locale.Lang.YearFormat,
|
|
|
|
|
DatePickerType.Time => Locale.Lang.TimeFormat,
|
|
|
|
|
DatePickerType.Week => $"{Locale.Lang.YearFormat}-0{Locale.Lang.Week}",
|
|
|
|
|
DatePickerType.Quarter => $"{Locale.Lang.YearFormat}-Q0",
|
|
|
|
|
_ => Locale.Lang.DateFormat,
|
|
|
|
|
};
|
2020-08-14 12:43:48 +08:00
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
}
|
|
|
|
|
return _internalFormat;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
private string GetTimeFormat()
|
|
|
|
|
{
|
|
|
|
|
if (IsShowTime)
|
2020-08-14 12:43:48 +08:00
|
|
|
|
{
|
2021-04-28 11:54:19 +08:00
|
|
|
|
if (_timeFormatProvided)
|
|
|
|
|
{
|
|
|
|
|
return $"{Locale.Lang.DateFormat} {ShowTimeFormat}";
|
|
|
|
|
}
|
|
|
|
|
return Locale.Lang.DateTimeFormat;
|
|
|
|
|
}
|
|
|
|
|
return Locale.Lang.DateFormat;
|
|
|
|
|
}
|
2020-08-14 12:43:48 +08:00
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
private FormatAnalyzer _formatAnalyzer;
|
|
|
|
|
public FormatAnalyzer FormatAnalyzer => _formatAnalyzer ??= new(InternalFormat, Picker, Locale);
|
|
|
|
|
|
|
|
|
|
public string GetFormatValue(DateTime value, int index)
|
|
|
|
|
{
|
|
|
|
|
string format;
|
|
|
|
|
if (string.IsNullOrEmpty(Format))
|
|
|
|
|
format = _pickerStatus[index]._initPicker switch
|
|
|
|
|
{
|
|
|
|
|
DatePickerType.Week => $"{Locale.Lang.YearFormat}-{DateHelper.GetWeekOfYear(value)}{Locale.Lang.Week}",
|
|
|
|
|
DatePickerType.Quarter => $"{Locale.Lang.YearFormat}-{DateHelper.GetDayOfQuarter(value)}",
|
|
|
|
|
_ => InternalFormat,
|
|
|
|
|
};
|
|
|
|
|
else
|
|
|
|
|
format = InternalFormat;
|
|
|
|
|
return value.ToString(format, CultureInfo.InvariantCulture);
|
2020-08-14 12:43:48 +08:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-10 13:19:07 +08:00
|
|
|
|
/// <summary>
|
2021-03-12 17:02:11 +08:00
|
|
|
|
/// Changes what date(s) will be visible on the picker.
|
2021-01-10 13:19:07 +08:00
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="date">New date to be saved.</param>
|
|
|
|
|
/// <param name="index">Index of the input box, where 0 = inputStart and 1 = inputEnd (only RangePicker)</param>
|
|
|
|
|
internal void ChangePickerValue(DateTime date, int? index = null)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2021-01-10 13:19:07 +08:00
|
|
|
|
if (index == null)
|
|
|
|
|
index = GetOnFocusPickerIndex();
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
2021-01-10 13:19:07 +08:00
|
|
|
|
PickerValues[index.Value] = date;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
if (IsRange)
|
|
|
|
|
{
|
2021-01-10 13:19:07 +08:00
|
|
|
|
if (!UseDefaultPickerValue[1] && !_pickerStatus[1]._hadSelectValue && index == 0)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2021-01-10 13:19:07 +08:00
|
|
|
|
PickerValues[1] = date;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
2021-01-10 13:19:07 +08:00
|
|
|
|
else if (!UseDefaultPickerValue[0] && !_pickerStatus[0]._hadSelectValue && index == 1)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
2021-01-10 13:19:07 +08:00
|
|
|
|
PickerValues[0] = date;
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-29 01:20:11 +08:00
|
|
|
|
if (OnPanelChange.HasDelegate)
|
|
|
|
|
{
|
|
|
|
|
OnPanelChange.InvokeAsync(new DateTimeChangedEventArgs
|
|
|
|
|
{
|
2021-01-10 13:19:07 +08:00
|
|
|
|
Date = PickerValues[index.Value],
|
2020-10-29 01:20:11 +08:00
|
|
|
|
DateString = _picker
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
StateHasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-03 10:24:16 +08:00
|
|
|
|
public void ChangePickerType(string type)
|
|
|
|
|
{
|
|
|
|
|
ChangePickerType(type, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public virtual void ChangePickerType(string type, int index)
|
2020-06-01 14:09:28 +08:00
|
|
|
|
{
|
|
|
|
|
_prePickerStack.Push(_picker);
|
|
|
|
|
_picker = type;
|
|
|
|
|
|
2020-10-29 01:20:11 +08:00
|
|
|
|
if (OnPanelChange.HasDelegate)
|
|
|
|
|
{
|
|
|
|
|
OnPanelChange.InvokeAsync(new DateTimeChangedEventArgs
|
|
|
|
|
{
|
|
|
|
|
Date = PickerValues[index],
|
|
|
|
|
DateString = _picker
|
|
|
|
|
});
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
|
|
|
|
|
StateHasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-07 19:13:34 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// 修改值
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="value"></param>
|
|
|
|
|
/// <param name="index"></param>
|
2020-06-01 14:09:28 +08:00
|
|
|
|
public virtual void ChangeValue(DateTime value, int index = 0)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-28 11:54:19 +08:00
|
|
|
|
public virtual void ClearValue(int index = 0, bool closeDropdown = true)
|
2020-07-06 19:09:01 +08:00
|
|
|
|
{
|
|
|
|
|
}
|
2020-07-11 23:03:13 +08:00
|
|
|
|
|
2020-06-01 14:09:28 +08:00
|
|
|
|
public virtual DateTime? GetIndexValue(int index)
|
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
2020-07-05 00:06:34 +08:00
|
|
|
|
|
|
|
|
|
public void InvokeStateHasChanged()
|
|
|
|
|
{
|
|
|
|
|
StateHasChanged();
|
|
|
|
|
}
|
2021-01-10 13:19:07 +08:00
|
|
|
|
|
|
|
|
|
protected TValue SortValue(TValue value)
|
|
|
|
|
{
|
|
|
|
|
if (value == null)
|
|
|
|
|
{
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
TValue orderedValue = value;
|
|
|
|
|
if (IsRange)
|
|
|
|
|
{
|
|
|
|
|
if (IsNullable)
|
|
|
|
|
{
|
|
|
|
|
var tempValue = value as DateTime?[];
|
2021-02-05 10:27:32 +08:00
|
|
|
|
if (tempValue[0] == null || tempValue[1] == null)
|
|
|
|
|
return orderedValue;
|
|
|
|
|
|
2021-01-10 13:19:07 +08:00
|
|
|
|
if ((tempValue[0] ?? DateTime.Now).CompareTo((tempValue[1] ?? DateTime.Now)) > 0)
|
|
|
|
|
orderedValue = DataConvertionExtensions.Convert<DateTime?[], TValue>(new DateTime?[] { tempValue[1], tempValue[0] });
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var tempValue = value as DateTime[];
|
|
|
|
|
if (tempValue[0].CompareTo(tempValue[1]) > 0)
|
|
|
|
|
orderedValue = DataConvertionExtensions.Convert<DateTime[], TValue>(new DateTime[] { tempValue[1], tempValue[0] });
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return orderedValue;
|
|
|
|
|
}
|
2020-06-01 14:09:28 +08:00
|
|
|
|
}
|
|
|
|
|
}
|