2020-06-02 14:25:44 +08:00
|
|
|
|
using System;
|
2020-08-26 07:28:44 +08:00
|
|
|
|
using System.Globalization;
|
|
|
|
|
using System.Text.RegularExpressions;
|
2020-06-02 14:25:44 +08:00
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Microsoft.AspNetCore.Components;
|
|
|
|
|
using Microsoft.AspNetCore.Components.Web;
|
|
|
|
|
using Microsoft.JSInterop;
|
|
|
|
|
|
|
|
|
|
namespace AntDesign
|
|
|
|
|
{
|
|
|
|
|
public partial class Dialog
|
|
|
|
|
{
|
|
|
|
|
private const string IdPrefix = "Ant-Design-";
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public DialogOptions Config { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public RenderFragment ChildContent { get; set; }
|
|
|
|
|
|
|
|
|
|
[Parameter]
|
|
|
|
|
public bool Visible { get; set; }
|
|
|
|
|
|
|
|
|
|
private string _maskAnimation = "";
|
|
|
|
|
private string _modalAnimation = "";
|
|
|
|
|
private string _maskHideClsName = "";
|
|
|
|
|
|
|
|
|
|
private bool _hasShow;
|
|
|
|
|
private bool _hasDestroy = true;
|
|
|
|
|
private string _wrapStyle = "";
|
2020-06-04 17:03:13 +08:00
|
|
|
|
private bool _disableBodyScroll;
|
2020-08-26 07:28:44 +08:00
|
|
|
|
private bool _doDragable = false;
|
|
|
|
|
|
2020-06-02 14:25:44 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// dialog root container
|
|
|
|
|
/// </summary>
|
|
|
|
|
private ElementReference _element;
|
|
|
|
|
|
2020-08-26 07:28:44 +08:00
|
|
|
|
private ElementReference _dialogHeader;
|
|
|
|
|
|
|
|
|
|
private ElementReference _modal;
|
|
|
|
|
|
|
|
|
|
private bool _isFirstRender = true;
|
|
|
|
|
|
|
|
|
|
#region ant-modal style
|
|
|
|
|
|
|
|
|
|
private string _modalStyle = null;
|
2020-06-02 14:25:44 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// ant-modal style
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
private string GetStyle()
|
|
|
|
|
{
|
2020-08-26 07:28:44 +08:00
|
|
|
|
if (_modalStyle == null)
|
|
|
|
|
{
|
|
|
|
|
var style = $"{Config.GetWidth()};";
|
|
|
|
|
|
|
|
|
|
if (Config.Draggable)
|
|
|
|
|
{
|
|
|
|
|
string left = $"margin: 0; padding-bottom:0;";
|
|
|
|
|
style += left;
|
|
|
|
|
}
|
|
|
|
|
_modalStyle = style;
|
|
|
|
|
}
|
2020-06-02 14:25:44 +08:00
|
|
|
|
|
|
|
|
|
if (!string.IsNullOrWhiteSpace(Style))
|
|
|
|
|
{
|
2020-08-26 07:28:44 +08:00
|
|
|
|
return _modalStyle + Style + ";";
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
2020-08-26 07:28:44 +08:00
|
|
|
|
return _modalStyle;
|
|
|
|
|
}
|
2020-06-02 14:25:44 +08:00
|
|
|
|
|
2020-08-26 07:28:44 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// if Modal is draggable, reset the position style similar with the first show
|
|
|
|
|
/// </summary>
|
|
|
|
|
internal async Task TryResetModalStyle()
|
|
|
|
|
{
|
|
|
|
|
if (Config.Draggable)
|
|
|
|
|
{
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.ResetModalPosition, _dialogHeader);
|
2020-08-26 07:28:44 +08:00
|
|
|
|
}
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-26 07:28:44 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
2020-06-02 14:25:44 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// append To body
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
private async Task AppendToContainer()
|
|
|
|
|
{
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.AddElementTo, _element, Config.GetContainer);
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#region mask and dialog click event
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// check is dialog click
|
|
|
|
|
/// </summary>
|
|
|
|
|
private bool _dialogMouseDown = false;
|
|
|
|
|
|
|
|
|
|
private void OnDialogMouseDown()
|
|
|
|
|
{
|
|
|
|
|
_dialogMouseDown = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task OnMaskMouseUp()
|
|
|
|
|
{
|
|
|
|
|
if (Config.MaskClosable && _dialogMouseDown)
|
|
|
|
|
{
|
|
|
|
|
await Task.Delay(50);
|
|
|
|
|
_dialogMouseDown = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task OnMaskClick(MouseEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (Config.MaskClosable
|
|
|
|
|
&& !_dialogMouseDown)
|
|
|
|
|
{
|
|
|
|
|
await CloseAsync();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#region keyboard control
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// TAB keyboard control
|
|
|
|
|
/// </summary>
|
|
|
|
|
private readonly string _sentinelStart = IdPrefix + Guid.NewGuid().ToString();
|
|
|
|
|
private readonly string _sentinelEnd = IdPrefix + Guid.NewGuid().ToString();
|
|
|
|
|
|
2020-08-26 07:28:44 +08:00
|
|
|
|
public string SentinelStart => _sentinelStart;
|
2020-06-02 14:25:44 +08:00
|
|
|
|
|
|
|
|
|
private async Task OnKeyDown(KeyboardEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (Config.Keyboard && e.Key == "Escape")
|
|
|
|
|
{
|
|
|
|
|
await CloseAsync();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (Visible)
|
|
|
|
|
{
|
|
|
|
|
if (e.Key == "Tab")
|
|
|
|
|
{
|
2020-09-09 00:52:14 +08:00
|
|
|
|
var activeElement = await JsInvokeAsync<string>(JSInteropConstants.GetActiveElement, _sentinelEnd);
|
2020-06-02 14:25:44 +08:00
|
|
|
|
if (e.ShiftKey)
|
|
|
|
|
{
|
|
|
|
|
if (activeElement == _sentinelStart)
|
|
|
|
|
{
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.Focus, "#" + _sentinelEnd);
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (activeElement == _sentinelEnd)
|
|
|
|
|
{
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.Focus, "#" + _sentinelStart);
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private async Task OnCloserClick(MouseEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
await CloseAsync();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private async Task CloseAsync()
|
|
|
|
|
{
|
|
|
|
|
if (_hasDestroy)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
2020-06-23 15:19:44 +08:00
|
|
|
|
if (Config.OnCancel != null)
|
2020-06-02 14:25:44 +08:00
|
|
|
|
{
|
2020-06-23 15:19:44 +08:00
|
|
|
|
await Config.OnCancel.Invoke(null);
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#region control show and hide class name and style
|
|
|
|
|
|
2020-09-19 09:15:16 +08:00
|
|
|
|
private async Task Show()
|
2020-06-02 14:25:44 +08:00
|
|
|
|
{
|
|
|
|
|
if (_hasShow)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Visible)
|
|
|
|
|
{
|
2020-08-26 07:28:44 +08:00
|
|
|
|
if (Config.Draggable)
|
|
|
|
|
{
|
|
|
|
|
_wrapStyle = "display:flex;justify-content: center;";
|
|
|
|
|
if (Config.Centered)
|
|
|
|
|
{
|
|
|
|
|
_wrapStyle += "align-items: center;";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_wrapStyle += "align-items: flex-start;";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_wrapStyle = "";
|
|
|
|
|
}
|
2020-06-02 14:25:44 +08:00
|
|
|
|
_maskHideClsName = "";
|
|
|
|
|
_maskAnimation = ModalAnimation.MaskEnter;
|
|
|
|
|
_modalAnimation = ModalAnimation.ModalEnter;
|
|
|
|
|
|
2020-09-19 09:15:16 +08:00
|
|
|
|
// wait for animation, "antZoomIn" animation take 0.3s
|
|
|
|
|
// see: @animation-duration-slow: 0.3s;
|
|
|
|
|
await Task.Delay(300);
|
|
|
|
|
|
2020-06-02 14:25:44 +08:00
|
|
|
|
_hasShow = true;
|
2020-09-19 09:15:16 +08:00
|
|
|
|
|
|
|
|
|
StateHasChanged();
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async Task Hide()
|
|
|
|
|
{
|
|
|
|
|
if (!_hasShow)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!Visible)
|
|
|
|
|
{
|
|
|
|
|
_maskAnimation = ModalAnimation.MaskLeave;
|
|
|
|
|
_modalAnimation = ModalAnimation.ModalLeave;
|
|
|
|
|
await Task.Delay(200);
|
|
|
|
|
_wrapStyle = "display: none;";
|
|
|
|
|
_maskHideClsName = "ant-modal-mask-hidden";
|
|
|
|
|
_hasShow = false;
|
|
|
|
|
StateHasChanged();
|
2020-06-23 15:19:44 +08:00
|
|
|
|
if (Config.OnClosed != null)
|
2020-06-02 14:25:44 +08:00
|
|
|
|
{
|
2020-06-23 15:19:44 +08:00
|
|
|
|
await Config.OnClosed.Invoke();
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-19 09:15:16 +08:00
|
|
|
|
public bool IsShow()
|
|
|
|
|
{
|
|
|
|
|
return _hasShow;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-02 14:25:44 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
private string GetMaskClsName()
|
|
|
|
|
{
|
|
|
|
|
string clsName = _maskHideClsName;
|
|
|
|
|
clsName += _maskAnimation;
|
|
|
|
|
return clsName;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetModalClsName()
|
|
|
|
|
{
|
|
|
|
|
string clsName = Config.ClassName;
|
|
|
|
|
|
|
|
|
|
return clsName + _modalAnimation;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#region override
|
|
|
|
|
|
|
|
|
|
protected override async Task OnParametersSetAsync()
|
|
|
|
|
{
|
|
|
|
|
//Reduce one rendering when showing and not destroyed
|
|
|
|
|
if (Visible)
|
|
|
|
|
{
|
|
|
|
|
if (!_hasDestroy)
|
|
|
|
|
{
|
2020-09-19 09:15:16 +08:00
|
|
|
|
await Show();
|
2020-06-02 14:25:44 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_wrapStyle = "display: none;";
|
|
|
|
|
_maskHideClsName = "ant-modal-mask-hidden";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
await Hide();
|
|
|
|
|
}
|
|
|
|
|
await base.OnParametersSetAsync();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
protected override async Task OnAfterRenderAsync(bool isFirst)
|
|
|
|
|
{
|
2020-08-26 07:28:44 +08:00
|
|
|
|
_isFirstRender = isFirst;
|
|
|
|
|
|
2020-06-02 14:25:44 +08:00
|
|
|
|
if (Visible)
|
|
|
|
|
{
|
|
|
|
|
if (_hasDestroy)
|
|
|
|
|
{
|
|
|
|
|
await AppendToContainer();
|
|
|
|
|
_hasDestroy = false;
|
|
|
|
|
Show();
|
|
|
|
|
StateHasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-04 17:03:13 +08:00
|
|
|
|
if (!_disableBodyScroll)
|
|
|
|
|
{
|
|
|
|
|
_disableBodyScroll = true;
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.DisableBodyScroll);
|
2020-06-04 17:03:13 +08:00
|
|
|
|
}
|
2020-08-26 07:28:44 +08:00
|
|
|
|
|
|
|
|
|
if (Config.Draggable && !_doDragable)
|
|
|
|
|
{
|
|
|
|
|
_doDragable = true;
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.EnableDraggable, _dialogHeader, _modal, Config.DragInViewport);
|
2020-08-26 07:28:44 +08:00
|
|
|
|
}
|
2020-06-04 17:03:13 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (_disableBodyScroll)
|
|
|
|
|
{
|
|
|
|
|
_disableBodyScroll = false;
|
|
|
|
|
await Task.Delay(250);
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.EnableBodyScroll);
|
2020-06-04 17:03:13 +08:00
|
|
|
|
}
|
2020-08-26 07:28:44 +08:00
|
|
|
|
|
|
|
|
|
if (Config.Draggable && _doDragable)
|
|
|
|
|
{
|
|
|
|
|
_doDragable = false;
|
2020-09-09 00:52:14 +08:00
|
|
|
|
await JsInvokeAsync(JSInteropConstants.DisableDraggable, _dialogHeader);
|
2020-08-26 07:28:44 +08:00
|
|
|
|
}
|
2020-06-04 17:03:13 +08:00
|
|
|
|
}
|
2020-06-02 14:25:44 +08:00
|
|
|
|
await base.OnAfterRenderAsync(isFirst);
|
|
|
|
|
}
|
|
|
|
|
#endregion
|
|
|
|
|
}
|
|
|
|
|
}
|