login service by localstorg
This commit is contained in:
@@ -40,7 +40,17 @@ namespace Back.Common
|
|||||||
PersianCalendar PersianCal = new PersianCalendar();
|
PersianCalendar PersianCal = new PersianCalendar();
|
||||||
return PersianCal.GetYear(date).ToString("0000") +
|
return PersianCal.GetYear(date).ToString("0000") +
|
||||||
PersianCal.GetMonth(date).ToString("00") +
|
PersianCal.GetMonth(date).ToString("00") +
|
||||||
PersianCal.GetDayOfMonth(date).ToString("00");
|
PersianCal.GetDayOfMonth(date).ToString("00")
|
||||||
|
;
|
||||||
|
}
|
||||||
|
public static string ConvertMiladiToShamsiByTime(this DateTime date)
|
||||||
|
{
|
||||||
|
PersianCalendar PersianCal = new PersianCalendar();
|
||||||
|
return PersianCal.GetYear(date).ToString("0000") +
|
||||||
|
PersianCal.GetMonth(date).ToString("00") +
|
||||||
|
PersianCal.GetDayOfMonth(date).ToString("00") +
|
||||||
|
date.Hour.ToString("00") + date.Minute.ToString("00") + date.Second.ToString("00")
|
||||||
|
;
|
||||||
}
|
}
|
||||||
public static string ShamciToFormatShamci(this string str)
|
public static string ShamciToFormatShamci(this string str)
|
||||||
{
|
{
|
||||||
@@ -52,6 +62,12 @@ namespace Back.Common
|
|||||||
PersianCalendar p = new PersianCalendar();
|
PersianCalendar p = new PersianCalendar();
|
||||||
return p.ToDateTime(Convert.ToInt32(value.Substring(0, 4)), Convert.ToInt32(value.Substring(4, 2)), Convert.ToInt32(value.Substring(6, 2)), 0, 0, 0, 0);
|
return p.ToDateTime(Convert.ToInt32(value.Substring(0, 4)), Convert.ToInt32(value.Substring(4, 2)), Convert.ToInt32(value.Substring(6, 2)), 0, 0, 0, 0);
|
||||||
}
|
}
|
||||||
|
public static DateTime ToMiladiByTime(this string value)
|
||||||
|
{
|
||||||
|
PersianCalendar p = new PersianCalendar();
|
||||||
|
return p.ToDateTime(Convert.ToInt32(value.Substring(0, 4)), Convert.ToInt32(value.Substring(4, 2)), Convert.ToInt32(value.Substring(6, 2)), Convert.ToInt32(value.Substring(8, 2)), Convert.ToInt32(value.Substring(10, 2)), Convert.ToInt32(value.Substring(12, 2)), 0);
|
||||||
|
|
||||||
|
}
|
||||||
public static async Task<PagingDto<T>> Paging<T>(this IQueryable<T> values, int pageId, int take)
|
public static async Task<PagingDto<T>> Paging<T>(this IQueryable<T> values, int pageId, int take)
|
||||||
{
|
{
|
||||||
if (/*values.Count()<1000 && */pageId == 0 && take == 0)
|
if (/*values.Count()<1000 && */pageId == 0 && take == 0)
|
||||||
|
@@ -26,14 +26,20 @@ namespace Back.Controllers
|
|||||||
if (result != null) return Ok(result);
|
if (result != null) return Ok(result);
|
||||||
else return NotFound("کاربری با این مشخصات یافت نشد");
|
else return NotFound("کاربری با این مشخصات یافت نشد");
|
||||||
}
|
}
|
||||||
[HttpGet("test")]
|
[HttpGet("CheckAuthenticate")]
|
||||||
|
public async Task<ActionResult<UserAuthenticationDTO>> CheckAuthenticate()
|
||||||
public async Task<ActionResult> test()
|
|
||||||
{
|
{
|
||||||
return Ok();
|
// var accessToken = Request.Headers["Authorization"].ToString().Split(' ')[1];
|
||||||
|
|
||||||
|
|
||||||
|
var claim = HttpContext.User.Claims.First(c => c.Type == "UserID");
|
||||||
|
var UserID = claim.Value;
|
||||||
|
var result = await _servUser.UserAuthentication(UserID,newtoken:false);
|
||||||
|
return Ok(result);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -42,7 +42,7 @@ namespace Back.Services
|
|||||||
// .ThenInclude(ti => ti.CalculationType)
|
// .ThenInclude(ti => ti.CalculationType)
|
||||||
.FirstOrDefaultAsync();
|
.FirstOrDefaultAsync();
|
||||||
}
|
}
|
||||||
public async Task<UserAuthenticationDTO?> UserAuthentication(string UserNameORUserID, string Password="")
|
public async Task<UserAuthenticationDTO?> UserAuthentication(string UserNameORUserID, string Password="",bool newtoken=true)
|
||||||
{
|
{
|
||||||
UserAuthenticationDTO ret = new UserAuthenticationDTO();
|
UserAuthenticationDTO ret = new UserAuthenticationDTO();
|
||||||
User? user = null;
|
User? user = null;
|
||||||
@@ -54,7 +54,10 @@ namespace Back.Services
|
|||||||
|
|
||||||
if (user == null)
|
if (user == null)
|
||||||
return null;
|
return null;
|
||||||
ret.Token =await CerateToken(user.ID, user.Username);
|
|
||||||
|
string Jwt_Lifetime_Minutes = await GetJwt_Lifetime_Minutes();
|
||||||
|
|
||||||
|
ret.Token =newtoken ? await CerateToken(user.ID, user.Username, Jwt_Lifetime_Minutes) : user.Token;
|
||||||
ret.FullName = user.Fullname;
|
ret.FullName = user.Fullname;
|
||||||
ret.Photo = user.Photo==null ? null : Convert.ToBase64String(user.Photo);
|
ret.Photo = user.Photo==null ? null : Convert.ToBase64String(user.Photo);
|
||||||
//foreach (var rol in user.RolUsers)
|
//foreach (var rol in user.RolUsers)
|
||||||
@@ -127,6 +130,9 @@ namespace Back.Services
|
|||||||
Logo = user.RolUsers.First().Company.Logo == null ? null : Convert.ToBase64String(user.RolUsers.First().Company.Logo)
|
Logo = user.RolUsers.First().Company.Logo == null ? null : Convert.ToBase64String(user.RolUsers.First().Company.Logo)
|
||||||
|
|
||||||
};
|
};
|
||||||
|
var dt = newtoken ? DateTime.Now : user.DateLastLogin.ToMiladiByTime();
|
||||||
|
ret.enterDate= dt;
|
||||||
|
ret.exitDate= dt.AddMinutes(Convert.ToInt32(Jwt_Lifetime_Minutes));
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
public async Task<User> AddUser(User item)
|
public async Task<User> AddUser(User item)
|
||||||
@@ -154,7 +160,10 @@ namespace Back.Services
|
|||||||
}
|
}
|
||||||
public async Task<User?> GetUserByUserID(int UserID)
|
public async Task<User?> GetUserByUserID(int UserID)
|
||||||
{
|
{
|
||||||
return await _RepoUser.Get(w => w.ID == UserID).FirstOrDefaultAsync();
|
return await _RepoUser.Get(w => w.ID == UserID)
|
||||||
|
.Include(ti => ti.RolUsers)
|
||||||
|
.ThenInclude(ti => ti.Company)
|
||||||
|
.FirstOrDefaultAsync();
|
||||||
}
|
}
|
||||||
public async Task SetTokenAndDateLogininDB(int UserID,string Token)
|
public async Task SetTokenAndDateLogininDB(int UserID,string Token)
|
||||||
{
|
{
|
||||||
@@ -162,7 +171,7 @@ namespace Back.Services
|
|||||||
if (user != null)
|
if (user != null)
|
||||||
{
|
{
|
||||||
user.Token = Token;
|
user.Token = Token;
|
||||||
user.DateLastLogin=DateTime.Now.ConvertMiladiToShamsi();
|
user.DateLastLogin=DateTime.Now.ConvertMiladiToShamsiByTime();
|
||||||
await _RepoUser.UpdateAsync(user);
|
await _RepoUser.UpdateAsync(user);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -229,9 +238,9 @@ namespace Back.Services
|
|||||||
return await _RepoUser.UpdateByObjAsync(user);
|
return await _RepoUser.UpdateByObjAsync(user);
|
||||||
}
|
}
|
||||||
//--------internal
|
//--------internal
|
||||||
private async Task<string> CerateToken(int UserId, string UserName)
|
private async Task<string> GetJwt_Lifetime_Minutes()
|
||||||
{
|
{
|
||||||
string Jwt_Lifetime_Minutes = "";
|
string Jwt_Lifetime_Minutes = "60";
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
Jwt_Lifetime_Minutes = _configuration["Fixedvalues:Jwt_Lifetime_Minutes"].ToString();
|
Jwt_Lifetime_Minutes = _configuration["Fixedvalues:Jwt_Lifetime_Minutes"].ToString();
|
||||||
@@ -254,6 +263,10 @@ namespace Back.Services
|
|||||||
//To DO
|
//To DO
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return Jwt_Lifetime_Minutes;
|
||||||
|
}
|
||||||
|
private async Task<string> CerateToken(int UserId, string UserName,string Jwt_Lifetime_Minutes)
|
||||||
|
{
|
||||||
#region CreateToken
|
#region CreateToken
|
||||||
var securityKey = new SymmetricSecurityKey(
|
var securityKey = new SymmetricSecurityKey(
|
||||||
Encoding.ASCII.GetBytes(Fixedvalues.SecretForKey)
|
Encoding.ASCII.GetBytes(Fixedvalues.SecretForKey)
|
||||||
|
@@ -7,6 +7,8 @@ namespace Shared.DTOs
|
|||||||
|
|
||||||
public string FullName { get; set; }
|
public string FullName { get; set; }
|
||||||
public string Token { get; set; }
|
public string Token { get; set; }
|
||||||
|
public DateTime enterDate { get; set; }
|
||||||
|
public DateTime exitDate { get; set; }
|
||||||
public string Photo { get; set; }
|
public string Photo { get; set; }
|
||||||
public CompanyAuthenticationDTO Company { get; set; }
|
public CompanyAuthenticationDTO Company { get; set; }
|
||||||
|
|
||||||
|
9
TaxPayerFull/FixedValues.cs
Normal file
9
TaxPayerFull/FixedValues.cs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
using Shared.DTOs;
|
||||||
|
|
||||||
|
namespace Front
|
||||||
|
{
|
||||||
|
public static class Fixedvalues
|
||||||
|
{
|
||||||
|
public static UserAuthenticationDTO Userinfo=new UserAuthenticationDTO();
|
||||||
|
}
|
||||||
|
}
|
@@ -1,8 +1,9 @@
|
|||||||
@page "/Register"
|
@page "/Register"
|
||||||
|
@using Front.Services
|
||||||
@using Shared.DTOs
|
@using Shared.DTOs
|
||||||
@inject HttpClient _hc
|
@inject HttpClient _hc
|
||||||
@inject NavigationManager nav
|
@inject NavigationManager nav
|
||||||
@inject UserAuthenticationDTO userinfo
|
@inject localService localserv;
|
||||||
<PageTitle>ثبت نام</PageTitle>
|
<PageTitle>ثبت نام</PageTitle>
|
||||||
<main>
|
<main>
|
||||||
|
|
||||||
@@ -89,11 +90,13 @@
|
|||||||
|
|
||||||
protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
{
|
{
|
||||||
if (userinfo!=null)
|
|
||||||
nav.NavigateTo("/");
|
|
||||||
|
|
||||||
editContext = new EditContext(model);
|
editContext = new EditContext(model);
|
||||||
messageStore = new(editContext);
|
messageStore = new(editContext);
|
||||||
|
|
||||||
|
if (await localserv.OnlineUser())
|
||||||
|
nav.NavigateTo("/");
|
||||||
|
|
||||||
await base.OnInitializedAsync();
|
await base.OnInitializedAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,9 +1,11 @@
|
|||||||
@page "/Sign-in"
|
@page "/Sign-in"
|
||||||
|
@using Front.Services
|
||||||
@using Shared.DTOs
|
@using Shared.DTOs
|
||||||
|
@inject ILocalStorageService Storage;
|
||||||
|
@inject UserAuthenticationDTO userinfo
|
||||||
|
@inject localService localserv;
|
||||||
@inject HttpClient _hc
|
@inject HttpClient _hc
|
||||||
@inject NavigationManager nav
|
@inject NavigationManager nav
|
||||||
@inject UserAuthenticationDTO userinfo
|
|
||||||
<PageTitle>ورود</PageTitle>
|
<PageTitle>ورود</PageTitle>
|
||||||
<main>
|
<main>
|
||||||
|
|
||||||
@@ -133,18 +135,20 @@
|
|||||||
@code {
|
@code {
|
||||||
[SupplyParameterFromForm]
|
[SupplyParameterFromForm]
|
||||||
public Authentication? Model { get; set; }
|
public Authentication? Model { get; set; }
|
||||||
protected override void OnInitialized() => Model ??= new();
|
|
||||||
// alert
|
// alert
|
||||||
AlertColor alertColor = AlertColor.Primary;
|
AlertColor alertColor = AlertColor.Primary;
|
||||||
IconName alertIconName = IconName.CheckCircleFill;
|
IconName alertIconName = IconName.CheckCircleFill;
|
||||||
bool Hidealert = true;
|
bool Hidealert = true;
|
||||||
string alertMessage = "";
|
string alertMessage = "";
|
||||||
// protected override async Task OnInitializedAsync()
|
protected override async Task OnInitializedAsync()
|
||||||
// {
|
{
|
||||||
// var t1 = userinfo;
|
|
||||||
// var t2 = _hc;
|
if (await localserv.OnlineUser())
|
||||||
// await base.OnInitializedAsync();
|
nav.NavigateTo("/");
|
||||||
// }
|
|
||||||
|
Model ??= new();
|
||||||
|
await base.OnInitializedAsync();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@functions {
|
@functions {
|
||||||
private void ShowDangerAlert(string msg)
|
private void ShowDangerAlert(string msg)
|
||||||
@@ -154,7 +158,7 @@
|
|||||||
alertIconName = IconName.ExclamationTriangleFill;
|
alertIconName = IconName.ExclamationTriangleFill;
|
||||||
alertMessage = msg;
|
alertMessage = msg;
|
||||||
}
|
}
|
||||||
private async Task EndForm() =>nav.NavigateTo("/");
|
private async Task EndForm() => nav.NavigateTo("/");
|
||||||
|
|
||||||
|
|
||||||
private async Task OnLoginClick()
|
private async Task OnLoginClick()
|
||||||
@@ -162,13 +166,27 @@
|
|||||||
var request = await _hc.PostAsJsonAsync("User/authenticate", Model);
|
var request = await _hc.PostAsJsonAsync("User/authenticate", Model);
|
||||||
if (request.IsSuccessStatusCode)
|
if (request.IsSuccessStatusCode)
|
||||||
{
|
{
|
||||||
userinfo = await request.Content.ReadFromJsonAsync<UserAuthenticationDTO>();
|
var userinfomodel = await request.Content.ReadFromJsonAsync<UserAuthenticationDTO>();
|
||||||
|
|
||||||
|
_hc.DefaultRequestHeaders.Clear();
|
||||||
|
|
||||||
_hc.DefaultRequestHeaders.Add("Authorization",
|
_hc.DefaultRequestHeaders.Add("Authorization",
|
||||||
$"Bearer {userinfo?.Token}");
|
$"Bearer {userinfomodel?.Token}");
|
||||||
|
|
||||||
|
if (Model.Remember)
|
||||||
|
Storage.SetItem("token", userinfomodel?.Token);
|
||||||
|
|
||||||
|
userinfo.Token = userinfomodel?.Token;
|
||||||
|
userinfo.Company = userinfomodel?.Company;
|
||||||
|
userinfo.FullName = userinfomodel.FullName;
|
||||||
|
userinfo.Photo = userinfomodel.Photo;
|
||||||
|
userinfo.exitDate = userinfomodel.exitDate;
|
||||||
|
userinfo.enterDate = userinfomodel.enterDate;
|
||||||
|
|
||||||
|
|
||||||
|
nav.NavigateTo("/");
|
||||||
}
|
}
|
||||||
else if (request.StatusCode==System.Net.HttpStatusCode.NotFound)
|
else if (request.StatusCode == System.Net.HttpStatusCode.NotFound)
|
||||||
ShowDangerAlert("کاربری با این مشخصات یافت نشد");
|
ShowDangerAlert("کاربری با این مشخصات یافت نشد");
|
||||||
|
|
||||||
else ShowDangerAlert("خطای سیستمی");
|
else ShowDangerAlert("خطای سیستمی");
|
||||||
|
@@ -3,12 +3,17 @@ using Microsoft.AspNetCore.Components.WebAssembly.Hosting;
|
|||||||
using Front;
|
using Front;
|
||||||
using System.Globalization;
|
using System.Globalization;
|
||||||
using Shared.DTOs;
|
using Shared.DTOs;
|
||||||
|
using Front.Services;
|
||||||
|
using System.ComponentModel.Design;
|
||||||
|
|
||||||
var builder = WebAssemblyHostBuilder.CreateDefault(args);
|
var builder = WebAssemblyHostBuilder.CreateDefault(args);
|
||||||
builder.RootComponents.Add<App>("#app");
|
builder.RootComponents.Add<App>("#app");
|
||||||
builder.RootComponents.Add<HeadOutlet>("head::after");
|
builder.RootComponents.Add<HeadOutlet>("head::after");
|
||||||
builder.Services.AddBlazorBootstrap();
|
builder.Services.AddBlazorBootstrap();
|
||||||
builder.Services.AddScoped<UserAuthenticationDTO>();
|
|
||||||
|
builder.Services.AddScoped<ILocalStorageService, LocalStorageService>();
|
||||||
|
builder.Services.AddScoped<localService>();
|
||||||
|
builder.Services.AddScoped(sp => new UserAuthenticationDTO());
|
||||||
|
|
||||||
|
|
||||||
builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri("https://localhost:7075/api/") });
|
builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri("https://localhost:7075/api/") });
|
||||||
@@ -17,5 +22,14 @@ builder.Services.AddScoped(sp => new HttpClient { BaseAddress = new Uri("https:/
|
|||||||
|
|
||||||
CultureInfo.DefaultThreadCurrentCulture = new CultureInfo("fa-Ir");
|
CultureInfo.DefaultThreadCurrentCulture = new CultureInfo("fa-Ir");
|
||||||
|
|
||||||
await builder.Build().RunAsync();
|
var host = builder.Build();
|
||||||
|
|
||||||
|
|
||||||
|
await new MidLevel(
|
||||||
|
host.Services.GetRequiredService<ILocalStorageService>(),
|
||||||
|
host.Services.GetRequiredService<UserAuthenticationDTO>(),
|
||||||
|
host.Services.GetRequiredService<HttpClient>()
|
||||||
|
).InitAsync();
|
||||||
|
|
||||||
|
await host.RunAsync();
|
||||||
|
|
||||||
|
44
TaxPayerFull/Services/ILocalStorageService.cs
Normal file
44
TaxPayerFull/Services/ILocalStorageService.cs
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
using Microsoft.JSInterop;
|
||||||
|
using System.Text.Json;
|
||||||
|
|
||||||
|
namespace Front.Services
|
||||||
|
{
|
||||||
|
|
||||||
|
public interface ILocalStorageService
|
||||||
|
{
|
||||||
|
Task<T> GetItem<T>(string key);
|
||||||
|
Task SetItem<T>(string key, T value);
|
||||||
|
Task RemoveItem(string key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public class LocalStorageService : ILocalStorageService
|
||||||
|
{
|
||||||
|
private IJSRuntime _jsRuntime;
|
||||||
|
|
||||||
|
public LocalStorageService(IJSRuntime jsRuntime)
|
||||||
|
{
|
||||||
|
_jsRuntime = jsRuntime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<T> GetItem<T>(string key)
|
||||||
|
{
|
||||||
|
var json = await _jsRuntime.InvokeAsync<string>("localStorage.getItem", key);
|
||||||
|
|
||||||
|
if (json == null)
|
||||||
|
return default;
|
||||||
|
|
||||||
|
return JsonSerializer.Deserialize<T>(json);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task SetItem<T>(string key, T value)
|
||||||
|
{
|
||||||
|
await _jsRuntime.InvokeVoidAsync("localStorage.setItem", key, JsonSerializer.Serialize(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task RemoveItem(string key)
|
||||||
|
{
|
||||||
|
await _jsRuntime.InvokeVoidAsync("localStorage.removeItem", key);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
39
TaxPayerFull/Services/MidLevel.cs
Normal file
39
TaxPayerFull/Services/MidLevel.cs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
using Shared.DTOs;
|
||||||
|
using System.Net.Http.Json;
|
||||||
|
|
||||||
|
namespace Front.Services
|
||||||
|
{
|
||||||
|
public class MidLevel
|
||||||
|
{
|
||||||
|
private readonly ILocalStorageService _localStorage;
|
||||||
|
private readonly UserAuthenticationDTO _user;
|
||||||
|
private readonly HttpClient _httpClient;
|
||||||
|
public MidLevel(ILocalStorageService localStorage, UserAuthenticationDTO user, HttpClient httpClient)
|
||||||
|
{
|
||||||
|
_localStorage = localStorage;
|
||||||
|
_user = user;
|
||||||
|
_httpClient = httpClient;
|
||||||
|
}
|
||||||
|
public async Task InitAsync()
|
||||||
|
{
|
||||||
|
var token = await _localStorage.GetItem<string>("token");
|
||||||
|
if (!string.IsNullOrEmpty(token))
|
||||||
|
{
|
||||||
|
_httpClient.DefaultRequestHeaders.Add("Authorization", $"Bearer {token}");
|
||||||
|
|
||||||
|
var request = await _httpClient.GetAsync("User/CheckAuthenticate");
|
||||||
|
if (request.IsSuccessStatusCode)
|
||||||
|
{
|
||||||
|
var userinfomodel = await request.Content.ReadFromJsonAsync<UserAuthenticationDTO>();
|
||||||
|
_user.Token = token;
|
||||||
|
_user.Company = userinfomodel?.Company;
|
||||||
|
_user.FullName = userinfomodel.FullName;
|
||||||
|
_user.Photo = userinfomodel.Photo;
|
||||||
|
_user.exitDate = userinfomodel.exitDate;
|
||||||
|
_user.enterDate = userinfomodel.enterDate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
19
TaxPayerFull/Services/localService.cs
Normal file
19
TaxPayerFull/Services/localService.cs
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
using Shared.DTOs;
|
||||||
|
|
||||||
|
namespace Front.Services
|
||||||
|
{
|
||||||
|
public class localService
|
||||||
|
{
|
||||||
|
private readonly UserAuthenticationDTO _user;
|
||||||
|
public localService(UserAuthenticationDTO user)
|
||||||
|
{
|
||||||
|
_user = user;
|
||||||
|
}
|
||||||
|
public async Task<bool> OnlineUser()
|
||||||
|
{
|
||||||
|
if (_user != null && !string.IsNullOrEmpty(_user.Token) && _user.exitDate.AddMinutes(-5) > DateTime.Now)
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user